├── .babelrc ├── .gitignore ├── .npmignore ├── LICENSE ├── README.md ├── example ├── package.json ├── public │ ├── example-300.gif │ ├── example-600.gif │ ├── example.gif │ ├── favicon.ico │ ├── index.html │ ├── manifest.json │ └── mobilenet_v1_0.25_224.json ├── src │ ├── containers │ │ ├── App │ │ │ ├── index.tsx │ │ │ └── styles.scss │ │ └── Search │ │ │ ├── Topic.tsx │ │ │ ├── getImages.ts │ │ │ ├── index.tsx │ │ │ ├── loading.gif │ │ │ └── styles.scss │ ├── index.tsx │ └── mlClassifierUI.ts ├── tsconfig.json ├── tsconfig.prod.json ├── tsconfig.test.json ├── tslint.json └── yarn.lock ├── issue_template.md ├── package.json ├── rollup.config.js ├── rollup.config.js.umd ├── src ├── App │ ├── index.tsx │ └── styles.scss ├── Dropzone │ ├── getFile.ts │ ├── index.tsx │ ├── styles.scss │ ├── transformFiles.ts │ └── traverseFileTree.ts ├── Loading │ ├── index.tsx │ ├── loading.gif │ └── loading.ts ├── Model │ ├── Evaluator │ │ ├── Predictions │ │ │ ├── Prediction.tsx │ │ │ ├── index.tsx │ │ │ └── styles.scss │ │ └── index.tsx │ ├── Metrics │ │ ├── Info.tsx │ │ ├── Logs │ │ │ ├── index.tsx │ │ │ └── styles.scss │ │ ├── index.tsx │ │ └── styles.scss │ ├── index.tsx │ └── styles.scss ├── Preview │ ├── index.tsx │ └── styles.scss ├── global.d.ts ├── index.ts ├── types.ts └── utils │ ├── __tests__ │ └── classNames.test.ts │ ├── classNames.ts │ ├── getFilesAsImages.ts │ └── nextFrame.ts ├── tsconfig.json ├── tslint.json └── yarn.lock /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | [ 4 | "@babel/env", { 5 | "targets": { 6 | "node": "current" 7 | } 8 | } 9 | ], 10 | "@babel/typescript" 11 | ], 12 | "plugins": [ 13 | "transform-class-properties" 14 | ] 15 | } 16 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | 3 | # Don't include generated files 4 | *.css 5 | *.js 6 | *.d.ts 7 | *.tgz 8 | dist 9 | example/build 10 | 11 | # Various caches, error logs and such 12 | yarn-error.log 13 | .rpt2_cache 14 | package-lock.json 15 | 16 | # Don't include stylesheet typings 17 | src/*.css.d.ts 18 | src/**/*.css.d.ts 19 | 20 | .DS_Store 21 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | src 2 | node_modules 3 | yarn-error.log 4 | .rpt2_cache 5 | tsconfig.json 6 | tslint.json 7 | yarn.lock 8 | rollup.config.js 9 | scripts 10 | package-lock.json 11 | example 12 | *.tgz 13 | .storybook 14 | stories 15 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018 Kevin Scott 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | 23 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # ML Classifier UI 2 | 3 | ML Classifier is a React front end for a machine learning engine for quickly training image classification models in your browser. Models can be saved with a single command, and the resulting models reused to make image classification predictions. 4 | 5 | This package is the UI front end for [`ml-classifier`](https://github.com/thekevinscott/ml-classifier). 6 | 7 | ## Walkthrough 8 | 9 | A walkthrough of the code can be found in the article [Image Classification in the Browser with Javascript](https://thekevinscott.com/image-classification-with-javascript/). 10 | 11 | ## Demo 12 | 13 | An interactive [demo can be found here](https://thekevinscott.github.io/ml-classifier-ui/). 14 | 15 | ![Demo](https://github.com/thekevinscott/ml-classifier-ui/raw/master/example/public/example.gif) 16 | *Screenshot of demo* 17 | 18 | ## Getting Started 19 | 20 | ### Installation 21 | 22 | `ml-classifier-ui` can be installed via `yarn` or `npm`: 23 | 24 | ``` 25 | yarn add ml-classifier-ui 26 | ``` 27 | 28 | or 29 | 30 | ``` 31 | npm install ml-classifier-ui 32 | ``` 33 | 34 | ### Quick Start (Code Sandbox) 35 | 36 | You can fork a live running version at [codesandbox.io](https://codesandbox.io/s/218po5mzxn). 37 | 38 | ### Quick Start (Running locally) 39 | 40 | Start by instantiating a new MLClassifierUI. 41 | 42 | ``` 43 | import React from 'react'; 44 | import ReactDOM from 'react-dom'; 45 | import MLClassifierUI from 'ml-classifier-ui'; 46 | 47 | ReactDOM.render(, document.getElementById('root')); 48 | ``` 49 | 50 | ## API Documentation 51 | 52 | `MLClassifierUI` accepts a number of parameters: 53 | 54 | * **getMLClassifier** (`Function`) *Optional* - A callback that returns an instance of the underlying `ml-classifier` object. Call this if you want to programmatically call methods like `addData`, `train`, and `predict`. For more information on `ml-classifier`'s API methods [refer to it's documentation](https://github.com/thekevinscott/ml-classifier#api-documentation). 55 | * **methodParams** (`Object`) *Optional* - A set of parameters that will be passed in calls to `ml-classifier`'s methods. See below for more information. 56 | * **uploadFormat** (`string`) *Optional* - A string denoting what type of upload format to accept. Formats can be `flat` or `nested`. See below note for more information on that. If omitted, all formats are accepted. 57 | * **imageFormats** (`string[]`) *Optional* - An array of file extensions to accept. By default, all valid images are accepted. Images are transformed via the native `Image` tag in the browser, so if the browser can display the image, it'll be processed. 58 | * **showDownload** (`boolean`) *Optional* - A flag denoting whether to show a download button or not. Defaults to true. 59 | 60 | `MLClassifierUI` also accepts a number of callbacks that are called on the beginnings and ends of `ml-classifier` functions. [You can view a list of those here](https://github.com/thekevinscott/ml-classifier#parameters). 61 | 62 | ### `getMLClassifier` 63 | 64 | `getMLClassifier` returns an instance of `ml-classifier` for programmatic access to the underlying methods. 65 | 66 | #### Example 67 | 68 | ``` 69 | { 71 | mlClassifier.addData(...); 72 | }} 73 | /> 74 | ``` 75 | 76 | ### `methodParams` 77 | 78 | `methodParams` can be used to pass method-specific parameters to `ml-classifier`. The key will be used to determine which method to pass parameters to. 79 | 80 | Accepted keys are `train`, `evaluate`, and `save`. Other keys will be ignored. 81 | 82 | #### Example 83 | 84 | ``` 85 | 97 | ``` 98 | 99 | ### `uploadFormat` 100 | 101 | `uploadFormat` corresponds to how uploaded images should be organized. There are two options: 102 | 103 | #### `nested` 104 | Expects images to be organized in folders matching the label. Only the immediate parent folder's name will be used as the label. For example: 105 | 106 | ``` 107 | - containing-folder/ 108 | - dogs/ 109 | - IMG-1.jpg 110 | - IMG-2.jpg 111 | - IMG-3.jpg 112 | - cats/ 113 | - IMG-1.jpg 114 | - IMG-2.jpg 115 | - IMG-3.jpg 116 | ``` 117 | 118 | Will product an array of three `dogs` labels and three `cats` labels. 119 | 120 | Nested folders will be searched recursively, but only immediate parent folders' names will be used. If an invalidly nested structure is found an error will be thrown. 121 | 122 | #### `flat` (*currently in development*) 123 | Expects files' names to be the label. Nested folders will be searched recursively (if the browser supports it) to build a flat array of files. 124 | 125 | ``` 126 | - folder/ 127 | - dog-1.jpg 128 | - dog-2.jpg 129 | - dog-3.jpg 130 | - cat-1.jpg 131 | - cat-2.jpg 132 | - cat-3.jpg 133 | ``` 134 | 135 | #### Example 136 | 137 | ``` 138 | 141 | ``` 142 | 143 | ### `imageFormats` (*currently in development*) 144 | 145 | `imageFormats` denotes the list of acceptable image formats for upload. Any images not matching the list of acceptable formats will be ignored. 146 | 147 | #### Example 148 | 149 | ``` 150 | 156 | ``` 157 | 158 | ## Contributing 159 | 160 | Contributions are welcome! 161 | 162 | You can run the local example with: 163 | 164 | ``` 165 | yarn watch 166 | ``` 167 | 168 | `ml-classifier-ui` is written in Typescript and React. 169 | 170 | ### Tests 171 | 172 | Tests are a work in progress. Currently, the test suite only consists of unit tests. Pull requests for additional tests are welcome! 173 | 174 | Run tests with: 175 | 176 | ``` 177 | yarn test 178 | ``` 179 | 180 | ## Author 181 | 182 | * [Kevin Scott](https://thekevinscott.com) 183 | 184 | ## License 185 | 186 | This project is licensed under the MIT License - see the LICENSE file for details 187 | 188 | ![](https://ga-beacon.appspot.com/UA-112845439-4/ml-classifier-ui/readme) 189 | -------------------------------------------------------------------------------- /example/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ml-classifier-ui-example", 3 | "version": "0.1.0", 4 | "private": true, 5 | "homepage": "https://thekevinscott.github.io/ml-classifier-ui/", 6 | "dependencies": { 7 | "autoprefixer": "7.1.6", 8 | "babel-jest": "^22.1.0", 9 | "babel-loader": "^7.1.2", 10 | "babel-plugin-named-asset-import": "^0.3.6", 11 | "babel-preset-env": "^1.7.0", 12 | "babel-preset-react": "^6.24.1", 13 | "babel-preset-react-app": "^9.1.2", 14 | "case-sensitive-paths-webpack-plugin": "2.1.1", 15 | "chalk": "1.1.3", 16 | "classnames": "^2.2.6", 17 | "css-loader": "0.28.7", 18 | "dotenv": "4.0.0", 19 | "dotenv-expand": "4.2.0", 20 | "eslint": "^7.4.0", 21 | "eslint-loader": "^4.0.2", 22 | "extract-text-webpack-plugin": "3.0.2", 23 | "file-loader": "0.11.2", 24 | "fork-ts-checker-webpack-plugin": "^0.2.8", 25 | "free-google-image-search": "^1.0.0", 26 | "fs-extra": "3.0.1", 27 | "html-webpack-plugin": "^4.3.0", 28 | "jest": "22.4.2", 29 | "mini-css-extract-plugin": "^0.9.0", 30 | "ml-classifier-ui": "file:../", 31 | "node-sass": "^4.9.0", 32 | "object-assign": "4.1.1", 33 | "optimize-css-assets-webpack-plugin": "^5.0.3", 34 | "pnp-webpack-plugin": "^1.6.4", 35 | "postcss-flexbugs-fixes": "3.2.0", 36 | "postcss-loader": "2.0.8", 37 | "postcss-normalize": "^9.0.0", 38 | "postcss-safe-parser": "^4.0.2", 39 | "promise": "8.0.1", 40 | "raf": "3.4.0", 41 | "react": "^16.4.1", 42 | "react-dev-utils": "^10.2.1", 43 | "react-dom": "^16.4.2", 44 | "react-dropzone": "^4.2.11", 45 | "react-progressbar.js": "^0.2.0", 46 | "resolve": "1.6.0", 47 | "resolve-url-loader": "^3.1.1", 48 | "sass-loader": "^7.0.3", 49 | "source-map-loader": "^0.2.1", 50 | "style-loader": "0.19.0", 51 | "sw-precache-webpack-plugin": "0.11.4", 52 | "terser-webpack-plugin": "^3.0.6", 53 | "ts-jest": "22.0.1", 54 | "ts-loader": "^2.3.7", 55 | "tsconfig-paths-webpack-plugin": "^2.0.0", 56 | "tslint": "^5.7.0", 57 | "tslint-config-prettier": "^1.10.0", 58 | "tslint-react": "^3.2.0", 59 | "uglifyjs-webpack-plugin": "^1.1.8", 60 | "url-loader": "0.6.2", 61 | "webpack": "4.43.0", 62 | "webpack-dev-server": ">=3.1.11", 63 | "webpack-manifest-plugin": "1.3.2", 64 | "whatwg-fetch": "2.0.3", 65 | "workbox-webpack-plugin": "^5.1.3", 66 | "yargs": "^12.0.1" 67 | }, 68 | "scripts": { 69 | "start": "node scripts/start.js", 70 | "build": "node scripts/build.js", 71 | "test": "node scripts/test.js --env=jsdom" 72 | }, 73 | "devDependencies": { 74 | "@types/classnames": "^2.2.4", 75 | "@types/jest": "^23.1.1", 76 | "@types/node": "^10.3.4", 77 | "@types/react": "^16.4.6", 78 | "@types/react-dom": "^16.0.6", 79 | "@types/react-dropzone": "^4.2.0", 80 | "typescript": "^2.9.2" 81 | }, 82 | "babel": { 83 | "presets": [ 84 | "react-app" 85 | ] 86 | }, 87 | "eslintConfig": { 88 | "extends": "react-app" 89 | }, 90 | "browserslist": { 91 | "production": [ 92 | ">0.2%", 93 | "not dead", 94 | "not op_mini all" 95 | ], 96 | "development": [ 97 | "last 1 chrome version", 98 | "last 1 firefox version", 99 | "last 1 safari version" 100 | ] 101 | } 102 | } 103 | -------------------------------------------------------------------------------- /example/public/example-300.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thekevinscott/ml-classifier-ui/f3c81d5ee6c3680bd1da0ee923307fc1f6eaf121/example/public/example-300.gif -------------------------------------------------------------------------------- /example/public/example-600.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thekevinscott/ml-classifier-ui/f3c81d5ee6c3680bd1da0ee923307fc1f6eaf121/example/public/example-600.gif -------------------------------------------------------------------------------- /example/public/example.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thekevinscott/ml-classifier-ui/f3c81d5ee6c3680bd1da0ee923307fc1f6eaf121/example/public/example.gif -------------------------------------------------------------------------------- /example/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thekevinscott/ml-classifier-ui/f3c81d5ee6c3680bd1da0ee923307fc1f6eaf121/example/public/favicon.ico -------------------------------------------------------------------------------- /example/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 11 | 12 | 13 | 22 | Demo of ml-classifier-ui 23 | 24 | 25 | 26 | 33 | 34 | 35 | 36 | 37 | 40 |
41 | 51 | 52 | 53 | -------------------------------------------------------------------------------- /example/public/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "React App", 3 | "name": "Create React App Sample", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | } 10 | ], 11 | "start_url": "./index.html", 12 | "display": "standalone", 13 | "theme_color": "#000000", 14 | "background_color": "#ffffff" 15 | } 16 | -------------------------------------------------------------------------------- /example/public/mobilenet_v1_0.25_224.json: -------------------------------------------------------------------------------- 1 | {"modelTopology": {"keras_version": "2.1.4", "model_config": {"class_name": "Model", "config": {"layers": [{"class_name": "InputLayer", "inbound_nodes": [], "config": {"dtype": "float32", "batch_input_shape": [null, 224, 224, 3], "name": "input_1", "sparse": false}, "name": "input_1"}, {"class_name": "Conv2D", "inbound_nodes": [[["input_1", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv1", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [2, 2], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 8, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv1"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv1", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv1_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv1_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv1_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv1_relu"}, "name": "conv1_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv1_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_1", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_1"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_1", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_1_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_1_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_1_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_1_relu"}, "name": "conv_dw_1_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_1_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_1", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 16, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_1"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_1", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_1_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_1_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_1_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_1_relu"}, "name": "conv_pw_1_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_1_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_2", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [2, 2], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_2"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_2", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_2_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_2_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_2_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_2_relu"}, "name": "conv_dw_2_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_2_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_2", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 32, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_2"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_2", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_2_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_2_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_2_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_2_relu"}, "name": "conv_pw_2_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_2_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_3", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_3"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_3", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_3_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_3_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_3_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_3_relu"}, "name": "conv_dw_3_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_3_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_3", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 32, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_3"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_3", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_3_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_3_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_3_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_3_relu"}, "name": "conv_pw_3_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_3_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_4", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [2, 2], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_4"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_4", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_4_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_4_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_4_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_4_relu"}, "name": "conv_dw_4_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_4_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_4", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 64, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_4"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_4", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_4_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_4_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_4_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_4_relu"}, "name": "conv_pw_4_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_4_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_5", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_5"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_5", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_5_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_5_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_5_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_5_relu"}, "name": "conv_dw_5_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_5_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_5", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 64, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_5"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_5", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_5_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_5_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_5_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_5_relu"}, "name": "conv_pw_5_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_5_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_6", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [2, 2], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_6"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_6", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_6_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_6_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_6_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_6_relu"}, "name": "conv_dw_6_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_6_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_6", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 128, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_6"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_6", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_6_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_6_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_6_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_6_relu"}, "name": "conv_pw_6_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_6_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_7", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_7"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_7", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_7_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_7_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_7_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_7_relu"}, "name": "conv_dw_7_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_7_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_7", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 128, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_7"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_7", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_7_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_7_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_7_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_7_relu"}, "name": "conv_pw_7_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_7_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_8", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_8"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_8", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_8_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_8_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_8_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_8_relu"}, "name": "conv_dw_8_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_8_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_8", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 128, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_8"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_8", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_8_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_8_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_8_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_8_relu"}, "name": "conv_pw_8_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_8_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_9", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_9"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_9", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_9_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_9_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_9_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_9_relu"}, "name": "conv_dw_9_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_9_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_9", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 128, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_9"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_9", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_9_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_9_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_9_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_9_relu"}, "name": "conv_pw_9_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_9_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_10", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_10"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_10", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_10_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_10_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_10_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_10_relu"}, "name": "conv_dw_10_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_10_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_10", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 128, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_10"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_10", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_10_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_10_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_10_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_10_relu"}, "name": "conv_pw_10_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_10_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_11", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_11"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_11", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_11_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_11_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_11_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_11_relu"}, "name": "conv_dw_11_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_11_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_11", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 128, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_11"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_11", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_11_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_11_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_11_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_11_relu"}, "name": "conv_pw_11_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_11_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_12", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [2, 2], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_12"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_12", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_12_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_12_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_12_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_12_relu"}, "name": "conv_dw_12_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_12_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_12", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 256, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_12"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_12", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_12_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_12_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_12_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_12_relu"}, "name": "conv_pw_12_relu"}, {"class_name": "DepthwiseConv2D", "inbound_nodes": [[["conv_pw_12_relu", 0, 0, {}]]], "config": {"padding": "same", "depth_multiplier": 1, "name": "conv_dw_13", "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "depthwise_constraint": null, "strides": [1, 1], "dilation_rate": [1, 1], "depthwise_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "bias_initializer": {"class_name": "Zeros", "config": {}}, "depthwise_regularizer": null, "use_bias": false, "activity_regularizer": null, "kernel_size": [3, 3]}, "name": "conv_dw_13"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_dw_13", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_dw_13_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_dw_13_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_dw_13_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_dw_13_relu"}, "name": "conv_dw_13_relu"}, {"class_name": "Conv2D", "inbound_nodes": [[["conv_dw_13_relu", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_pw_13", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 256, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": false, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_pw_13"}, {"class_name": "BatchNormalization", "inbound_nodes": [[["conv_pw_13", 0, 0, {}]]], "config": {"gamma_initializer": {"class_name": "Ones", "config": {}}, "moving_mean_initializer": {"class_name": "Zeros", "config": {}}, "name": "conv_pw_13_bn", "epsilon": 0.001, "trainable": true, "center": true, "moving_variance_initializer": {"class_name": "Ones", "config": {}}, "beta_initializer": {"class_name": "Zeros", "config": {}}, "scale": true, "gamma_regularizer": null, "gamma_constraint": null, "beta_constraint": null, "beta_regularizer": null, "momentum": 0.99, "axis": -1}, "name": "conv_pw_13_bn"}, {"class_name": "Activation", "inbound_nodes": [[["conv_pw_13_bn", 0, 0, {}]]], "config": {"activation": "relu6", "trainable": true, "name": "conv_pw_13_relu"}, "name": "conv_pw_13_relu"}, {"class_name": "GlobalAveragePooling2D", "inbound_nodes": [[["conv_pw_13_relu", 0, 0, {}]]], "config": {"trainable": true, "name": "global_average_pooling2d_1", "data_format": "channels_last"}, "name": "global_average_pooling2d_1"}, {"class_name": "Reshape", "inbound_nodes": [[["global_average_pooling2d_1", 0, 0, {}]]], "config": {"target_shape": [1, 1, 256], "trainable": true, "name": "reshape_1"}, "name": "reshape_1"}, {"class_name": "Dropout", "inbound_nodes": [[["reshape_1", 0, 0, {}]]], "config": {"rate": 0.001, "noise_shape": null, "trainable": true, "seed": null, "name": "dropout"}, "name": "dropout"}, {"class_name": "Conv2D", "inbound_nodes": [[["dropout", 0, 0, {}]]], "config": {"kernel_initializer": {"class_name": "VarianceScaling", "config": {"distribution": "uniform", "scale": 1.0, "seed": null, "mode": "fan_avg"}}, "name": "conv_preds", "kernel_constraint": null, "bias_regularizer": null, "bias_constraint": null, "activation": "linear", "trainable": true, "data_format": "channels_last", "padding": "same", "strides": [1, 1], "dilation_rate": [1, 1], "kernel_regularizer": null, "filters": 1000, "bias_initializer": {"class_name": "Zeros", "config": {}}, "use_bias": true, "activity_regularizer": null, "kernel_size": [1, 1]}, "name": "conv_preds"}, {"class_name": "Activation", "inbound_nodes": [[["conv_preds", 0, 0, {}]]], "config": {"activation": "softmax", "trainable": true, "name": "act_softmax"}, "name": "act_softmax"}, {"class_name": "Reshape", "inbound_nodes": [[["act_softmax", 0, 0, {}]]], "config": {"target_shape": [1000], "trainable": true, "name": "reshape_2"}, "name": "reshape_2"}], "input_layers": [["input_1", 0, 0]], "name": "mobilenet_0.25_224", "output_layers": [["reshape_2", 0, 0]]}}, "backend": "tensorflow"}, "weightsManifest": [{"paths": ["group1-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 3, 8], "name": "conv1/kernel"}]}, {"paths": ["group2-shard1of1"], "weights": [{"dtype": "float32", "shape": [8], "name": "conv1_bn/gamma"}, {"dtype": "float32", "shape": [8], "name": "conv1_bn/beta"}, {"dtype": "float32", "shape": [8], "name": "conv1_bn/moving_mean"}, {"dtype": "float32", "shape": [8], "name": "conv1_bn/moving_variance"}]}, {"paths": ["group3-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 8, 1], "name": "conv_dw_1/depthwise_kernel"}]}, {"paths": ["group4-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 128, 1], "name": "conv_dw_10/depthwise_kernel"}]}, {"paths": ["group5-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_dw_10_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_10_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_10_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_10_bn/moving_variance"}]}, {"paths": ["group6-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 128, 1], "name": "conv_dw_11/depthwise_kernel"}]}, {"paths": ["group7-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_dw_11_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_11_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_11_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_11_bn/moving_variance"}]}, {"paths": ["group8-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 128, 1], "name": "conv_dw_12/depthwise_kernel"}]}, {"paths": ["group9-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_dw_12_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_12_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_12_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_12_bn/moving_variance"}]}, {"paths": ["group10-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 256, 1], "name": "conv_dw_13/depthwise_kernel"}]}, {"paths": ["group11-shard1of1"], "weights": [{"dtype": "float32", "shape": [256], "name": "conv_dw_13_bn/gamma"}, {"dtype": "float32", "shape": [256], "name": "conv_dw_13_bn/beta"}, {"dtype": "float32", "shape": [256], "name": "conv_dw_13_bn/moving_mean"}, {"dtype": "float32", "shape": [256], "name": "conv_dw_13_bn/moving_variance"}]}, {"paths": ["group12-shard1of1"], "weights": [{"dtype": "float32", "shape": [8], "name": "conv_dw_1_bn/gamma"}, {"dtype": "float32", "shape": [8], "name": "conv_dw_1_bn/beta"}, {"dtype": "float32", "shape": [8], "name": "conv_dw_1_bn/moving_mean"}, {"dtype": "float32", "shape": [8], "name": "conv_dw_1_bn/moving_variance"}]}, {"paths": ["group13-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 16, 1], "name": "conv_dw_2/depthwise_kernel"}]}, {"paths": ["group14-shard1of1"], "weights": [{"dtype": "float32", "shape": [16], "name": "conv_dw_2_bn/gamma"}, {"dtype": "float32", "shape": [16], "name": "conv_dw_2_bn/beta"}, {"dtype": "float32", "shape": [16], "name": "conv_dw_2_bn/moving_mean"}, {"dtype": "float32", "shape": [16], "name": "conv_dw_2_bn/moving_variance"}]}, {"paths": ["group15-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 32, 1], "name": "conv_dw_3/depthwise_kernel"}]}, {"paths": ["group16-shard1of1"], "weights": [{"dtype": "float32", "shape": [32], "name": "conv_dw_3_bn/gamma"}, {"dtype": "float32", "shape": [32], "name": "conv_dw_3_bn/beta"}, {"dtype": "float32", "shape": [32], "name": "conv_dw_3_bn/moving_mean"}, {"dtype": "float32", "shape": [32], "name": "conv_dw_3_bn/moving_variance"}]}, {"paths": ["group17-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 32, 1], "name": "conv_dw_4/depthwise_kernel"}]}, {"paths": ["group18-shard1of1"], "weights": [{"dtype": "float32", "shape": [32], "name": "conv_dw_4_bn/gamma"}, {"dtype": "float32", "shape": [32], "name": "conv_dw_4_bn/beta"}, {"dtype": "float32", "shape": [32], "name": "conv_dw_4_bn/moving_mean"}, {"dtype": "float32", "shape": [32], "name": "conv_dw_4_bn/moving_variance"}]}, {"paths": ["group19-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 64, 1], "name": "conv_dw_5/depthwise_kernel"}]}, {"paths": ["group20-shard1of1"], "weights": [{"dtype": "float32", "shape": [64], "name": "conv_dw_5_bn/gamma"}, {"dtype": "float32", "shape": [64], "name": "conv_dw_5_bn/beta"}, {"dtype": "float32", "shape": [64], "name": "conv_dw_5_bn/moving_mean"}, {"dtype": "float32", "shape": [64], "name": "conv_dw_5_bn/moving_variance"}]}, {"paths": ["group21-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 64, 1], "name": "conv_dw_6/depthwise_kernel"}]}, {"paths": ["group22-shard1of1"], "weights": [{"dtype": "float32", "shape": [64], "name": "conv_dw_6_bn/gamma"}, {"dtype": "float32", "shape": [64], "name": "conv_dw_6_bn/beta"}, {"dtype": "float32", "shape": [64], "name": "conv_dw_6_bn/moving_mean"}, {"dtype": "float32", "shape": [64], "name": "conv_dw_6_bn/moving_variance"}]}, {"paths": ["group23-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 128, 1], "name": "conv_dw_7/depthwise_kernel"}]}, {"paths": ["group24-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_dw_7_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_7_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_7_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_7_bn/moving_variance"}]}, {"paths": ["group25-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 128, 1], "name": "conv_dw_8/depthwise_kernel"}]}, {"paths": ["group26-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_dw_8_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_8_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_8_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_8_bn/moving_variance"}]}, {"paths": ["group27-shard1of1"], "weights": [{"dtype": "float32", "shape": [3, 3, 128, 1], "name": "conv_dw_9/depthwise_kernel"}]}, {"paths": ["group28-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_dw_9_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_9_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_9_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_dw_9_bn/moving_variance"}]}, {"paths": ["group29-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 256, 1000], "name": "conv_preds/kernel"}, {"dtype": "float32", "shape": [1000], "name": "conv_preds/bias"}]}, {"paths": ["group30-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 8, 16], "name": "conv_pw_1/kernel"}]}, {"paths": ["group31-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 128, 128], "name": "conv_pw_10/kernel"}]}, {"paths": ["group32-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_pw_10_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_10_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_10_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_10_bn/moving_variance"}]}, {"paths": ["group33-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 128, 128], "name": "conv_pw_11/kernel"}]}, {"paths": ["group34-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_pw_11_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_11_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_11_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_11_bn/moving_variance"}]}, {"paths": ["group35-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 128, 256], "name": "conv_pw_12/kernel"}]}, {"paths": ["group36-shard1of1"], "weights": [{"dtype": "float32", "shape": [256], "name": "conv_pw_12_bn/gamma"}, {"dtype": "float32", "shape": [256], "name": "conv_pw_12_bn/beta"}, {"dtype": "float32", "shape": [256], "name": "conv_pw_12_bn/moving_mean"}, {"dtype": "float32", "shape": [256], "name": "conv_pw_12_bn/moving_variance"}]}, {"paths": ["group37-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 256, 256], "name": "conv_pw_13/kernel"}]}, {"paths": ["group38-shard1of1"], "weights": [{"dtype": "float32", "shape": [256], "name": "conv_pw_13_bn/gamma"}, {"dtype": "float32", "shape": [256], "name": "conv_pw_13_bn/beta"}, {"dtype": "float32", "shape": [256], "name": "conv_pw_13_bn/moving_mean"}, {"dtype": "float32", "shape": [256], "name": "conv_pw_13_bn/moving_variance"}]}, {"paths": ["group39-shard1of1"], "weights": [{"dtype": "float32", "shape": [16], "name": "conv_pw_1_bn/gamma"}, {"dtype": "float32", "shape": [16], "name": "conv_pw_1_bn/beta"}, {"dtype": "float32", "shape": [16], "name": "conv_pw_1_bn/moving_mean"}, {"dtype": "float32", "shape": [16], "name": "conv_pw_1_bn/moving_variance"}]}, {"paths": ["group40-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 16, 32], "name": "conv_pw_2/kernel"}]}, {"paths": ["group41-shard1of1"], "weights": [{"dtype": "float32", "shape": [32], "name": "conv_pw_2_bn/gamma"}, {"dtype": "float32", "shape": [32], "name": "conv_pw_2_bn/beta"}, {"dtype": "float32", "shape": [32], "name": "conv_pw_2_bn/moving_mean"}, {"dtype": "float32", "shape": [32], "name": "conv_pw_2_bn/moving_variance"}]}, {"paths": ["group42-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 32, 32], "name": "conv_pw_3/kernel"}]}, {"paths": ["group43-shard1of1"], "weights": [{"dtype": "float32", "shape": [32], "name": "conv_pw_3_bn/gamma"}, {"dtype": "float32", "shape": [32], "name": "conv_pw_3_bn/beta"}, {"dtype": "float32", "shape": [32], "name": "conv_pw_3_bn/moving_mean"}, {"dtype": "float32", "shape": [32], "name": "conv_pw_3_bn/moving_variance"}]}, {"paths": ["group44-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 32, 64], "name": "conv_pw_4/kernel"}]}, {"paths": ["group45-shard1of1"], "weights": [{"dtype": "float32", "shape": [64], "name": "conv_pw_4_bn/gamma"}, {"dtype": "float32", "shape": [64], "name": "conv_pw_4_bn/beta"}, {"dtype": "float32", "shape": [64], "name": "conv_pw_4_bn/moving_mean"}, {"dtype": "float32", "shape": [64], "name": "conv_pw_4_bn/moving_variance"}]}, {"paths": ["group46-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 64, 64], "name": "conv_pw_5/kernel"}]}, {"paths": ["group47-shard1of1"], "weights": [{"dtype": "float32", "shape": [64], "name": "conv_pw_5_bn/gamma"}, {"dtype": "float32", "shape": [64], "name": "conv_pw_5_bn/beta"}, {"dtype": "float32", "shape": [64], "name": "conv_pw_5_bn/moving_mean"}, {"dtype": "float32", "shape": [64], "name": "conv_pw_5_bn/moving_variance"}]}, {"paths": ["group48-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 64, 128], "name": "conv_pw_6/kernel"}]}, {"paths": ["group49-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_pw_6_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_6_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_6_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_6_bn/moving_variance"}]}, {"paths": ["group50-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 128, 128], "name": "conv_pw_7/kernel"}]}, {"paths": ["group51-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_pw_7_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_7_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_7_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_7_bn/moving_variance"}]}, {"paths": ["group52-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 128, 128], "name": "conv_pw_8/kernel"}]}, {"paths": ["group53-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_pw_8_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_8_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_8_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_8_bn/moving_variance"}]}, {"paths": ["group54-shard1of1"], "weights": [{"dtype": "float32", "shape": [1, 1, 128, 128], "name": "conv_pw_9/kernel"}]}, {"paths": ["group55-shard1of1"], "weights": [{"dtype": "float32", "shape": [128], "name": "conv_pw_9_bn/gamma"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_9_bn/beta"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_9_bn/moving_mean"}, {"dtype": "float32", "shape": [128], "name": "conv_pw_9_bn/moving_variance"}]}]} -------------------------------------------------------------------------------- /example/src/containers/App/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | import MLClassifierUI from 'mlClassifierUI'; 4 | import Search, { 5 | IImage, 6 | } from '../Search'; 7 | 8 | const CORS_BYPASS = 'https://fast-cove-30289.herokuapp.com/'; 9 | 10 | const qs: { 11 | SHOW_HELP?: string; 12 | SHOW_DOWNLOAD?: string; 13 | } = (window.location.search.split('?').pop() || '').split('&').filter(p => p).map(p => p.split('=')).reduce((obj, [key, val]) => ({ 14 | ...obj, 15 | [key]: (val === "1" || val === "true") ? true : false, 16 | }), {}); 17 | 18 | const SHOW_HELP = qs.SHOW_HELP !== undefined ? qs.SHOW_HELP : true; 19 | const SHOW_DOWNLOAD = qs.SHOW_DOWNLOAD !== undefined ? qs.SHOW_DOWNLOAD : true; 20 | 21 | if (SHOW_HELP === true) { 22 | document.body.innerHTML += ``; 23 | } 24 | 25 | const splitImagesFromLabels = async (images: IImage[]) => { 26 | const origData: { 27 | images: string[]; 28 | labels: string[]; 29 | } = { 30 | images: [], 31 | labels: [], 32 | }; 33 | 34 | return images.reduce((data, image: IImage) => ({ 35 | images: data.images.concat(`${CORS_BYPASS}${image.src}`), 36 | // images: data.images.concat(`${image.src}`), 37 | labels: data.labels.concat(image.label), 38 | }), origData); 39 | } 40 | 41 | interface IState { 42 | training: boolean; 43 | evalImages?: IImage[]; 44 | } 45 | 46 | class App extends React.Component { 47 | public state: IState = { 48 | training: false, 49 | evalImages: undefined, 50 | }; 51 | 52 | private classifier:any; 53 | 54 | public getMLClassifier = (classifier: any) => { 55 | this.classifier = classifier; 56 | } 57 | 58 | public onBeginTraining = () => { 59 | this.setState({ 60 | training: true, 61 | }); 62 | } 63 | 64 | public train = async (trainImages: IImage[], evalImages?: IImage[]) => { 65 | this.onBeginTraining(); 66 | const { 67 | images, 68 | labels, 69 | } = await splitImagesFromLabels(trainImages); 70 | 71 | this.setState({ 72 | evalImages, 73 | }); 74 | 75 | await this.classifier.addData(images, labels, 'train'); 76 | } 77 | 78 | public onTrainComplete = async () => { 79 | if (this.state.evalImages && this.state.evalImages.length) { 80 | const { 81 | images, 82 | labels, 83 | } = await splitImagesFromLabels(this.state.evalImages); 84 | 85 | for (let i = 0; i < images.length; i++) { 86 | const src = images[i]; 87 | const label = labels[i]; 88 | 89 | this.classifier.predict(src, label); 90 | 91 | // const prediction = await this.predictSingleImage(src, label); 92 | // callback({ 93 | // src, 94 | // label, 95 | // prediction, 96 | // }); 97 | } 98 | 99 | // return await this.classifier.addData(images, labels, 'eval'); 100 | } 101 | } 102 | 103 | public render() { 104 | return ( 105 | 106 |
107 |
108 | 114 |
115 | {SHOW_HELP && this.state.training === false && ( 116 |
117 |

Instructions

118 |

119 | Drag and drop some labeled images below to begin training your classifier. You can download image datasets here. 120 |

121 |

Organize your images into folders, where the folders' names are the desired labels.

122 |
123 | 124 | 125 | 126 |
127 |
128 | )} 129 |
130 | {SHOW_HELP && this.state.training === false && ( 131 | 132 |
133 |
134 |

Don't have any images handy? Search below for some images. Select up to 10 that match your query.

135 |

Note This can be a little buggy at the moment due to CORS issues. Working on making it better!

136 |
137 | 140 |
141 | )} 142 |
143 | ); 144 | } 145 | } 146 | 147 | export default App; 148 | -------------------------------------------------------------------------------- /example/src/containers/App/styles.scss: -------------------------------------------------------------------------------- 1 | .app { 2 | display: flex; 3 | flex-direction: row; 4 | width: 300px; 5 | display: flex; 6 | margin-right: 20px; 7 | 8 | &.centeredApp { 9 | margin-right: 0; 10 | } 11 | } 12 | 13 | .classifierContainer { 14 | display: flex; 15 | flex-direction: row; 16 | 17 | .imgContainer { 18 | max-height: 400px; 19 | border-radius: 5px; 20 | overflow: hidden; 21 | box-shadow: 0 2px 4px rgba(0,0,0,0.2); 22 | } 23 | 24 | img { 25 | height: 100%; 26 | max-height: 600px; 27 | } 28 | 29 | &.center { 30 | justify-content: center; 31 | } 32 | } 33 | 34 | .info { 35 | margin: 0 20px 0 0; 36 | 37 | h2 { 38 | margin-top: 0; 39 | } 40 | 41 | a { 42 | color: #100382bd; 43 | text-decoration: underline; 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /example/src/containers/Search/Topic.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | import loading from './loading.gif'; 4 | import { 5 | IImgurImage, 6 | } from './getImages'; 7 | 8 | const Topic = ({ 9 | topic, 10 | images, 11 | picked, 12 | handlePicked, 13 | handleRemove, 14 | }: { 15 | topic: string; 16 | images: IImgurImage[]; 17 | picked: string[]; 18 | handlePicked: (topic: string) => (id: string) => () => void; 19 | handleRemove: (topic: string) => (id: string) => () => void; 20 | }) => ( 21 |
22 |
23 | {images.length ? ( 24 | 31 | ) : ( 32 | 33 | )} 34 |
35 |
36 | ); 37 | 38 | const splitImages = (images: IImgurImage[], picked: string[]) => { 39 | const startingData: { 40 | pickedImages: IImgurImage[]; 41 | unpickedImages: IImgurImage[]; 42 | } = { 43 | pickedImages: [], 44 | unpickedImages: [], 45 | }; 46 | return images.reduce(({ 47 | pickedImages, 48 | unpickedImages, 49 | }, image: IImgurImage) => { 50 | if (picked.includes(image.id)) { 51 | return { 52 | unpickedImages, 53 | pickedImages: [ 54 | ...pickedImages, 55 | image, 56 | ], 57 | }; 58 | } 59 | 60 | return { 61 | pickedImages, 62 | unpickedImages: [ 63 | ...unpickedImages, 64 | image, 65 | ], 66 | }; 67 | }, startingData); 68 | }; 69 | 70 | const Images = ({ 71 | title, 72 | images, 73 | picked, 74 | handlePicked, 75 | handleRemove, 76 | }: { 77 | title: string; 78 | images: IImgurImage[]; 79 | picked: string[]; 80 | handlePicked: (id: string) => () => void; 81 | handleRemove: (id: string) => () => void; 82 | }) => { 83 | const { 84 | pickedImages, 85 | unpickedImages, 86 | } = splitImages(images, picked); 87 | 88 | const training = pickedImages.slice(0, 5); 89 | const validation = pickedImages.slice(5); 90 | 91 | return ( 92 | 93 |
94 | 95 | {training.length > 0 ? ( 96 | 101 | ): null} 102 | {validation.length > 0 ? ( 103 | 108 | ): null} 109 |
110 | {pickedImages.length < 10 ? ( 111 |
112 | {unpickedImages.map((image: IImgurImage) => ( 113 | 118 | ))} 119 |
120 | ) :

All done with these images!

} 121 |
122 | ); 123 | }; 124 | 125 | interface IImageSetProps { 126 | title: string; 127 | images: IImgurImage[]; 128 | handleRemove: (id: string) => () => void; 129 | } 130 | 131 | interface IState { 132 | mounted: boolean; 133 | } 134 | 135 | class ImageSet extends React.Component { 136 | public state: IState = { 137 | mounted: false, 138 | }; 139 | 140 | public componentDidMount() { 141 | setTimeout(() => { 142 | this.setState({ 143 | mounted: true, 144 | }); 145 | }, 10); 146 | } 147 | 148 | public render() { 149 | const { 150 | images, 151 | title, 152 | handleRemove 153 | } = this.props; 154 | 155 | const className = `${styles.set} ${this.state.mounted ? styles.show : ''}`; 156 | 157 | return ( 158 |
159 | 160 | {images.map((image: IImgurImage) => ( 161 | 165 | 166 | 167 | ))} 168 |
169 | ); 170 | } 171 | } 172 | 173 | const SearchImage = ({ 174 | src, 175 | handleClick, 176 | children, 177 | }: { 178 | src: string; 179 | handleClick?: () => void; 180 | children?: any; 181 | }) => ( 182 |
183 | {children} 184 | 185 |
186 | ); 187 | 188 | export default Topic; 189 | -------------------------------------------------------------------------------- /example/src/containers/Search/getImages.ts: -------------------------------------------------------------------------------- 1 | const clientID = '21211812267bd50'; 2 | 3 | interface IData { 4 | data: IImgurDatum[]; 5 | } 6 | 7 | interface IImgurDatum { 8 | id: string; 9 | images: IImgurImage[]; 10 | } 11 | 12 | export interface IImgurImage { 13 | id: string; 14 | link: string; 15 | size: number; 16 | width: number; 17 | height: number; 18 | gifv?: string; 19 | type: string; 20 | } 21 | 22 | const getImages = (q: string) => fetch(`https://api.imgur.com/3/gallery/search?q=${q}`, { 23 | headers: { 24 | Authorization: `Client-ID ${clientID}`, 25 | }, 26 | }).then(resp => resp.json()).then(({ data }: IData) => { 27 | const startingData: IImgurImage[] = []; 28 | const c = data.reduce((collectedImages, { images }) => { 29 | if (!images) { 30 | return collectedImages; 31 | } 32 | 33 | return collectedImages.concat(images); 34 | }, startingData); 35 | 36 | return c.filter((image: IImgurImage) => { 37 | return image && !image.gifv && [ 38 | 'image/jpeg', 39 | 'image/png', 40 | 'image/gif', 41 | ].includes(image.type); 42 | }); 43 | // .map((image: IImgurImage) => image.link); 44 | }); 45 | 46 | export default getImages; 47 | -------------------------------------------------------------------------------- /example/src/containers/Search/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | import getImages, { 4 | IImgurImage, 5 | } from './getImages'; 6 | import Topic from './Topic'; 7 | 8 | interface IProps { 9 | train: (images: IImage[], evalImages: IImage[]) => {}; 10 | } 11 | 12 | export interface IImage { 13 | src: string; 14 | label: string; 15 | } 16 | 17 | interface IState { 18 | value: string; 19 | topics: string[]; 20 | searches: { 21 | [index: string]: IImgurImage[]; 22 | } 23 | picked: { 24 | [index: string]: string[]; 25 | } 26 | } 27 | 28 | class Search extends React.Component { 29 | public state: IState = { 30 | value: '', 31 | topics: [], 32 | searches: { }, 33 | picked: { }, 34 | }; 35 | 36 | public handleKeyDown = (e:React.KeyboardEvent) => { 37 | if (e.key === 'Enter') { 38 | e.preventDefault(); 39 | const value = this.state.value; 40 | this.setState({ 41 | topics: [ 42 | ...this.state.topics, 43 | value, 44 | ], 45 | searches: { 46 | [value]: [], 47 | ...this.state.searches, 48 | }, 49 | picked: { 50 | [value]: [], 51 | ...this.state.picked, 52 | }, 53 | value: '', 54 | }); 55 | getImages(this.state.value).then(images => { 56 | this.setState({ 57 | searches: { 58 | ...this.state.searches, 59 | [value]: images, 60 | }, 61 | }); 62 | }); 63 | } else if (e.key === 'Backspace' && this.state.value === '' && this.state.topics.length > 0) { 64 | this.setState({ 65 | topics: this.state.topics.slice(0, -1), 66 | }); 67 | } 68 | } 69 | 70 | public handleChange = (e:React.ChangeEvent) => { 71 | this.setState({ 72 | value: e.target.value, 73 | }); 74 | }; 75 | 76 | public train = (e:React.MouseEvent) => { 77 | 78 | const images = transformImages(this.state.picked, this.state.searches, 0, 5); 79 | const evalImages = transformImages(this.state.picked, this.state.searches, 5, 10); 80 | 81 | this.props.train(images, evalImages); 82 | } 83 | 84 | // public componentDidMount() { 85 | // this.props.train([{ 86 | // label: 'cat', 87 | // src: 'https://i.imgur.com/40QdFBf.jpg', 88 | // }, { 89 | // label: 'cat', 90 | // src: 'https://imgur.com/cnAtiVO.jpg', 91 | // }, { 92 | // label: 'cat', 93 | // src: 'https://i.imgur.com/6Z4v0Xy.jpg', 94 | // }, { 95 | // label: 'cat', 96 | // src: 'https://i.imgur.com/5HvlY3r.jpg', 97 | // }, { 98 | // label: 'cat', 99 | // src: 'https://i.imgur.com/aHOLAm4.jpg', 100 | // }, { 101 | // label: 'dog', 102 | // src: 'https://i.imgur.com/ALmkDwN.jpg', 103 | // }, { 104 | // label: 'dog', 105 | // src: 'https://i.imgur.com/JpGAInT.jpg', 106 | // }, { 107 | // label: 'dog', 108 | // src: 'https://i.imgur.com/vdrzl74.jpg', 109 | // }, { 110 | // label: 'dog', 111 | // src: 'https://i.imgur.com/xFczRzG.jpg', 112 | // }, { 113 | // label: 'dog', 114 | // src: 'https://i.imgur.com/nBpidmQ.jpg', 115 | // }], [ 116 | 117 | 118 | 119 | // { 120 | // label: 'dog', 121 | // src: 'https://i.imgur.com/nZg4Nn8.jpg', 122 | // }, { 123 | // label: 'cat', 124 | // src: 'https://i.imgur.com/LyH5RK2.jpg', 125 | // }, { 126 | // label: 'cat', 127 | // src: 'https://i.imgur.com/MNHiUjf.jpg', 128 | // }]); 129 | // } 130 | 131 | public handleRemoveTopic = (idx: number) => (e:React.MouseEvent) => { 132 | e.preventDefault(); 133 | 134 | this.setState({ 135 | topics: this.state.topics.reduce((topics: string[], topic, topicId) => { 136 | if (topicId === idx) { 137 | return topics; 138 | } 139 | 140 | return topics.concat(topic); 141 | }, []), 142 | }); 143 | } 144 | 145 | public handleSelectTopicImage = (topic: string) => (id: string) => () => { 146 | if (this.state.picked[topic].length < 10) { 147 | this.setState({ 148 | picked: { 149 | ...this.state.picked, 150 | [topic]: (this.state.picked[topic] || []).concat(id), 151 | }, 152 | }); 153 | } 154 | } 155 | 156 | public handleRemoveTopicImage = (topic: string) => (id: string) => () => { 157 | this.setState({ 158 | picked: { 159 | ...this.state.picked, 160 | [topic]: (this.state.picked[topic] || []).reduce((pickeds: string[], picked) => { 161 | if (picked === id) { 162 | return pickeds; 163 | } 164 | 165 | return pickeds.concat(picked); 166 | }, []), 167 | }, 168 | }); 169 | } 170 | 171 | public render() { 172 | const disabled = this.state.topics.length < 2; 173 | return ( 174 |
175 |
176 | {this.state.topics.map((topic, idx) => ( 177 |
178 | 179 | {topic} 180 |
181 | ))} 182 | 189 | Search provided by Imgur 190 |
191 | {this.state.topics.map((topic, key) => ( 192 | 200 | ))} 201 | 208 |
209 | ); 210 | } 211 | } 212 | 213 | const transformImages = ( 214 | picked: { [index: string]: string[] }, 215 | searches: { [index: string]: IImgurImage[]}, 216 | start: number, 217 | end?: number 218 | ) => { 219 | const originalData: Array<{ 220 | src: string; 221 | label: string; 222 | }> = []; 223 | 224 | return Object.entries(picked).reduce((collected, [label, pickedImages]) => { 225 | return collected.concat(pickedImages.slice(start, end).map((imageId: string) => { 226 | const image = searches[label].filter(({ id }: IImgurImage) => id === imageId)[0]; 227 | 228 | return { 229 | src: image.link, 230 | label, 231 | }; 232 | })); 233 | }, originalData); 234 | } 235 | 236 | export default Search; 237 | -------------------------------------------------------------------------------- /example/src/containers/Search/loading.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thekevinscott/ml-classifier-ui/f3c81d5ee6c3680bd1da0ee923307fc1f6eaf121/example/src/containers/Search/loading.gif -------------------------------------------------------------------------------- /example/src/containers/Search/styles.scss: -------------------------------------------------------------------------------- 1 | .search { 2 | flex: 1; 3 | 4 | .input { 5 | position: relative; 6 | border: 1px solid rgba(0,0,0,0.2); 7 | border-radius: 4px; 8 | display: flex; 9 | margin-bottom: 20px; 10 | 11 | small { 12 | color: rgba(0,0,0,0.6); 13 | position: absolute; 14 | top: 45px; 15 | bottom: 0px; 16 | font-size: 10px; 17 | right: 0px; 18 | } 19 | } 20 | 21 | input { 22 | flex: 1; 23 | display: block; 24 | background: transparent; 25 | border: none; 26 | font-size: 16px; 27 | padding: 10px 10px; 28 | box-sizing: border-box; 29 | outline: none; 30 | } 31 | 32 | button { 33 | padding: 10px 20px; 34 | background: rgba(90,90,255,0.15); 35 | border: 1px solid rgba(0,0,0,0.1); 36 | border-radius: 5px; 37 | font-size: 16px; 38 | outline: none; 39 | 40 | &.add { 41 | $size: 16px; 42 | font-size: 14px; 43 | border: none; 44 | display: flex; 45 | height: $size; 46 | padding: 0; 47 | top: 2px; 48 | left: 2px; 49 | width: $size; 50 | justify-content: center; 51 | align-items: center; 52 | position: absolute; 53 | background: white; 54 | border-radius: $size; 55 | } 56 | } 57 | } 58 | 59 | .topic { 60 | background: rgba(50,50,255,0.04); 61 | border: 1px solid rgba(50,50,255,0.2); 62 | border-radius: 4px; 63 | margin: 5px; 64 | display: flex; 65 | justify-content: center; 66 | align-items: center; 67 | padding: 0 10px 0 0; 68 | 69 | button { 70 | background: none; 71 | border: none; 72 | margin: 0; 73 | padding: 0 5px; 74 | font-size: 20px; 75 | color: rgba(0,0,0,0.6); 76 | margin-top: -2px; 77 | font-size: 16px; 78 | outline: none; 79 | cursor: pointer; 80 | 81 | } 82 | } 83 | 84 | .images { 85 | border: 1px solid rgba(0,0,0,0.1); 86 | padding: 10px; 87 | border-radius: 5px; 88 | margin: 10px 0; 89 | max-height: 460px; 90 | overflow-y: scroll; 91 | 92 | label { 93 | display: block; 94 | font-weight: bold; 95 | margin: 0; 96 | // margin-bottom: 5px; 97 | // margin-top: 20px; 98 | } 99 | } 100 | 101 | .loading { 102 | width: 16px; 103 | height: 16px; 104 | } 105 | 106 | .carousel { 107 | .set { 108 | margin-left: 10px; 109 | display: flex; 110 | flex-direction: row; 111 | border-radius: 5px; 112 | transition-duration: 0.4s; 113 | width: 100%; 114 | max-height: 0; 115 | overflow-x: scroll; 116 | margin-bottom: 20px; 117 | 118 | label { 119 | position: absolute; 120 | z-index: 1; 121 | background: rgba(255,255,255,0.8); 122 | font-size: 12px; 123 | display: block; 124 | color: rgba(0,0,0,0.8); 125 | margin-right: 5px; 126 | } 127 | 128 | &.show { 129 | max-height: 200px; 130 | } 131 | } 132 | 133 | img { 134 | max-width: 100px; 135 | max-height: 100px; 136 | display: block; 137 | } 138 | } 139 | 140 | .trainModel { 141 | margin-top: 20px; 142 | } 143 | 144 | .image { 145 | display: inline-block; 146 | position: relative; 147 | cursor: pointer; 148 | margin-right: 5px; 149 | 150 | img { 151 | display: block; 152 | } 153 | 154 | button { 155 | position: absolute; 156 | height: 100%; 157 | width: 100%; 158 | opacity: 0; 159 | transition-duration: 0.2s; 160 | background: rgba(0,0,0,0.6); 161 | font-size: 60px; 162 | border-radius: 0; 163 | color: white; 164 | 165 | &:hover { 166 | opacity: 1; 167 | } 168 | } 169 | } 170 | 171 | .top { 172 | display: flex; 173 | min-height: 100px; 174 | } 175 | -------------------------------------------------------------------------------- /example/src/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import * as ReactDOM from 'react-dom'; 3 | import App from 'containers/App'; 4 | import './index.css'; 5 | 6 | ReactDOM.render( 7 | , 8 | document.getElementById('root') as HTMLElement, 9 | ); 10 | -------------------------------------------------------------------------------- /example/src/mlClassifierUI.ts: -------------------------------------------------------------------------------- 1 | import '../../dist/index.css'; 2 | export { default } from '../../'; 3 | -------------------------------------------------------------------------------- /example/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "baseUrl": ".", 4 | "outDir": "build/dist", 5 | "module": "esnext", 6 | "target": "es5", 7 | "lib": ["esnext", "dom"], 8 | 9 | "sourceMap": true, 10 | "allowJs": true, 11 | "jsx": "react", 12 | "moduleResolution": "node", 13 | "rootDir": "src", 14 | "forceConsistentCasingInFileNames": true, 15 | "noImplicitReturns": true, 16 | "noImplicitThis": true, 17 | "noImplicitAny": true, 18 | "strictNullChecks": true, 19 | "suppressImplicitAnyIndexErrors": true, 20 | "paths": { 21 | "*": [ 22 | "src/*" 23 | ] 24 | }, 25 | "noUnusedLocals": true 26 | }, 27 | "exclude": [ 28 | "node_modules", 29 | "build", 30 | "scripts", 31 | "acceptance-tests", 32 | "webpack", 33 | "jest", 34 | "*.test.ts", 35 | "src/setupTests.ts" 36 | ] 37 | } 38 | -------------------------------------------------------------------------------- /example/tsconfig.prod.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "./tsconfig.json" 3 | } -------------------------------------------------------------------------------- /example/tsconfig.test.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "./tsconfig.json", 3 | "compilerOptions": { 4 | "module": "commonjs" 5 | } 6 | } -------------------------------------------------------------------------------- /example/tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": [ 3 | "tslint:recommended", 4 | "tslint-react", 5 | "tslint-config-prettier" 6 | ], 7 | "jsEnable": false, 8 | "linterOptions": { 9 | "exclude": [ 10 | "config/**/*.js", 11 | "node_modules/**/*.ts" 12 | ] 13 | }, 14 | "rules": { 15 | "no-var-requires": false, 16 | "prefer-for-of": false, 17 | "object-literal-sort-keys": false, 18 | "ordered-imports": false, 19 | "no-debugger": false, 20 | "no-console": false 21 | }, 22 | "jsRules": { 23 | "no-conditional-assignment": false, 24 | "no-bitwise": false, 25 | "no-duplicate-variable": false, 26 | "no-string-literal": false, 27 | "no-shadowed-variable": false, 28 | "one-variable-per-declaration": false, 29 | "variable-name": false, 30 | "curly": false, 31 | "prefer-for-of": false, 32 | "object-literal-sort-keys": false, 33 | "ordered-imports": false, 34 | "no-debugger": false, 35 | "no-console": false 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /issue_template.md: -------------------------------------------------------------------------------- 1 | Issue tracker is **ONLY** used for reporting bugs. If you are running into issues with Tensorflow.js or Machine Learning in general, please use [stackoverflow](https://stackoverflow.com). 2 | 3 | 4 | 5 | **Describe the bug** 6 | A clear and concise description of what the bug is. 7 | 8 | **To Reproduce** 9 | Steps to reproduce the behavior: 10 | 1. Go to '...' 11 | 2. Click on '....' 12 | 3. Scroll down to '....' 13 | 4. See error 14 | 15 | **Expected behavior** 16 | A clear and concise description of what you expected to happen. 17 | 18 | **Codepen or Repo** 19 | Please provide a link to a hosted code snippet (on Codepen or similar site) _or_ a Github repository. 20 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ml-classifier-ui", 3 | "repository": "https://github.com/thekevinscott/ml-classifier-ui", 4 | "version": "0.5.1", 5 | "homepage": "https://thekevinscott.github.io/ml-classifier-ui/", 6 | "description": "A package for creating image-based machine learning models", 7 | "main": "dist/index.js", 8 | "typings": "dist/index.d.ts", 9 | "scripts": { 10 | "clean": "npm run clean:generated_files && npm run clean:style_types", 11 | "clean:generated_files": "rimraf index.js index.d.ts index.css utils Preview Model Dropzone App dist", 12 | "clean:style_types": "rimraf src/**/*.css.d.ts src/**/*.scss.d.ts", 13 | "watch": "npm run build && rerun-script", 14 | "build": "rollup -c", 15 | "example-start": "node example/scripts/start.js", 16 | "example-build-and-start": "concurrently \"npm run watch\" \"npm run example-start\"", 17 | "example-deploy": "gh-pages -d example/build", 18 | "example-build": "node example/scripts/build.js", 19 | "example-build-and-deploy": "npm run example-build && npm run example-deploy", 20 | "test": "jest", 21 | "storybook": "start-storybook -p 6006", 22 | "build-storybook": "build-storybook" 23 | }, 24 | "watches": { 25 | "build": [ 26 | "src/**/*.scss", 27 | "src/**/*.css", 28 | "src/**/*.ts", 29 | "src/**/*.tsx" 30 | ] 31 | }, 32 | "keywords": [ 33 | "tensorflowjs", 34 | "artificial intelligence", 35 | "machine learning", 36 | "deep learning", 37 | "image classification", 38 | "tensorflow" 39 | ], 40 | "author": "Kevin Scott", 41 | "license": "MIT", 42 | "jest": { 43 | "testMatch": [ 44 | "/src/**/__tests__/**/*.(j|t)s?(x)", 45 | "/src/**/?(*.)(spec|test).(j|t)s?(x)" 46 | ], 47 | "transform": { 48 | "^.+\\.ts?$": "babel-jest" 49 | }, 50 | "transformIgnorePatterns": [ 51 | "[/\\\\]node_modules[/\\\\].+\\.(js|ts)$" 52 | ], 53 | "moduleFileExtensions": [ 54 | "ts", 55 | "js", 56 | "json" 57 | ] 58 | }, 59 | "peerDependencies": { 60 | "react": "^16.4.1" 61 | }, 62 | "devDependencies": { 63 | "@babel/core": "^7.0.0-beta.51", 64 | "@babel/plugin-proposal-decorators": "^7.0.0-beta.51", 65 | "@babel/plugin-transform-classes": "^7.0.0-beta.51", 66 | "@babel/preset-env": "^7.0.0-beta.51", 67 | "@babel/preset-stage-0": "^7.0.0-beta.51", 68 | "@babel/preset-typescript": "^7.0.0-beta.51", 69 | "@storybook/addon-actions": "^3.4.8", 70 | "@storybook/addon-links": "^3.4.8", 71 | "@storybook/react": "^3.4.8", 72 | "@types/jest": "^23.1.1", 73 | "@types/node": "^10.3.4", 74 | "@types/react": "^16.4.6", 75 | "@types/react-dom": "^16.0.6", 76 | "autoprefixer": "^8.6.5", 77 | "babel-core": "^7.0.0-0", 78 | "babel-jest": "^23.2.0", 79 | "babel-plugin-transform-class-properties": "^6.24.1", 80 | "babel-plugin-transform-runtime": "^6.23.0", 81 | "babel-polyfill": "^6.26.0", 82 | "concurrently": "^3.6.0", 83 | "express": "^4.16.3", 84 | "gh-pages": "^1.2.0", 85 | "jest": "^24.9.0", 86 | "node-sass": "^4.9.2", 87 | "npm-run-all": "^4.1.3", 88 | "react": "^16.4.1", 89 | "react-dom": "^16.4.1", 90 | "regenerator-runtime": "^0.12.0", 91 | "rerun-script": "^0.6.0", 92 | "rimraf": "^2.6.2", 93 | "rollup": "^0.62.0", 94 | "rollup-plugin-commonjs": "^9.1.3", 95 | "rollup-plugin-image": "^1.0.2", 96 | "rollup-plugin-node-resolve": "^3.3.0", 97 | "rollup-plugin-postcss-modules": "^1.0.8", 98 | "rollup-plugin-replace": "^2.0.0", 99 | "rollup-plugin-typescript2": "^0.15.1", 100 | "sass-loader": "^7.0.3", 101 | "ts-jest": "^24.0.2", 102 | "ts-loader": "^2.3.7", 103 | "tslint": "^5.7.0", 104 | "tslint-config-prettier": "^1.10.0", 105 | "typescript": "3.0.0-rc", 106 | "uglify-es": "^3.3.9" 107 | }, 108 | "dependencies": { 109 | "ml-classifier": "0.5.1" 110 | } 111 | } 112 | -------------------------------------------------------------------------------- /rollup.config.js: -------------------------------------------------------------------------------- 1 | /* globals Promise */ 2 | import path from 'path'; 3 | import sass from 'node-sass'; 4 | import typescript from 'rollup-plugin-typescript2'; 5 | import replace from 'rollup-plugin-replace'; 6 | import postcss from 'rollup-plugin-postcss-modules'; 7 | import image from 'rollup-plugin-image'; 8 | import autoprefixer from 'autoprefixer'; 9 | 10 | export default { 11 | input: './src/index.ts', 12 | output: { 13 | moduleName: 'MLClassifierUI', 14 | name: 'MLClassifierUI', 15 | file: './dist/index.js', 16 | format: 'es', 17 | }, 18 | plugins: [ 19 | typescript({ 20 | typescript: require("typescript"), 21 | }), 22 | replace({ 23 | 'process.env.NODE_ENV': JSON.stringify('production'), 24 | }), 25 | postcss({ 26 | preprocessor: (content, id) => new Promise(resolve => { 27 | const result = sass.renderSync({ file: id }); 28 | resolve({ code: result.css.toString() }); 29 | }), 30 | extract: true, 31 | plugins: [autoprefixer()], 32 | // sometimes this writes out blank definitions 33 | writeDefinitions: false, 34 | modules: true, 35 | extensions: [ 36 | '.css', 37 | '.scss', 38 | ], 39 | use: [ 40 | [ 41 | 'sass', 42 | { 43 | includePaths: [ 44 | path.resolve('node_modules'), 45 | ], 46 | }, 47 | ], 48 | ], 49 | }), 50 | image(), 51 | ], 52 | }; 53 | -------------------------------------------------------------------------------- /rollup.config.js.umd: -------------------------------------------------------------------------------- 1 | /* globals Promise */ 2 | import path from 'path'; 3 | import sass from 'node-sass'; 4 | import typescript from 'rollup-plugin-typescript2'; 5 | import replace from 'rollup-plugin-replace'; 6 | import commonjs from 'rollup-plugin-commonjs'; 7 | import postcss from 'rollup-plugin-postcss-modules'; 8 | import image from 'rollup-plugin-image'; 9 | import resolve from 'rollup-plugin-node-resolve'; 10 | import autoprefixer from 'autoprefixer'; 11 | 12 | export default { 13 | input: './src/index.ts', 14 | output: { 15 | moduleName: 'MLClassifierUI', 16 | external: [ 17 | 'react', 18 | 'react-dom', 19 | 'tf', 20 | ], 21 | globals: { 22 | 'react': 'React', 23 | 'react-dom': 'ReactDOM', 24 | }, 25 | name: 'MLClassifierUI', 26 | file: './dist/index.js', 27 | format: 'umd', 28 | }, 29 | plugins: [ 30 | resolve({ 31 | jsnext: true, 32 | main: true, 33 | extensions: [ '.ts', '.tsx', '.js', '.json' ], 34 | only: [ 35 | 'ml-classifier', 36 | ], 37 | }), 38 | typescript({ 39 | typescript: require("typescript"), 40 | }), 41 | replace({ 42 | 'process.env.NODE_ENV': JSON.stringify('production'), 43 | }), 44 | commonjs({ 45 | ignoreGlobal: false, // Default: false 46 | }), 47 | image(), 48 | postcss({ 49 | preprocessor: (content, id) => new Promise(resolve => { 50 | const result = sass.renderSync({ file: id }); 51 | resolve({ code: result.css.toString() }); 52 | }), 53 | extract: true, 54 | plugins: [autoprefixer()], 55 | // sometimes this writes out blank definitions 56 | writeDefinitions: false, 57 | modules: true, 58 | extensions: [ 59 | '.css', 60 | '.scss', 61 | ], 62 | use: [ 63 | [ 64 | 'sass', 65 | { 66 | includePaths: [ 67 | path.resolve('node_modules'), 68 | ], 69 | }, 70 | ], 71 | ], 72 | }), 73 | ], 74 | }; 75 | -------------------------------------------------------------------------------- /src/App/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | 3 | import Dropzone from '../Dropzone'; 4 | import { 5 | getFilesAsImageArray, 6 | IFileData, 7 | splitImagesFromLabels, 8 | } from 'utils/getFilesAsImages'; 9 | import nextFrame from 'utils/nextFrame'; 10 | 11 | import { 12 | ITrainResult, 13 | } from '../types'; 14 | 15 | import Model, { ImageError } from '../Model'; 16 | import Preview from '../Preview'; 17 | 18 | import MLClassifier from 'ml-classifier'; 19 | 20 | import styles from './styles.scss'; 21 | 22 | export interface IImage { 23 | imageSrc: string; 24 | label: string; 25 | } 26 | interface IParams { 27 | [index: string]: any; 28 | } 29 | 30 | interface IState { 31 | status: string; 32 | images?: string[]; 33 | files: any[]; 34 | labels: string[]; 35 | downloading: boolean; 36 | predictions: { 37 | src: string; 38 | prediction: string; 39 | label: string; 40 | }[]; 41 | logs: { 42 | [index: string]: any; 43 | }; 44 | accuracy: { 45 | training?: number; 46 | evaluation?: number; 47 | }; 48 | errors?: ImageError[]; 49 | } 50 | 51 | interface IProps { 52 | params: { 53 | train?: IParams; 54 | evaluate?: IParams; 55 | save?: IParams; 56 | }; 57 | getMLClassifier?: Function; 58 | uploadFormat: string; 59 | imagesFormats: string[]; 60 | showDownload?: boolean; 61 | onLoadStart?: Function; 62 | onLoadComplete?: Function; 63 | onAddDataStart?: Function; 64 | onAddDataComplete?: Function; 65 | onClearDataStart?: Function; 66 | onClearDataComplete?: Function; 67 | onTrainStart?: Function; 68 | onTrainComplete?: Function; 69 | onPredictComplete?: Function; 70 | onPredictStart?: Function; 71 | onEvaluateStart?: Function; 72 | onEvaluateComplete?: Function; 73 | onSaveStart?: Function; 74 | onSaveComplete?: Function; 75 | } 76 | 77 | class MLClassifierUI extends React.Component { 78 | public static defaultProps: Partial = { 79 | params: { }, 80 | uploadFormat: 'nested', 81 | imagesFormats: undefined, 82 | showDownload: true, 83 | }; 84 | 85 | private classifier: any; 86 | 87 | constructor(props: IProps) { 88 | super(props); 89 | 90 | this.state = { 91 | errors: [], 92 | files: [], 93 | status: 'empty', 94 | images: undefined, 95 | downloading: false, 96 | predictions: [], 97 | logs: {}, 98 | labels: [], 99 | accuracy: { 100 | training: undefined, 101 | evaluation: undefined, 102 | }, 103 | }; 104 | } 105 | 106 | componentDidMount = async () => { 107 | this.classifier = new MLClassifier({ 108 | onLoadStart: this.props.onLoadStart, 109 | onLoadComplete: this.props.onLoadComplete, 110 | onAddDataStart: this.onAddDataStart, 111 | onAddDataComplete: this.onAddDataComplete, 112 | onClearDataStart: this.props.onClearDataStart, 113 | onClearDataComplete: this.props.onClearDataComplete, 114 | onTrainStart: this.props.onTrainStart, 115 | onTrainComplete: this.props.onTrainComplete, 116 | onPredictStart: this.props.onPredictStart, 117 | onPredictComplete: this.onPredictComplete, 118 | onEvaluateStart: this.props.onEvaluateStart, 119 | onEvaluateComplete: this.props.onEvaluateComplete, 120 | onSaveStart: this.props.onSaveStart, 121 | onSaveComplete: this.props.onSaveComplete, 122 | }); 123 | 124 | if (this.props.getMLClassifier) { 125 | this.props.getMLClassifier(this.classifier); 126 | } 127 | } 128 | 129 | private onDrop = (files: FileList) => { 130 | this.setState({ 131 | status: 'uploading', 132 | }); 133 | } 134 | 135 | private onAddDataStart = async (imageSrcs: string[], _labels: any, dataType: string) => { 136 | this.setState({ 137 | status: 'parsing' 138 | }); 139 | 140 | if (this.props.onAddDataStart) { 141 | this.props.onAddDataStart(); 142 | } 143 | 144 | if (dataType === 'train') { 145 | this.setState({ 146 | status: 'training', 147 | images: imageSrcs, 148 | }); 149 | } 150 | } 151 | 152 | private onParseFiles = async (origFiles: FileList) => { 153 | const imageFiles: IFileData[] = await getFilesAsImageArray(origFiles); 154 | 155 | const { 156 | images, 157 | labels, 158 | files, 159 | } = await splitImagesFromLabels(imageFiles); 160 | 161 | this.setState({ 162 | files, 163 | }); 164 | 165 | return this.classifier.addData(images, labels, 'train'); 166 | } 167 | 168 | private onAddDataComplete = async (imageSrcs: string[], labels: string[], dataType: string, errors?: ImageError[]) => { 169 | if (this.props.onAddDataComplete) { 170 | this.props.onAddDataComplete(imageSrcs, labels, dataType, errors); 171 | } 172 | if (dataType === 'train') { 173 | this.setState({ 174 | status: 'training', 175 | images: imageSrcs, 176 | labels, 177 | errors: (errors || []).map((error: ImageError) => { 178 | return { 179 | ...error, 180 | file: this.state.files[error.index], 181 | }; 182 | }), 183 | }); 184 | 185 | const train = this.props.params.train || {}; 186 | const result: ITrainResult = await this.classifier.train({ 187 | ...train, 188 | callbacks: { 189 | onBatchEnd: async (batch: any, logs: any) => { 190 | if (train.callbacks && train.callbacks.onBatchEnd) { 191 | train.callbacks.onBatchEnd(batch, logs); 192 | } 193 | const loss = logs.loss.toFixed(5); 194 | // log(batch, logs); 195 | // log('Loss is: ' + logs.loss.toFixed(5)); 196 | this.setState({ 197 | logs: { 198 | ...this.state.logs, 199 | loss: (this.state.logs.loss || []).concat(loss), 200 | } 201 | }); 202 | 203 | await nextFrame(); 204 | }, 205 | }, 206 | }); 207 | 208 | const { 209 | history: { 210 | acc, 211 | // loss, 212 | }, 213 | } = result; 214 | 215 | const training = acc[acc.length - 1]; 216 | this.setState({ 217 | status: 'trained', 218 | accuracy: { 219 | ...this.state.accuracy, 220 | training, 221 | }, 222 | }); 223 | } 224 | } 225 | 226 | public onPredictComplete = async (src: string, label: string, pred: string | number) => { 227 | if (this.props.onPredictComplete) { 228 | this.props.onPredictComplete(src, label, pred); 229 | } 230 | const prediction = `${pred}`; 231 | 232 | this.setState({ 233 | predictions: this.state.predictions.concat({ 234 | src, 235 | prediction, 236 | label, 237 | }), 238 | }); 239 | } 240 | 241 | public predict = async (imageFiles: IFileData[]) => { 242 | for (let i = 0; i < imageFiles.length; i++) { 243 | const { 244 | src, 245 | label, 246 | } = imageFiles[i]; 247 | 248 | await this.classifier.predict(src, label); 249 | } 250 | }; 251 | 252 | handleDownload = async () => { 253 | this.setState({ 254 | downloading: true, 255 | }); 256 | await this.classifier.save((this.props.params || {}).save); 257 | this.setState({ 258 | downloading: false, 259 | }); 260 | }; 261 | 262 | public render() { 263 | return ( 264 |
265 | {this.state.status === 'empty' && ( 266 | 270 | )} 271 | {['training', 'uploading', 'parsing'].includes(this.state.status) && ( 272 | 275 | )} 276 | {this.state.status === 'trained' && this.state.images && ( 277 | 287 | )} 288 |
289 | ); 290 | } 291 | } 292 | 293 | export default MLClassifierUI; 294 | -------------------------------------------------------------------------------- /src/App/styles.scss: -------------------------------------------------------------------------------- 1 | .classifier { 2 | max-width: 300px; 3 | width: 300px; 4 | background: white; 5 | height: 300px; 6 | 7 | h1,h2,h3,h4,h5,h6 { 8 | font-weight: inherit; 9 | } 10 | 11 | button { 12 | cursor: pointer; 13 | } 14 | 15 | ul, li { 16 | list-style: none; 17 | margin: 0; 18 | padding: 0; 19 | } 20 | 21 | * { 22 | box-sizing: border-box; 23 | } 24 | 25 | a, button { 26 | cursor: pointer; 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /src/Dropzone/getFile.ts: -------------------------------------------------------------------------------- 1 | const getFile = async (item: any, path: string) => new Promise(resolve => { 2 | item.file((file: any) => { 3 | const reader = new FileReader(); 4 | reader.onload = (e: any) => { 5 | return resolve({ 6 | file, 7 | src: e.target.result, 8 | path: path.split('/').join(''), 9 | }); 10 | }; 11 | reader.readAsDataURL(file); 12 | }); 13 | }); 14 | 15 | export default getFile; 16 | -------------------------------------------------------------------------------- /src/Dropzone/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import classNames from 'utils/classNames'; 3 | import transformFiles from './transformFiles'; 4 | import styles from './styles.scss'; 5 | 6 | interface IProps { 7 | onDrop?: Function; 8 | onParseFiles: Function; 9 | style?: any; 10 | children?: any; 11 | } 12 | 13 | interface IState { 14 | over: boolean; 15 | } 16 | 17 | class Dropzone extends React.Component { 18 | private timeout: number; 19 | constructor(props: IProps) { 20 | super(props); 21 | 22 | this.state = { 23 | over: false, 24 | }; 25 | } 26 | 27 | public handleDrop = async (e: React.DragEvent) => { 28 | if (this.props.onDrop) { 29 | this.props.onDrop(); 30 | } 31 | e.preventDefault(); 32 | e.persist(); 33 | 34 | // console.log('pre transforming files'); 35 | const folders = await transformFiles(e); 36 | // console.log('post transforming files'); 37 | if (e.dataTransfer.items) { 38 | e.dataTransfer.items.clear(); 39 | } else { 40 | e.dataTransfer.clearData(); 41 | } 42 | this.props.onParseFiles(folders); 43 | } 44 | 45 | public handleDrag = (over: boolean) => { 46 | return (e: React.DragEvent) => { 47 | e.preventDefault(); 48 | this.setState({ 49 | over, 50 | }); 51 | } 52 | } 53 | 54 | componentWillUnmount() { 55 | if (this.timeout) { 56 | clearTimeout(this.timeout); 57 | } 58 | } 59 | 60 | public stop = (e: any) => { 61 | if (this.timeout) { 62 | clearTimeout(this.timeout); 63 | } 64 | if (this.state.over === false) { 65 | this.setState({ 66 | over: true, 67 | }); 68 | } 69 | this.timeout = setTimeout(() => { 70 | this.setState({ 71 | over: false, 72 | }); 73 | }, 50); 74 | e.preventDefault(); 75 | } 76 | 77 | public render() { 78 | const className = classNames(styles.container, { 79 | [styles.over]: this.state.over, 80 | }); 81 | return ( 82 |
91 | {this.props.children || (Drop Images To Begin Training)} 92 | 99 |
100 | ); 101 | } 102 | }; 103 | 104 | export default Dropzone; 105 | -------------------------------------------------------------------------------- /src/Dropzone/styles.scss: -------------------------------------------------------------------------------- 1 | .container { 2 | text-align: center; 3 | color: rgba(0,0,0,0.4); 4 | border-radius: 5px; 5 | height: 100%; 6 | width: 100%; 7 | display: flex; 8 | position: relative; 9 | justify-content: center; 10 | align-items: center; 11 | background: rgba(0,0,0,0); 12 | border: 2px dashed rgba(0,0,0,0.2); 13 | transition-duration: 0.2s; 14 | 15 | &.over { 16 | background: rgba(155,77,202,0.2); 17 | border: 2px dashed #9b4dca; 18 | transition-duration: 0.1s; 19 | } 20 | } 21 | 22 | .input { 23 | display: none; 24 | } 25 | -------------------------------------------------------------------------------- /src/Dropzone/transformFiles.ts: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import traverseFileTree from './traverseFileTree'; 3 | 4 | const transformFiles = async (e: React.DragEvent) => { 5 | const d = e.dataTransfer || {}; 6 | const items = d.items || []; 7 | let images: any = []; 8 | for (let i=0; i { 17 | if (image.path) { 18 | return { 19 | ...allImages, 20 | [image.path]: (allImages[image.path] || []).concat(image), 21 | }; 22 | } 23 | 24 | return allImages; 25 | }, {}); 26 | return folders; 27 | }; 28 | 29 | export default transformFiles; 30 | -------------------------------------------------------------------------------- /src/Dropzone/traverseFileTree.ts: -------------------------------------------------------------------------------- 1 | import getFile from './getFile'; 2 | 3 | const readDir = async (dir: any, path: string) => new Promise(resolve => { 4 | const dirReader = dir.createReader(); 5 | let items: any = []; 6 | dirReader.readEntries(async (entries: any) => { 7 | for (let i = 0; i { 23 | return await readDir(item, path); 24 | }; 25 | 26 | export default traverseFileTree; 27 | -------------------------------------------------------------------------------- /src/Loading/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import loading from './loading'; 3 | 4 | const SIZE = 32; 5 | 6 | interface IProps { 7 | size?: number; 8 | } 9 | 10 | const Loading: React.SFC = ({ 11 | size, 12 | }) => ( 13 | Loading... 21 | ); 22 | 23 | export default Loading; 24 | -------------------------------------------------------------------------------- /src/Loading/loading.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thekevinscott/ml-classifier-ui/f3c81d5ee6c3680bd1da0ee923307fc1f6eaf121/src/Loading/loading.gif -------------------------------------------------------------------------------- /src/Loading/loading.ts: -------------------------------------------------------------------------------- 1 | const loadingGif = ''; 2 | 3 | export default loadingGif; 4 | -------------------------------------------------------------------------------- /src/Model/Evaluator/Predictions/Prediction.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | 4 | export interface IPrediction { 5 | prediction: string; 6 | label: string; 7 | src: string; 8 | }; 9 | 10 | interface IProps { 11 | prediction: IPrediction; 12 | } 13 | 14 | const Prediction: React.SFC = ({ 15 | prediction: { 16 | src, 17 | prediction, 18 | label, 19 | }, 20 | }) => ( 21 |
  • 22 | 23 |
      24 |
    • Prediction: {prediction}
    • 25 |
    • Label: {label}
    • 26 |
    27 |
  • 28 | ); 29 | 30 | export default Prediction; 31 | -------------------------------------------------------------------------------- /src/Model/Evaluator/Predictions/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | import Prediction, { 4 | IPrediction, 5 | } from './Prediction'; 6 | 7 | interface IProps { 8 | predictions: IPrediction[]; 9 | } 10 | 11 | const Predictions: React.SFC = ({ 12 | predictions, 13 | }) => { 14 | return ( 15 |
      16 | {predictions.map((prediction: IPrediction, idx: number) => ( 17 | 18 | ))} 19 |
    20 | ); 21 | }; 22 | 23 | export default Predictions; 24 | -------------------------------------------------------------------------------- /src/Model/Evaluator/Predictions/styles.scss: -------------------------------------------------------------------------------- 1 | .container { 2 | img { 3 | width: 100%; 4 | max-width: 300px; 5 | display: block; 6 | } 7 | } 8 | 9 | .info { 10 | border-radius: 0 0 5px 5px; 11 | background: rgba(0,13,51,0.6); 12 | padding: 10px !important; 13 | display: flex; 14 | color: rgba(255,255,255,0.8); 15 | flex: 1; 16 | margin-bottom: 20px !important; 17 | 18 | li { 19 | flex: 1; 20 | } 21 | } 22 | 23 | .label { 24 | text-align: right; 25 | } 26 | -------------------------------------------------------------------------------- /src/Model/Evaluator/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import Dropzone from '../../Dropzone'; 3 | import { 4 | IFileData, 5 | getFilesAsImageArray, 6 | } from '../../utils/getFilesAsImages'; 7 | import Predictions from './Predictions'; 8 | import { 9 | IPrediction, 10 | } from './Predictions/Prediction'; 11 | 12 | interface IProps { 13 | predict: Function; 14 | predictions: IPrediction[]; 15 | } 16 | 17 | interface IState { 18 | imagesParsed: number; 19 | totalFiles: number; 20 | } 21 | 22 | class Evaluator extends React.Component { 23 | constructor(props: IProps) { 24 | super(props); 25 | 26 | this.state = { 27 | imagesParsed: 0, 28 | totalFiles: 0, 29 | }; 30 | } 31 | 32 | private onParseFiles = async (files: FileList) => { 33 | const imageFiles: IFileData[] = await getFilesAsImageArray(files); 34 | this.props.predict(imageFiles); 35 | } 36 | 37 | render() { 38 | return ( 39 | 40 | 44 | Drop Images to test 45 | 46 | 47 | ); 48 | } 49 | }; 50 | 51 | export default Evaluator; 52 | -------------------------------------------------------------------------------- /src/Model/Metrics/Info.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | 4 | export interface IMetricsInfoProps { 5 | title: string; 6 | data: IDatum[]; 7 | } 8 | 9 | export interface IDatum { 10 | data: string | number; 11 | label: string; 12 | } 13 | 14 | const Info: React.SFC = ({ 15 | title, 16 | data, 17 | }) => { 18 | return ( 19 | 20 |

    {title}

    21 |
      22 | {data.map((datum: IDatum) => ( 23 |
    • 24 | {datum.data} 25 | 26 |
    • 27 | ))} 28 |
    29 |
    30 | ); 31 | }; 32 | 33 | export default Info; 34 | -------------------------------------------------------------------------------- /src/Model/Metrics/Logs/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import classNames from 'utils/classNames'; 3 | import styles from './styles.scss'; 4 | 5 | interface IProps { 6 | logs: { 7 | [index: string]: any; 8 | }; 9 | } 10 | 11 | interface IState { 12 | expanded: boolean; 13 | } 14 | 15 | class Logs extends React.Component { 16 | public state: IState = { 17 | expanded: false, 18 | }; 19 | 20 | handleClick = (e:any) => { 21 | this.setState({ 22 | expanded: !this.state.expanded, 23 | }); 24 | } 25 | 26 | render() { 27 | const { 28 | logs, 29 | } = this.props; 30 | 31 | return ( 32 |
    37 |
    38 | 39 | 40 |
    41 |
    42 |           {logs.loss.join('\n')}
    43 |         
    44 |
    45 | ); 46 | } 47 | } 48 | 49 | export default Logs; 50 | -------------------------------------------------------------------------------- /src/Model/Metrics/Logs/styles.scss: -------------------------------------------------------------------------------- 1 | .logs { 2 | position: relative; 3 | min-height: 40px; 4 | display: flex; 5 | flex: 1; 6 | margin-right: 20px; 7 | 8 | &.expanded { 9 | .expand { 10 | a { 11 | transform: rotate(90deg); 12 | } 13 | } 14 | 15 | pre { 16 | display: block; 17 | max-height: 100px; 18 | padding: 5px 10px; 19 | } 20 | } 21 | 22 | pre { 23 | transition-duration: 0.5s; 24 | max-height: 0px; 25 | margin: 0 auto; 26 | overflow-y: scroll; 27 | font-size: 12px; 28 | width: 100%; 29 | background: rgba(255,255,255,1.0); 30 | border-radius: 4px; 31 | padding: 0 10px; 32 | color: rgba(0,0,0,0.6); 33 | } 34 | } 35 | 36 | .expand { 37 | height: 21px; 38 | position: absolute; 39 | display: flex; 40 | align-items: center; 41 | top: -20px; 42 | left: 0; 43 | 44 | label { 45 | font-size: 12px; 46 | } 47 | 48 | a { 49 | font-size: 16px; 50 | transition-duration: 0.2s; 51 | display: inline-block; 52 | margin-right: 5px; 53 | } 54 | } 55 | -------------------------------------------------------------------------------- /src/Model/Metrics/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | import Info, { 4 | IDatum, 5 | } from './Info'; 6 | // import { 7 | // IImageData, 8 | // } from 'utils/getFilesAsImages'; 9 | import Logs from './Logs'; 10 | export interface ImageError { 11 | image: any; 12 | error: Error; 13 | file?: any; 14 | index: number; 15 | } 16 | 17 | interface IProps { 18 | labels?: string[]; 19 | downloading: boolean; 20 | onDownload?: Function; 21 | errors?: ImageError[]; 22 | logs: { 23 | [index: string]: any; 24 | }; 25 | accuracy: IDatum[]; 26 | } 27 | 28 | // interface IData { 29 | // label: string; 30 | // } 31 | 32 | const getData = (labels: string[] = []) => { 33 | const numOfLabels = labels.reduce((obj, label) => ({ 34 | ...obj, 35 | [label]: (obj[label] || 0) + 1, 36 | }), {}); 37 | 38 | return Object.keys(numOfLabels).map((label) => { 39 | return { 40 | label, 41 | data: numOfLabels[label], 42 | }; 43 | }); 44 | }; 45 | 46 | const getErrors = (errors: ImageError[] = []) => errors.map(({ file }, key) => ( 47 |

    48 | Error loading image: {`${file.path}/${file.file.name}`} 49 |

    50 | )); 51 | 52 | class Metrics extends React.Component { 53 | render() { 54 | const { 55 | labels, 56 | onDownload, 57 | downloading, 58 | accuracy, 59 | logs, 60 | errors, 61 | } = this.props; 62 | 63 | return ( 64 |
    65 | 69 |
    70 | {errors && getErrors(errors)} 71 |
    72 | 76 |
    77 | 78 | {onDownload && ( 79 | 85 | )} 86 |
    87 |
    88 | ); 89 | } 90 | } 91 | 92 | export default Metrics; 93 | 94 | export { IMetricsInfoProps, IDatum } from './Info'; 95 | -------------------------------------------------------------------------------- /src/Model/Metrics/styles.scss: -------------------------------------------------------------------------------- 1 | .container { 2 | border-radius: 5px 5px 0 0; 3 | width: 100%; 4 | height: 100%; 5 | background: rgba(0,13,51,0.6); 6 | padding: 20px; 7 | color: rgba(255,255,255,0.8); 8 | font-size: 20px; 9 | display: flex; 10 | flex-direction: column; 11 | 12 | button { 13 | width: 120px; 14 | height: 40px; 15 | border: none; 16 | border-radius: 5px; 17 | } 18 | } 19 | 20 | .footer { 21 | flex: 1; 22 | display: flex; 23 | align-items: flex-end; 24 | } 25 | 26 | .datum { 27 | display: flex; 28 | flex-direction: column; 29 | justify-content: center; 30 | align-items: center; 31 | flex: 1; 32 | 33 | data { 34 | font-size: 40px; 35 | } 36 | 37 | label { 38 | color: rgba(255,255,255,0.6); 39 | } 40 | } 41 | 42 | .data { 43 | display: flex; 44 | margin-bottom: 20px !important; 45 | } 46 | 47 | .header { 48 | font-size: 16px; 49 | margin: 0; 50 | } 51 | 52 | .error { 53 | margin: 0; 54 | font-size: 12px; 55 | } 56 | 57 | .errors { 58 | margin-bottom: 20px; 59 | } 60 | -------------------------------------------------------------------------------- /src/Model/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import styles from './styles.scss'; 3 | // const styles = require('./styles.scss'); 4 | import Evaluator from './Evaluator'; 5 | import Metrics, { 6 | IDatum, 7 | ImageError, 8 | } from './Metrics'; 9 | import { 10 | IPrediction, 11 | } from './Evaluator/Predictions/Prediction'; 12 | // import { 13 | // IImageData, 14 | // } from '../utils/getFilesAsImages'; 15 | 16 | interface IProps { 17 | labels: string[]; 18 | downloading: boolean; 19 | onDownload?: Function; 20 | predict?: Function; 21 | predictions: IPrediction[]; 22 | errors?: ImageError[]; 23 | logs: { 24 | [index: string]: any; 25 | }; 26 | accuracy: { 27 | training?: number; 28 | evaluation?: number; 29 | }; 30 | } 31 | 32 | interface IState { 33 | } 34 | 35 | const getEvaluation = (predictions: any[]) => { 36 | if (predictions.length > 0) { 37 | return predictions.reduce((sum, { 38 | prediction, 39 | label, 40 | }) => sum + (prediction === label ? 1 : 0), 0) / predictions.length; 41 | } 42 | 43 | return null; 44 | }; 45 | 46 | class Model extends React.Component { 47 | render() { 48 | const { 49 | labels, 50 | onDownload, 51 | downloading, 52 | predict, 53 | predictions, 54 | logs, 55 | accuracy: { 56 | training, 57 | }, 58 | errors, 59 | } = this.props; 60 | 61 | const evaluation = getEvaluation(predictions); 62 | 63 | const accuracy: IDatum[] = [{ 64 | data: training ? `${Math.round(training * 100)}%` : '--', 65 | label: 'Training', 66 | }, { 67 | data: evaluation ? `${Math.round(evaluation * 100)}%` : '--', 68 | label: 'Evaluation', 69 | }]; 70 | 71 | return ( 72 |
    73 | 81 | {predict && ( 82 | 86 | )} 87 |
    88 | ); 89 | } 90 | } 91 | 92 | export default Model; 93 | 94 | export { ImageError } from './Metrics'; 95 | -------------------------------------------------------------------------------- /src/Model/styles.scss: -------------------------------------------------------------------------------- 1 | .container { 2 | display: flex; 3 | flex-direction: column; 4 | } 5 | -------------------------------------------------------------------------------- /src/Preview/index.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import classNames from 'utils/classNames'; 3 | import styles from './styles.scss'; 4 | import Loading from '../Loading'; 5 | // import { 6 | // loadImage, 7 | // } from 'utils/getFilesAsImages'; 8 | // const styles = require('./styles.scss'); 9 | // import { 10 | // IImageData, 11 | // } from '../utils/getFilesAsImages'; 12 | 13 | interface IProps { 14 | images?: string[]; 15 | } 16 | 17 | interface IState { 18 | imageIdx: number; 19 | // images: { 20 | // [index:string]: HTMLImageElement | null; 21 | // }; 22 | } 23 | 24 | const LOOP_SPEED = 75; 25 | 26 | class Preview extends React.Component { 27 | constructor(props: IProps) { 28 | super(props); 29 | 30 | this.state = { 31 | imageIdx: 0, 32 | // images: { }, 33 | }; 34 | } 35 | 36 | private timeout: any; 37 | 38 | componentWillMount() { 39 | this.loopImages(); 40 | } 41 | 42 | // componentWillReceiveProps(nextProps: IProps) { 43 | // if ((this.props.images || []).length !== (nextProps.images || []).length) { 44 | // (nextProps.images || []).map((src: string) => { 45 | // if (this.state.images[src] === undefined) { 46 | // console.log('load image', src); 47 | 48 | // this.setState({ 49 | // images: { 50 | // ...this.state.images, 51 | // [src]: null, 52 | // }, 53 | // }); 54 | 55 | // loadImage(src).then(image => { 56 | // this.setState({ 57 | // images: { 58 | // ...this.state.images, 59 | // [src]: image, 60 | // }, 61 | // }); 62 | // }); 63 | // } 64 | // }); 65 | // } 66 | // } 67 | 68 | componentWillUnmount() { 69 | if (this.timeout) { 70 | clearTimeout(this.timeout); 71 | } 72 | } 73 | 74 | private last: any; 75 | loopImages = () => { 76 | if (this.last) { 77 | const now = (new Date()).getTime(); 78 | const diff = now - this.last; 79 | if (diff > LOOP_SPEED + 200) { // 200 ms wiggle room 80 | // TODO: this indicates a UI slowdown 81 | // console.log('diff', diff); 82 | } 83 | } 84 | if (this.timeout) { 85 | clearTimeout(this.timeout); 86 | } 87 | 88 | this.setState({ 89 | imageIdx: this.state.imageIdx + 1, 90 | }); 91 | 92 | this.last = (new Date()).getTime(); 93 | this.timeout = setTimeout(() => { 94 | this.loopImages(); 95 | }, LOOP_SPEED); 96 | } 97 | 98 | render() { 99 | // const images = Object.values(this.state.images); 100 | const { 101 | images, 102 | } = this.props; 103 | 104 | const src = images && images[this.state.imageIdx % images.length]; 105 | 106 | const className = classNames(styles.container, { 107 | [styles.images]: images && images.length > 0, 108 | }); 109 | 110 | return ( 111 |
    112 | {src ? ( 113 |
    119 | ) : ( 120 |
    121 | 122 | Reading images 123 |
    124 | )} 125 |
    126 | ); 127 | } 128 | } 129 | 130 | export default Preview; 131 | -------------------------------------------------------------------------------- /src/Preview/styles.scss: -------------------------------------------------------------------------------- 1 | .loader { 2 | opacity: 1; 3 | position: absolute; 4 | height: 100%; 5 | width: 100%; 6 | display: flex; 7 | justify-content: center; 8 | flex-direction: column; 9 | align-items: center; 10 | color: rgba(0,0,0,0.3); 11 | 12 | img { 13 | height: auto; 14 | width: auto; 15 | } 16 | } 17 | 18 | .container { 19 | border-radius: 5px; 20 | display: flex; 21 | justify-content: center; 22 | position: relative; 23 | align-items: center; 24 | width: 100%; 25 | height: 100%; 26 | border: 2px dashed rgba(0,0,0,0.2); 27 | 28 | .img { 29 | width: 100%; 30 | height: 100%; 31 | background-color: rgba(0,0,0,0.2); 32 | background-repeat: no-repeat; 33 | background-position: 50% 50%; 34 | background-size: cover; 35 | position: relative; 36 | } 37 | 38 | &.images { 39 | border: 2px dashed rgba(0,0,0,0); 40 | } 41 | } 42 | 43 | .loading { 44 | width: 16px; 45 | height: 16px; 46 | } 47 | -------------------------------------------------------------------------------- /src/global.d.ts: -------------------------------------------------------------------------------- 1 | declare module '*.scss' { 2 | const content: any; 3 | export default content; 4 | } 5 | declare module '*.css' { 6 | const content: any; 7 | export default content; 8 | } 9 | 10 | declare module '*.png' 11 | declare module '*.jpg' 12 | declare module '*.jpeg' 13 | declare module '*.gif' 14 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | export { default } from './App'; 2 | -------------------------------------------------------------------------------- /src/types.ts: -------------------------------------------------------------------------------- 1 | export interface IProps { 2 | } 3 | 4 | export interface ITrainResult { 5 | epoch: number[]; 6 | history: { 7 | acc: number[]; 8 | loss: number[]; 9 | }; 10 | model: any; 11 | params: any; 12 | validationData: any; 13 | } 14 | -------------------------------------------------------------------------------- /src/utils/__tests__/classNames.test.ts: -------------------------------------------------------------------------------- 1 | import classNames from '../classNames'; 2 | 3 | describe('classNames', () => { 4 | test('it handles nothing', () => { 5 | expect(classNames()).toEqual(''); 6 | }); 7 | 8 | test('it handles a single class', () => { 9 | expect(classNames('foo')).toEqual('foo'); 10 | }); 11 | 12 | test('it handles multiple classes', () => { 13 | expect(classNames('foo', 'bar', 'baz')).toEqual('foo bar baz'); 14 | }); 15 | 16 | test('it does not include undefined or null', () => { 17 | expect(classNames('foo', undefined, null, 'baz')).toEqual('foo baz'); 18 | }); 19 | 20 | test('it handles an object where value is true', () => { 21 | expect(classNames({ 22 | foo: true, 23 | })).toEqual('foo'); 24 | }); 25 | 26 | test('it handles an object where value are true and false', () => { 27 | expect(classNames({ 28 | foo: true, 29 | bar: false, 30 | })).toEqual('foo'); 31 | }); 32 | 33 | test('it handles an object where value are true and false', () => { 34 | expect(classNames('bar', { 35 | foo: true, 36 | quom: false, 37 | }, 'baz')).toEqual('bar foo baz'); 38 | }); 39 | }); 40 | -------------------------------------------------------------------------------- /src/utils/classNames.ts: -------------------------------------------------------------------------------- 1 | interface IClassNameObject { 2 | [index:string]: boolean | undefined; 3 | } 4 | 5 | const arr: string[] = []; 6 | 7 | type Name = string|IClassNameObject|undefined|null; 8 | 9 | const transformObjToArr = (name: IClassNameObject) => Object.entries(name).filter(entry => entry[1]).map(([ key ]) => key); 10 | 11 | const classNames = (...names: Name[]) => (names || []).reduce((arr, name: Name) => { 12 | if (!name) { 13 | return arr; 14 | } 15 | 16 | if (typeof name === 'string') { 17 | return arr.concat(name); 18 | } 19 | 20 | return arr.concat(transformObjToArr(name)); 21 | }, arr).join(' '); 22 | 23 | export default classNames; 24 | -------------------------------------------------------------------------------- /src/utils/getFilesAsImages.ts: -------------------------------------------------------------------------------- 1 | export interface IFileData { 2 | label: string; 3 | file: any; 4 | src: string; 5 | }; 6 | 7 | export interface IImageData { 8 | label: string; 9 | image: HTMLImageElement; 10 | }; 11 | 12 | export const loadImage = async (src: string) => new Promise((resolve, reject) => { 13 | const image = new Image(); 14 | image.src = src; 15 | image.onload = () => resolve(image); 16 | image.onerror = (err) => reject(err); 17 | }); 18 | 19 | export const getFilesAsImageArray = async (files: FileList): Promise => { 20 | const classes = Object.keys(files); 21 | const images = []; 22 | for (let i = 0; i < classes.length; i++) { 23 | const label = classes[i]; 24 | for (let j = 0; j < files[label].length; j++) { 25 | const file = files[label][j]; 26 | images.push({ 27 | label, 28 | src: file.src, 29 | file, 30 | }); 31 | } 32 | } 33 | return images; 34 | }; 35 | 36 | export default getFilesAsImageArray; 37 | 38 | export const splitImagesFromLabels = async (images: IFileData[]) => { 39 | const origData: { 40 | images: string[]; 41 | labels: string[]; 42 | files: any[]; 43 | } = { 44 | images: [], 45 | labels: [], 46 | files: [], 47 | }; 48 | 49 | return images.reduce((data, { 50 | src, 51 | label, 52 | file, 53 | }: IFileData) => ({ 54 | images: data.images.concat(src), 55 | labels: data.labels.concat(label), 56 | files: data.files.concat(file), 57 | }), origData); 58 | } 59 | -------------------------------------------------------------------------------- /src/utils/nextFrame.ts: -------------------------------------------------------------------------------- 1 | const nextFrame = () => new Promise(resolve => { 2 | window.requestAnimationFrame(() => resolve()); 3 | }); 4 | export default nextFrame; 5 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "baseUrl": ".", 4 | "outDir": "dist", 5 | "module": "esnext", 6 | "target": "es5", 7 | "lib": ["esnext", "dom"], 8 | "sourceMap": true, 9 | "allowJs": false, 10 | "declaration": false, 11 | "removeComments": true, 12 | "moduleResolution": "node", 13 | "rootDir": "src", 14 | "jsx": "react", 15 | "forceConsistentCasingInFileNames": true, 16 | "noImplicitReturns": true, 17 | "noImplicitThis": true, 18 | "noUnusedLocals": true, 19 | "noImplicitAny": true, 20 | "strictNullChecks": true, 21 | "suppressImplicitAnyIndexErrors": true, 22 | "paths": { 23 | "*": [ 24 | "src/*" 25 | ] 26 | } 27 | }, 28 | "include": [ 29 | "src/**/*" 30 | ], 31 | "exclude": [ 32 | "__tests__", 33 | "__mocks__", 34 | "**/*.test.ts", 35 | "**/*.scss.d.ts", 36 | "src/types.ts", 37 | "node_modules" 38 | ] 39 | } 40 | 41 | 42 | -------------------------------------------------------------------------------- /tslint.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": [ 3 | "tslint:recommended", 4 | "tslint-react", 5 | "tslint-config-prettier" 6 | ], 7 | "linterOptions": { 8 | "exclude": [ 9 | "config/**/*.js", 10 | "node_modules/**/*.ts" 11 | ] 12 | }, 13 | "rules": { 14 | "prefer-for-of": false, 15 | "object-literal-sort-keys": false, 16 | "ordered-imports": false, 17 | "no-debugger": false, 18 | "no-console": false 19 | }, 20 | "jsRules": { 21 | "prefer-for-of": false, 22 | "object-literal-sort-keys": false, 23 | "ordered-imports": false, 24 | "no-debugger": false, 25 | "no-console": false 26 | } 27 | } 28 | 29 | --------------------------------------------------------------------------------