| @ -0,0 +1,18 @@ | |||
| { | |||
| "presets": [ | |||
| ["env", { | |||
| "modules": false, | |||
| "targets": { | |||
| "browsers": ["> 1%", "last 2 versions", "not ie <= 8"] | |||
| } | |||
| }], | |||
| "stage-2" | |||
| ], | |||
| "plugins": ["transform-vue-jsx", "transform-runtime"], | |||
| "env": { | |||
| "test": { | |||
| "presets": ["env", "stage-2"], | |||
| "plugins": ["transform-vue-jsx", "transform-es2015-modules-commonjs", "dynamic-import-node"] | |||
| } | |||
| } | |||
| } | |||
| @ -0,0 +1,9 @@ | |||
| root = true | |||
| [*] | |||
| charset = utf-8 | |||
| indent_style = space | |||
| indent_size = 2 | |||
| end_of_line = lf | |||
| insert_final_newline = true | |||
| trim_trailing_whitespace = true | |||
| @ -0,0 +1,10 @@ | |||
| // https://github.com/michael-ciniawsky/postcss-load-config | |||
| module.exports = { | |||
| "plugins": { | |||
| "postcss-import": {}, | |||
| "postcss-url": {}, | |||
| // to edit target browsers: use "browserslist" field in package.json | |||
| "autoprefixer": {} | |||
| } | |||
| } | |||
| @ -0,0 +1,41 @@ | |||
| 'use strict' | |||
| require('./check-versions')() | |||
| process.env.NODE_ENV = 'production' | |||
| const ora = require('ora') | |||
| const rm = require('rimraf') | |||
| const path = require('path') | |||
| const chalk = require('chalk') | |||
| const webpack = require('webpack') | |||
| const config = require('../config') | |||
| const webpackConfig = require('./webpack.prod.conf') | |||
| const spinner = ora('building for production...') | |||
| spinner.start() | |||
| rm(path.join(config.build.assetsRoot, config.build.assetsSubDirectory), err => { | |||
| if (err) throw err | |||
| webpack(webpackConfig, (err, stats) => { | |||
| spinner.stop() | |||
| if (err) throw err | |||
| process.stdout.write(stats.toString({ | |||
| colors: true, | |||
| modules: false, | |||
| children: false, // If you are using ts-loader, setting this to true will make TypeScript errors show up during build. | |||
| chunks: false, | |||
| chunkModules: false | |||
| }) + '\n\n') | |||
| if (stats.hasErrors()) { | |||
| console.log(chalk.red(' Build failed with errors.\n')) | |||
| process.exit(1) | |||
| } | |||
| console.log(chalk.cyan(' Build complete.\n')) | |||
| console.log(chalk.yellow( | |||
| ' Tip: built files are meant to be served over an HTTP server.\n' + | |||
| ' Opening index.html over file:// won\'t work.\n' | |||
| )) | |||
| }) | |||
| }) | |||
| @ -0,0 +1,54 @@ | |||
| 'use strict' | |||
| const chalk = require('chalk') | |||
| const semver = require('semver') | |||
| const packageConfig = require('../package.json') | |||
| const shell = require('shelljs') | |||
| function exec (cmd) { | |||
| return require('child_process').execSync(cmd).toString().trim() | |||
| } | |||
| const versionRequirements = [ | |||
| { | |||
| name: 'node', | |||
| currentVersion: semver.clean(process.version), | |||
| versionRequirement: packageConfig.engines.node | |||
| } | |||
| ] | |||
| if (shell.which('npm')) { | |||
| versionRequirements.push({ | |||
| name: 'npm', | |||
| currentVersion: exec('npm --version'), | |||
| versionRequirement: packageConfig.engines.npm | |||
| }) | |||
| } | |||
| module.exports = function () { | |||
| const warnings = [] | |||
| for (let i = 0; i < versionRequirements.length; i++) { | |||
| const mod = versionRequirements[i] | |||
| if (!semver.satisfies(mod.currentVersion, mod.versionRequirement)) { | |||
| warnings.push(mod.name + ': ' + | |||
| chalk.red(mod.currentVersion) + ' should be ' + | |||
| chalk.green(mod.versionRequirement) | |||
| ) | |||
| } | |||
| } | |||
| if (warnings.length) { | |||
| console.log('') | |||
| console.log(chalk.yellow('To use this template, you must update following to modules:')) | |||
| console.log() | |||
| for (let i = 0; i < warnings.length; i++) { | |||
| const warning = warnings[i] | |||
| console.log(' ' + warning) | |||
| } | |||
| console.log() | |||
| process.exit(1) | |||
| } | |||
| } | |||
| @ -0,0 +1,102 @@ | |||
| 'use strict' | |||
| const path = require('path') | |||
| const config = require('../config') | |||
| const ExtractTextPlugin = require('extract-text-webpack-plugin') | |||
| const packageConfig = require('../package.json') | |||
| exports.assetsPath = function (_path) { | |||
| const assetsSubDirectory = process.env.NODE_ENV === 'production' | |||
| ? config.build.assetsSubDirectory | |||
| : config.dev.assetsSubDirectory | |||
| return path.posix.join(assetsSubDirectory, _path) | |||
| } | |||
| exports.cssLoaders = function (options) { | |||
| options = options || {} | |||
| const cssLoader = { | |||
| loader: 'css-loader', | |||
| options: { | |||
| sourceMap: options.sourceMap | |||
| } | |||
| } | |||
| const postcssLoader = { | |||
| loader: 'postcss-loader', | |||
| options: { | |||
| sourceMap: options.sourceMap | |||
| } | |||
| } | |||
| // generate loader string to be used with extract text plugin | |||
| function generateLoaders (loader, loaderOptions) { | |||
| const loaders = options.usePostCSS ? [cssLoader, postcssLoader] : [cssLoader] | |||
| if (loader) { | |||
| loaders.push({ | |||
| loader: loader + '-loader', | |||
| options: Object.assign({}, loaderOptions, { | |||
| sourceMap: options.sourceMap | |||
| }) | |||
| }) | |||
| } | |||
| // Extract CSS when that option is specified | |||
| // (which is the case during production build) | |||
| if (options.extract) { | |||
| return ExtractTextPlugin.extract({ | |||
| use: loaders, | |||
| fallback: 'vue-style-loader', | |||
| publicPath:'../../' | |||
| }) | |||
| } else { | |||
| return ['vue-style-loader'].concat(loaders) | |||
| } | |||
| } | |||
| // https://vue-loader.vuejs.org/en/configurations/extract-css.html | |||
| return { | |||
| css: generateLoaders(), | |||
| postcss: generateLoaders(), | |||
| less: generateLoaders('less'), | |||
| sass: generateLoaders('sass', { indentedSyntax: true }), | |||
| scss: generateLoaders('sass'), | |||
| stylus: generateLoaders('stylus'), | |||
| styl: generateLoaders('stylus') | |||
| } | |||
| } | |||
| // Generate loaders for standalone style files (outside of .vue) | |||
| exports.styleLoaders = function (options) { | |||
| const output = [] | |||
| const loaders = exports.cssLoaders(options) | |||
| for (const extension in loaders) { | |||
| const loader = loaders[extension] | |||
| output.push({ | |||
| test: new RegExp('\\.' + extension + '$'), | |||
| use: loader | |||
| }) | |||
| } | |||
| return output | |||
| } | |||
| exports.createNotifierCallback = () => { | |||
| const notifier = require('node-notifier') | |||
| return (severity, errors) => { | |||
| if (severity !== 'error') return | |||
| const error = errors[0] | |||
| const filename = error.file && error.file.split('!').pop() | |||
| notifier.notify({ | |||
| title: packageConfig.name, | |||
| message: severity + ': ' + error.name, | |||
| subtitle: filename || '', | |||
| icon: path.join(__dirname, 'logo.png') | |||
| }) | |||
| } | |||
| } | |||
| @ -0,0 +1,22 @@ | |||
| 'use strict' | |||
| const utils = require('./utils') | |||
| const config = require('../config') | |||
| const isProduction = process.env.NODE_ENV === 'production' | |||
| const sourceMapEnabled = isProduction | |||
| ? config.build.productionSourceMap | |||
| : config.dev.cssSourceMap | |||
| module.exports = { | |||
| loaders: utils.cssLoaders({ | |||
| sourceMap: sourceMapEnabled, | |||
| extract: isProduction | |||
| }), | |||
| cssSourceMap: sourceMapEnabled, | |||
| cacheBusting: config.dev.cacheBusting, | |||
| transformToRequire: { | |||
| video: ['src', 'poster'], | |||
| source: 'src', | |||
| img: 'src', | |||
| image: 'xlink:href' | |||
| } | |||
| } | |||
| @ -0,0 +1,106 @@ | |||
| 'use strict' | |||
| const path = require('path') | |||
| const utils = require('./utils') | |||
| const config = require('../config') | |||
| const vueLoaderConfig = require('./vue-loader.conf') | |||
| function resolve(dir) { | |||
| return path.join(__dirname, '..', dir) | |||
| } | |||
| module.exports = { | |||
| context: path.resolve(__dirname, '../'), | |||
| entry: { | |||
| // app: './src/main.js' | |||
| app: ['babel-polyfill','./src/main.js'] | |||
| }, | |||
| output: { | |||
| path: config.build.assetsRoot, | |||
| filename: '[name].js', | |||
| publicPath: process.env.NODE_ENV === 'production' ? | |||
| config.build.assetsPublicPath : | |||
| config.dev.assetsPublicPath | |||
| }, | |||
| resolve: { | |||
| extensions: ['.js', '.vue', '.json'], | |||
| alias: { | |||
| 'vue$': 'vue/dist/vue.esm.js', | |||
| '@': resolve('src'), | |||
| 'jquery': path.resolve(__dirname, '../node_modules/jquery/src/jquery') | |||
| } | |||
| }, | |||
| module: { | |||
| rules: [{ | |||
| test: /\.vue$/, | |||
| loader: 'vue-loader', | |||
| options: vueLoaderConfig | |||
| }, | |||
| { | |||
| test: /\.js$/, | |||
| loader: 'babel-loader', | |||
| include: [resolve('src'), resolve('test'), resolve('node_modules/webpack-dev-server/client')] | |||
| }, | |||
| { | |||
| test: /\.js$/, | |||
| loader: 'babel-loader', | |||
| include: [resolve('src'), resolve('test'), resolve('node_modules/@jiaminghi/data-view/lib')] | |||
| }, | |||
| { | |||
| test: /\.js$/, | |||
| loader: 'babel-loader', | |||
| include: [resolve('src'),resolve('test'),resolve('node_modules/element-ui/src'),resolve('/node_modules/element-ui/packages') | |||
| ] | |||
| }, | |||
| { | |||
| test: /\.js$/, | |||
| loader: 'babel-loader', | |||
| include: [resolve('src'),resolve('test'),resolve('node_modules/element-ui/src'),resolve('/node_modules/vue-baidu-map/components/base') | |||
| ] | |||
| }, | |||
| { | |||
| test: /\.js$/, | |||
| loader: 'babel-loader', | |||
| include: [resolve('src'),resolve('test'),resolve('node_modules/js-base64') | |||
| ] | |||
| }, | |||
| { | |||
| test: /\.(png|jpe?g|gif|svg)(\?.*)?$/, | |||
| loader: 'url-loader', | |||
| options: { | |||
| limit: 10000, | |||
| name: utils.assetsPath('img/[name].[hash:7].[ext]') | |||
| } | |||
| }, | |||
| { | |||
| test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/, | |||
| loader: 'url-loader', | |||
| options: { | |||
| limit: 10000, | |||
| name: utils.assetsPath('media/[name].[hash:7].[ext]') | |||
| } | |||
| }, | |||
| { | |||
| test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/, | |||
| loader: 'url-loader', | |||
| options: { | |||
| limit: 10000, | |||
| name: utils.assetsPath('fonts/[name].[hash:7].[ext]') | |||
| } | |||
| } | |||
| ] | |||
| }, | |||
| node: { | |||
| // prevent webpack from injecting useless setImmediate polyfill because Vue | |||
| // source contains it (although only uses it if it's native). | |||
| setImmediate: false, | |||
| // prevent webpack from injecting mocks to Node native modules | |||
| // that does not make sense for the client | |||
| dgram: 'empty', | |||
| fs: 'empty', | |||
| net: 'empty', | |||
| tls: 'empty', | |||
| child_process: 'empty' | |||
| } | |||
| } | |||
| @ -0,0 +1,96 @@ | |||
| 'use strict' | |||
| const utils = require('./utils') | |||
| const webpack = require('webpack') | |||
| const config = require('../config') | |||
| const merge = require('webpack-merge') | |||
| const path = require('path') | |||
| const baseWebpackConfig = require('./webpack.base.conf') | |||
| const CopyWebpackPlugin = require('copy-webpack-plugin') | |||
| const HtmlWebpackPlugin = require('html-webpack-plugin') | |||
| const FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin') | |||
| const portfinder = require('portfinder') | |||
| const HOST = process.env.HOST | |||
| const PORT = process.env.PORT && Number(process.env.PORT) | |||
| const devWebpackConfig = merge(baseWebpackConfig, { | |||
| module: { | |||
| rules: utils.styleLoaders({ sourceMap: config.dev.cssSourceMap, usePostCSS: true }) | |||
| }, | |||
| // cheap-module-eval-source-map is faster for development | |||
| devtool: config.dev.devtool, | |||
| // these devServer options should be customized in /config/index.js | |||
| devServer: { | |||
| clientLogLevel: 'warning', | |||
| historyApiFallback: { | |||
| rewrites: [ | |||
| { from: /.*/, to: path.posix.join(config.dev.assetsPublicPath, 'index.html') }, | |||
| ], | |||
| }, | |||
| https:false, | |||
| hot: true, | |||
| contentBase: false, // since we use CopyWebpackPlugin. | |||
| compress: true, | |||
| host: HOST || config.dev.host, | |||
| port: PORT || config.dev.port, | |||
| open: config.dev.autoOpenBrowser, | |||
| overlay: config.dev.errorOverlay | |||
| ? { warnings: false, errors: true } | |||
| : false, | |||
| publicPath: config.dev.assetsPublicPath, | |||
| proxy: config.dev.proxyTable, | |||
| quiet: true, // necessary for FriendlyErrorsPlugin | |||
| watchOptions: { | |||
| poll: config.dev.poll, | |||
| } | |||
| }, | |||
| plugins: [ | |||
| new webpack.DefinePlugin({ | |||
| 'process.env': require('../config/dev.env') | |||
| }), | |||
| new webpack.HotModuleReplacementPlugin(), | |||
| new webpack.NamedModulesPlugin(), // HMR shows correct file names in console on update. | |||
| new webpack.NoEmitOnErrorsPlugin(), | |||
| // https://github.com/ampedandwired/html-webpack-plugin | |||
| new HtmlWebpackPlugin({ | |||
| filename: 'index.html', | |||
| template: 'index.html', | |||
| inject: true | |||
| }), | |||
| // copy custom static assets | |||
| new CopyWebpackPlugin([ | |||
| { | |||
| from: path.resolve(__dirname, '../static'), | |||
| to: config.dev.assetsSubDirectory, | |||
| ignore: ['.*'] | |||
| } | |||
| ]) | |||
| ] | |||
| }) | |||
| module.exports = new Promise((resolve, reject) => { | |||
| portfinder.basePort = process.env.PORT || config.dev.port | |||
| portfinder.getPort((err, port) => { | |||
| if (err) { | |||
| reject(err) | |||
| } else { | |||
| // publish the new Port, necessary for e2e tests | |||
| process.env.PORT = port | |||
| // add port to devServer config | |||
| devWebpackConfig.devServer.port = port | |||
| // Add FriendlyErrorsPlugin | |||
| devWebpackConfig.plugins.push(new FriendlyErrorsPlugin({ | |||
| compilationSuccessInfo: { | |||
| messages: [`Your application is running here: http://${devWebpackConfig.devServer.host}:${port}`], | |||
| }, | |||
| onErrors: config.dev.notifyOnErrors | |||
| ? utils.createNotifierCallback() | |||
| : undefined | |||
| })) | |||
| resolve(devWebpackConfig) | |||
| } | |||
| }) | |||
| }) | |||
| @ -0,0 +1,140 @@ | |||
| 'use strict' | |||
| const path = require('path') | |||
| const utils = require('./utils') | |||
| const webpack = require('webpack') | |||
| const config = require('../config') | |||
| const merge = require('webpack-merge') | |||
| const baseWebpackConfig = require('./webpack.base.conf') | |||
| const CopyWebpackPlugin = require('copy-webpack-plugin') | |||
| const HtmlWebpackPlugin = require('html-webpack-plugin') | |||
| const ExtractTextPlugin = require('extract-text-webpack-plugin') | |||
| const OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin') | |||
| const env = process.env.NODE_ENV === 'testing' | |||
| ? require('../config/test.env') | |||
| : require('../config/prod.env') | |||
| const webpackConfig = merge(baseWebpackConfig, { | |||
| module: { | |||
| rules: utils.styleLoaders({ | |||
| sourceMap: config.build.productionSourceMap, | |||
| extract: true, | |||
| usePostCSS: true | |||
| }) | |||
| }, | |||
| devtool: config.build.productionSourceMap ? config.build.devtool : false, | |||
| output: { | |||
| publicPath: '/', | |||
| path: config.build.assetsRoot, | |||
| filename: utils.assetsPath('js/[name].[chunkhash].js'), | |||
| chunkFilename: utils.assetsPath('js/[id].[chunkhash].js') | |||
| }, | |||
| plugins: [ | |||
| // http://vuejs.github.io/vue-loader/en/workflow/production.html | |||
| new webpack.DefinePlugin({ | |||
| 'process.env': env | |||
| }), | |||
| // extract css into its own file | |||
| new ExtractTextPlugin({ | |||
| filename: utils.assetsPath('css/[name].[contenthash].css'), | |||
| // Setting the following option to `false` will not extract CSS from codesplit chunks. | |||
| // Their CSS will instead be inserted dynamically with style-loader when the codesplit chunk has been loaded by webpack. | |||
| // It's currently set to `true` because we are seeing that sourcemaps are included in the codesplit bundle as well when it's `false`, | |||
| // increasing file size: https://github.com/vuejs-templates/webpack/issues/1110 | |||
| allChunks: true, | |||
| }), | |||
| // Compress extracted CSS. We are using this plugin so that possible | |||
| // duplicated CSS from different components can be deduped. | |||
| new OptimizeCSSPlugin({ | |||
| cssProcessorOptions: config.build.productionSourceMap | |||
| ? { safe: true, map: { inline: false } } | |||
| : { safe: true } | |||
| }), | |||
| // generate dist index.html with correct asset hash for caching. | |||
| // you can customize output by editing /index.html | |||
| // see https://github.com/ampedandwired/html-webpack-plugin | |||
| new HtmlWebpackPlugin({ | |||
| filename: process.env.NODE_ENV === 'testing' | |||
| ? 'index.html' | |||
| : config.build.index, | |||
| template: 'index.html', | |||
| inject: true, | |||
| minify: { | |||
| removeComments: true, | |||
| collapseWhitespace: true, | |||
| removeAttributeQuotes: true | |||
| // more options: | |||
| // https://github.com/kangax/html-minifier#options-quick-reference | |||
| }, | |||
| // necessary to consistently work with multiple chunks via CommonsChunkPlugin | |||
| chunksSortMode: 'dependency' | |||
| }), | |||
| // keep module.id stable when vendor modules does not change | |||
| new webpack.HashedModuleIdsPlugin(), | |||
| // enable scope hoisting | |||
| new webpack.optimize.ModuleConcatenationPlugin(), | |||
| // split vendor js into its own file | |||
| new webpack.optimize.CommonsChunkPlugin({ | |||
| name: 'vendor', | |||
| minChunks (module) { | |||
| // any required modules inside node_modules are extracted to vendor | |||
| return ( | |||
| module.resource && | |||
| /\.js$/.test(module.resource) && | |||
| module.resource.indexOf( | |||
| path.join(__dirname, '../node_modules') | |||
| ) === 0 | |||
| ) | |||
| } | |||
| }), | |||
| // extract webpack runtime and module manifest to its own file in order to | |||
| // prevent vendor hash from being updated whenever app bundle is updated | |||
| new webpack.optimize.CommonsChunkPlugin({ | |||
| name: 'manifest', | |||
| minChunks: Infinity | |||
| }), | |||
| // This instance extracts shared chunks from code splitted chunks and bundles them | |||
| // in a separate chunk, similar to the vendor chunk | |||
| // see: https://webpack.js.org/plugins/commons-chunk-plugin/#extra-async-commons-chunk | |||
| new webpack.optimize.CommonsChunkPlugin({ | |||
| name: 'app', | |||
| async: 'vendor-async', | |||
| children: true, | |||
| minChunks: 3 | |||
| }), | |||
| // copy custom static assets | |||
| new CopyWebpackPlugin([ | |||
| { | |||
| from: path.resolve(__dirname, '../static'), | |||
| to: config.build.assetsSubDirectory, | |||
| ignore: ['.*'] | |||
| } | |||
| ]) | |||
| ] | |||
| }) | |||
| if (config.build.productionGzip) { | |||
| const CompressionWebpackPlugin = require('compression-webpack-plugin') | |||
| webpackConfig.plugins.push( | |||
| new CompressionWebpackPlugin({ | |||
| asset: '[path].gz[query]', | |||
| algorithm: 'gzip', | |||
| test: new RegExp( | |||
| '\\.(' + | |||
| config.build.productionGzipExtensions.join('|') + | |||
| ')$' | |||
| ), | |||
| threshold: 10240, | |||
| minRatio: 0.8 | |||
| }) | |||
| ) | |||
| } | |||
| if (config.build.bundleAnalyzerReport) { | |||
| const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin | |||
| webpackConfig.plugins.push(new BundleAnalyzerPlugin()) | |||
| } | |||
| module.exports = webpackConfig | |||
| @ -0,0 +1,7 @@ | |||
| 'use strict' | |||
| const merge = require('webpack-merge') | |||
| const prodEnv = require('./prod.env') | |||
| module.exports = merge(prodEnv, { | |||
| NODE_ENV: '"development"' | |||
| }) | |||
| @ -0,0 +1,60 @@ | |||
| 'use strict' | |||
| // Template version: 1.3.1 | |||
| // see http://vuejs-templates.github.io/webpack for documentation. | |||
| const path = require('path') | |||
| module.exports = { | |||
| dev: { | |||
| // Paths | |||
| assetsSubDirectory: 'static', | |||
| assetsPublicPath: '/', | |||
| proxyTable: { | |||
| '/api': { | |||
| target: 'http://apartmentcloud.xiaozhisz.cn/', | |||
| // target: 'http://test.zhiweisz.cn/', | |||
| // target: 'http://192.168.1.26:8600/', | |||
| changeOrigin: true, //是否跨域 | |||
| pathRewrite: { | |||
| //重写路径 | |||
| '^/api': '' //代理路径 | |||
| } | |||
| } | |||
| }, | |||
| configureWebpack: { | |||
| resolve: { | |||
| extensions: ['.js', '.vue', '.json'], | |||
| alias: { | |||
| vue$: 'vue/dist/vue.esm.js', | |||
| '@': path.join(__dirname, '/', 'src') | |||
| } | |||
| } | |||
| }, | |||
| host: '192.168.1.20', | |||
| port: 8082, | |||
| autoOpenBrowser: false, | |||
| errorOverlay: true, | |||
| notifyOnErrors: true, | |||
| poll: false, | |||
| devtool: 'cheap-module-eval-source-map', | |||
| cacheBusting: true, | |||
| cssSourceMap: true | |||
| }, | |||
| build: { | |||
| // Template for index.html | |||
| index: path.resolve(__dirname, '../dist/index.html'), | |||
| assetsRoot: path.resolve(__dirname, '../dist'), | |||
| assetsSubDirectory: 'static', | |||
| assetsPublicPath: '/', | |||
| /** | |||
| * Source Maps | |||
| */ | |||
| productionSourceMap: false, | |||
| devtool: '#source-map', | |||
| productionGzip: false, | |||
| productionGzipExtensions: ['js', 'css'], | |||
| bundleAnalyzerReport: process.env.npm_config_report | |||
| } | |||
| } | |||
| @ -0,0 +1,4 @@ | |||
| 'use strict' | |||
| module.exports = { | |||
| NODE_ENV: '"production"' | |||
| } | |||
| @ -0,0 +1,7 @@ | |||
| 'use strict' | |||
| const merge = require('webpack-merge') | |||
| const devEnv = require('./dev.env') | |||
| module.exports = merge(devEnv, { | |||
| NODE_ENV: '"testing"' | |||
| }) | |||
| @ -0,0 +1 @@ | |||
| <!DOCTYPE html><html><head><meta http-equiv=Content-Language><meta name=viewport content="width=device-width,initial-scale=1"><meta http-equiv=X-UA-Compatible content="IE=edge,chrome=1"><meta http-equiv=Content-Type content="text/html;charset=utf-8"><title>智慧公寓管理系统</title><link href=/static/css/app.4235ce62fb9c140db2020a89f8ca28fc.css rel=stylesheet></head><body><div id=app></div><script type=text/javascript src=/static/js/manifest.18ec6b5593d846b34fd0.js></script><script type=text/javascript src=/static/js/vendor.9c8489941f3a5aa906bb.js></script><script type=text/javascript src=/static/js/app.509b36048f5247e714bd.js></script></body></html> | |||
| @ -0,0 +1,500 @@ | |||
| <html> | |||
| <head> | |||
| <!-- <meta http-equiv="Content-Language" content="zh-cn"> | |||
| <meta name="GENERATOR" content="Microsoft FrontPage 6.0"> | |||
| <meta name="ProgId" content="FrontPage.Editor.Document"> --> | |||
| <meta http-equiv="Content-Type" content="text/html; charset=gb2312"> | |||
| <title>USB IC 卡读卡器控件调用例程 V1.0</title> | |||
| <script language="javascript"> | |||
| var strls = ""; | |||
| var errorno = ""; | |||
| //控制字定义,控制字指定,控制字的含义请查看本公司网站提供的动态库说明 | |||
| //javascript无法自定义常量, 你可以使用变量代替 | |||
| var BLOCK0_EN = 0x01;//读第一块的(16个字节) | |||
| var BLOCK1_EN = 0x02;//读第二块的(16个字节) | |||
| var BLOCK2_EN = 0x04;//读第三块的(16个字节) | |||
| var NEEDSERIAL = 0x08;//仅读指定序列号的卡 | |||
| var EXTERNKEY = 0x10;//用明码认证密码,产品开发完成后,建议把密码放到设备的只写区,然后用该区的密码后台认证,这样谁都不知道密码是多少,需要这方面支持请联系 | |||
| var NEEDHALT = 0x20;//读/写完卡后立即休眠该卡,相当于这张卡不在感应区。要相重新操作该卡必要拿开卡再放上去 | |||
| var myctrlword = 0; | |||
| var myareano = 0; | |||
| var authmode = 0; | |||
| var mypiccserial = ""; | |||
| var mypicckey = ""; | |||
| var piccdata0_2 = ""; | |||
| var mypicckey_old = "";//旧密码 | |||
| var mypicckey_new = "";//新密码 | |||
| function readcard() | |||
| { | |||
| //指定控制字 | |||
| myctrlword=BLOCK0_EN + BLOCK1_EN + BLOCK2_EN + EXTERNKEY; | |||
| //指定区号 | |||
| myareano = 8; //指定为第8区 | |||
| //批定密码模式 | |||
| authmode = 1; //大于0表示用A密码认证,推荐用A密码认证 | |||
| //指定序列号,未知卡序列号时可指定为8个0 | |||
| mypiccserial="00000000"; | |||
| //指定密码,以下密码为厂家出厂密码 | |||
| mypicckey = "ffffffffffff"; | |||
| strls=IcCardReader.piccreadex(myctrlword, mypiccserial,myareano,authmode,mypicckey); | |||
| errorno = strls.substr(0,4); | |||
| if(errorno == "ER00"){ | |||
| beep() | |||
| } | |||
| switch(errorno) | |||
| { | |||
| case "ER08": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER09": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER10": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER11": | |||
| CardIDShower.value = "密码认证错误\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER12": | |||
| CardIDShower.value = "密码认证错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER13": | |||
| CardIDShower.value = "读卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("读卡错误"); | |||
| break; | |||
| case "ER14": | |||
| CardIDShower.value = "写卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("写卡错误"); | |||
| break; | |||
| case "ER21": | |||
| alert("没找到动态库"); | |||
| break; | |||
| case "ER22": | |||
| alert("动态库或驱动程序异常"); | |||
| break; | |||
| case "ER23": | |||
| alert("读卡器未插上或动态库或驱动程序异常"); | |||
| break; | |||
| case "ER24": | |||
| alert("操作超时,一般是动态库没有反应"); | |||
| break; | |||
| case "ER25": | |||
| alert("发送字数不够"); | |||
| break; | |||
| case "ER26": | |||
| alert("发送的CRC错"); | |||
| break; | |||
| case "ER27": | |||
| alert("接收的字数不够"); | |||
| break; | |||
| case "ER28": | |||
| alert("接收的CRC错"); | |||
| break; | |||
| case "ER29": | |||
| alert("函数输入参数格式错误,请仔细查看" ); | |||
| break; | |||
| default : | |||
| //读卡成功,其中ER00表示完全成功,ER01表示完全没读到卡数据,ER02表示仅读该卡的第一块成功,,ER02表示仅读该卡的第一二块成功,这是刷卡太快原因 | |||
| CardIDShower.value = "读卡成功" + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "该区第一块十六进制数据为:" + strls.substr(14,32) + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "该区第二块十六进制数据为:" + strls.substr(46,32) + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "该区第三块十六进制数据为:" + strls.substr(78,32) + "\r\n"; | |||
| break; | |||
| } | |||
| } | |||
| function writecard() | |||
| { | |||
| //指定控制字 | |||
| myctrlword=BLOCK0_EN + BLOCK1_EN + BLOCK2_EN + EXTERNKEY; | |||
| //指定区号 | |||
| myareano = 8; //指定为第8区 | |||
| //批定密码模式 | |||
| authmode = 1; //大于0表示用A密码认证,推荐用A密码认证 | |||
| //指定序列号,未知卡序列号时可指定为8个0 | |||
| mypiccserial="00000000"; | |||
| //指定密码,以下密码为厂家出厂密码 | |||
| mypicckey = "ffffffffffff"; | |||
| //指定写卡内容,长度为48个字节,其中每个字节以两个字符表示为十六进制数 | |||
| piccdata0_2 = "000102030405060708090A0B0C0D0E0F101112131415161718191A1B1C1D1E1F202122232425262728292A2B2C2D2E2F"; | |||
| strls=IcCardReader.piccwriteex(myctrlword, mypiccserial,myareano,authmode,mypicckey,piccdata0_2); | |||
| errorno = strls.substr(0,4); | |||
| switch(errorno) | |||
| { | |||
| case "ER08": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER09": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER10": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER11": | |||
| CardIDShower.value = "密码认证错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER12": | |||
| CardIDShower.value = "密码认证错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER13": | |||
| CardIDShower.value = "读卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("读卡错误"); | |||
| break; | |||
| case "ER14": | |||
| CardIDShower.value = "写卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| alert("写卡错误"); | |||
| break; | |||
| case "ER21": | |||
| alert("没找到动态库"); | |||
| break; | |||
| case "ER22": | |||
| alert("动态库或驱动程序异常"); | |||
| break; | |||
| case "ER23": | |||
| alert("读卡器未插上或动态库或驱动程序异常"); | |||
| break; | |||
| case "ER24": | |||
| alert("操作超时,一般是动态库没有反应"); | |||
| break; | |||
| case "ER25": | |||
| alert("发送字数不够"); | |||
| break; | |||
| case "ER26": | |||
| alert("发送的CRC错"); | |||
| break; | |||
| case "ER27": | |||
| alert("接收的字数不够"); | |||
| break; | |||
| case "ER28": | |||
| alert("接收的CRC错"); | |||
| break; | |||
| case "ER29": | |||
| alert("函数输入参数格式错误,请仔细查看"); | |||
| break; | |||
| default ://写卡成功,其中ER00表示完全成功,ER01表示完全没写到卡数据,ER02表示仅写该卡的第一块成功,,ER02表示仅写该卡的第一二块成功,这是刷卡太快原因 | |||
| CardIDShower.value = "写卡成功" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5,8) + "\r\n"; | |||
| break; | |||
| } | |||
| } | |||
| function writecarduid() { | |||
| //指定控制字 | |||
| myctrlword = BLOCK0_EN; | |||
| //批定密码模式 | |||
| authmode = 1; //大于0表示用A密码认证,推荐用A密码认证 | |||
| //指定序列号,未知卡序列号时可指定为8个0 | |||
| mypiccserial = "12345678"; | |||
| //指定密码,以下密码为厂家出厂密码 | |||
| mypicckey = "ffffffffffff"; | |||
| //指定写卡内容,长度为48个字节,其中每个字节以两个字符表示为十六进制数 | |||
| piccdata0 = "12345678000102030405060708090A0B0"; | |||
| strls = IcCardReader.piccwriteserial(myctrlword, mypiccserial,authmode, mypicckey, piccdata0); | |||
| errorno = strls.substr(0, 4); | |||
| switch (errorno) { | |||
| case "ER08": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER09": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER10": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER11": | |||
| CardIDShower.value = "密码认证错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5, 8) + "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER12": | |||
| CardIDShower.value = "密码认证错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5, 8) + "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER13": | |||
| CardIDShower.value = "读卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5, 8) + "\r\n"; | |||
| alert("读卡错误"); | |||
| break; | |||
| case "ER14": | |||
| CardIDShower.value = "写卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5, 8) + "\r\n"; | |||
| alert("写卡错误"); | |||
| break; | |||
| case "ER21": | |||
| alert("没找到动态库"); | |||
| break; | |||
| case "ER22": | |||
| alert("动态库或驱动程序异常"); | |||
| break; | |||
| case "ER23": | |||
| alert("读卡器未插上或动态库或驱动程序异常"); | |||
| break; | |||
| case "ER24": | |||
| alert("操作超时,一般是动态库没有反应"); | |||
| break; | |||
| case "ER25": | |||
| alert("发送字数不够"); | |||
| break; | |||
| case "ER26": | |||
| alert("发送的CRC错"); | |||
| break; | |||
| case "ER27": | |||
| alert("接收的字数不够"); | |||
| break; | |||
| case "ER28": | |||
| alert("接收的CRC错"); | |||
| break; | |||
| case "ER29": | |||
| alert("函数输入参数格式错误,请仔细查看"); | |||
| break; | |||
| default://写卡成功,其中ER00表示完全成功,ER01表示完全没写到卡数据,ER02表示仅写该卡的第一块成功,,ER02表示仅写该卡的第一二块成功,这是刷卡太快原因 | |||
| CardIDShower.value = "写UID号成功" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + "卡十六进制序列号为:" + strls.substr(5, 8) + "\r\n"; | |||
| break; | |||
| } | |||
| } | |||
| function changecardkeyex() | |||
| { | |||
| //指定控制字 | |||
| myctrlword= EXTERNKEY; | |||
| //指定区号 | |||
| myareano = 8; //指定为第8区 | |||
| //批定密码模式 | |||
| authmode = 1; //大于0表示用A密码认证,推荐用A密码认证 | |||
| //指定序列号,未知卡序列号时可指定为8个0 | |||
| mypiccserial="00000000"; | |||
| //旧密码 | |||
| mypicckey_old = "ffffffffffff"; | |||
| //新密码 | |||
| mypicckey_new = "ffffffffffffFF078069ffffffffffff";//其中最前面的ffffffffffff为A密码,中间的FF078069为访问控制位,最后面的ffffffffffff为B密码 | |||
| strls=IcCardReader.piccchangesinglekeyex(myctrlword, mypiccserial,myareano,authmode,mypicckey_old,mypicckey_new) | |||
| errorno = strls.substr(0,4); | |||
| if(errorno == "ER00") | |||
| { | |||
| CardIDShower.value = CardIDShower.value + "修改密码成功,卡序列为:" + strls.substr(strls.length - 8,8) + "\r\n"; | |||
| } | |||
| else if(errorno == "ER08") | |||
| { | |||
| alert("寻不到卡"); | |||
| } | |||
| else | |||
| { | |||
| CardIDShower.value = CardIDShower.value + "修改密码:错误:" + errorno + "\r\n"; | |||
| } | |||
| } | |||
| function getdevicenumber() | |||
| { | |||
| strls=IcCardReader.pcdgetdevicenumber(); | |||
| errorno = strls.substr(0,4); | |||
| if(errorno == "ER00") | |||
| { | |||
| CardIDShower.value = CardIDShower.value + "设备硬件号为:" + strls.substr(strls.length - 8,8) + "\r\n"; | |||
| } | |||
| } | |||
| function beep() | |||
| { | |||
| IcCardReader.pcdbeep(100);//100表示响100毫秒 | |||
| } | |||
| function clears1() | |||
| { | |||
| //alert("abc"); | |||
| CardIDShower.value = ""; | |||
| } | |||
| window.onerror=function() | |||
| { | |||
| alert("不好意思,出错了!"); | |||
| return true;//屏蔽系统事件 | |||
| } | |||
| </script> | |||
| <object classid="clsid:05782014-9FF7-468C-BE96-8EDC73084202" id="IcCardReader" viewastext | |||
| width="0" height="0"> | |||
| </object> | |||
| </head> | |||
| <body> | |||
| <table border="0" style="border-collapse: collapse" width="200" height="100"> | |||
| <tr> | |||
| <td width="160"> | |||
| <textarea rows="2" cols="40" id="CardIDShower" name="S1"></textarea> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <!-- <input type="button" value="读卡" onclick="javascript:readcard()" /> --> | |||
| <button onclick="javascript:readcard()">读卡</button> | |||
| </td> | |||
| </tr> | |||
| <!-- <tr> | |||
| <td width="160"> | |||
| <input type="button" value=" 写 卡 " onclick="javascript:writecard()" /> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <input type="button" value="蜂鸣器响" onclick="javascript:beep()" /> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <input type="button" value="更改卡密码" onclick="javascript:changecardkeyex()" /> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <input type="button" value="读取读写器硬件号" onclick="javascript:getdevicenumber()" /> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <input type="button" value="写卡UID号为12345678" onclick="javascript:writecarduid()" /> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <input type="button" value="清空提示" onclick="javascript:clears1()" /> | |||
| </td> | |||
| </tr> --> | |||
| </table> | |||
| <!-- <p> | |||
| <font style="font-size: 9pt">提示:</font></p> | |||
| <p> | |||
| <font style="font-size: 9pt"> 1、运行本例程前,需先注册 IcCardReader 控件,注册方法为:“开始”->“运行”->输入“regsvr32 ***\IcCardReader.ocx”->“确定”。其中“***”为控件所在路径。</font></p> | |||
| <p> | |||
| <font style="font-size: 9pt"> 2、在IE的Internet 属性设定,让浏览器允许运行 ActiveX 控件。</font></p> | |||
| <p> | |||
| <font style="font-size: 9pt"> 3、如果是在服务器端运行本网页,还需在IE的Internet 属性中设定,将服务器网址设为可信站点,否则网页无权运行本地控件。</font></p> --> | |||
| </body> | |||
| </html> | |||
| @ -0,0 +1,213 @@ | |||
| <template> | |||
| <div> | |||
| <!-- <object | |||
| classid="clsid:05782014-9FF7-468C-BE96-8EDC73084202" | |||
| id="IcCardReader" | |||
| viewastext | |||
| width="0" | |||
| height="0" | |||
| ></object> --> | |||
| <table | |||
| border="0" | |||
| style="border-collapse: collapse" | |||
| width="200" | |||
| height="100" | |||
| > | |||
| <tr> | |||
| <td width="160"> | |||
| <textarea rows="2" cols="40" id="CardIDShower" name="S1"></textarea> | |||
| </td> | |||
| </tr> | |||
| <tr> | |||
| <td width="160"> | |||
| <input | |||
| type="button" | |||
| value=" 读 卡 " | |||
| @click="readcard()" | |||
| /> | |||
| </td> | |||
| </tr> | |||
| </table> | |||
| </div> | |||
| </template> | |||
| <script> | |||
| var strls = ""; | |||
| var errorno = ""; | |||
| //控制字定义,控制字指定,控制字的含义请查看本公司网站提供的动态库说明 | |||
| //javascript无法自定义常量, 你可以使用变量代替 | |||
| var BLOCK0_EN = 0x01;//读第一块的(16个字节) | |||
| var BLOCK1_EN = 0x02;//读第二块的(16个字节) | |||
| var BLOCK2_EN = 0x04;//读第三块的(16个字节) | |||
| var NEEDSERIAL = 0x08;//仅读指定序列号的卡 | |||
| var EXTERNKEY = 0x10;//用明码认证密码,产品开发完成后,建议把密码放到设备的只写区,然后用该区的密码后台认证,这样谁都不知道密码是多少,需要这方面支持请联系 | |||
| var NEEDHALT = 0x20;//读/写完卡后立即休眠该卡,相当于这张卡不在感应区。要相重新操作该卡必要拿开卡再放上去 | |||
| var myctrlword = 0; | |||
| var myareano = 0; | |||
| var authmode = 0; | |||
| var mypiccserial = ""; | |||
| var mypicckey = ""; | |||
| var piccdata0_2 = ""; | |||
| var mypicckey_old = "";//旧密码 | |||
| var mypicckey_new = "";//新密码 | |||
| export default { | |||
| name: "index", | |||
| data() { | |||
| return {}; | |||
| }, | |||
| methods: { | |||
| readcard() { | |||
| //指定控制字 | |||
| myctrlword = BLOCK0_EN + BLOCK1_EN + BLOCK2_EN + EXTERNKEY; | |||
| //指定区号 | |||
| myareano = 8; //指定为第8区 | |||
| //批定密码模式 | |||
| authmode = 1; //大于0表示用A密码认证,推荐用A密码认证 | |||
| //指定序列号,未知卡序列号时可指定为8个0 | |||
| mypiccserial = "00000000"; | |||
| //指定密码,以下密码为厂家出厂密码 | |||
| mypicckey = "ffffffffffff"; | |||
| var CardIDShower = document.getElementById('CardIDShower') | |||
| strls = window.IcCardReader.piccreadex( | |||
| myctrlword, | |||
| mypiccserial, | |||
| myareano, | |||
| authmode, | |||
| mypicckey | |||
| ); | |||
| errorno = strls.substr(0, 4); | |||
| if (errorno == "ER00") { | |||
| this.beep(); | |||
| } | |||
| switch (errorno) { | |||
| case "ER08": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER09": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER10": | |||
| alert("寻不到卡"); | |||
| break; | |||
| case "ER11": | |||
| CardIDShower.value = "密码认证错误\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + | |||
| "卡十六进制序列号为:" + | |||
| strls.substr(5, 8) + | |||
| "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER12": | |||
| CardIDShower.value = "密码认证错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + | |||
| "卡十六进制序列号为:" + | |||
| strls.substr(5, 8) + | |||
| "\r\n"; | |||
| alert("密码认证错误"); | |||
| break; | |||
| case "ER13": | |||
| CardIDShower.value = "读卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + | |||
| "卡十六进制序列号为:" + | |||
| strls.substr(5, 8) + | |||
| "\r\n"; | |||
| alert("读卡错误"); | |||
| break; | |||
| case "ER14": | |||
| CardIDShower.value = "写卡错误" + "\r\n"; | |||
| CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + | |||
| "卡十六进制序列号为:" + | |||
| strls.substr(5, 8) + | |||
| "\r\n"; | |||
| alert("写卡错误"); | |||
| break; | |||
| case "ER21": | |||
| alert("没找到动态库"); | |||
| break; | |||
| case "ER22": | |||
| alert("动态库或驱动程序异常"); | |||
| break; | |||
| case "ER23": | |||
| alert("读卡器未插上或动态库或驱动程序异常"); | |||
| break; | |||
| case "ER24": | |||
| alert("操作超时,一般是动态库没有反应"); | |||
| break; | |||
| case "ER25": | |||
| alert("发送字数不够"); | |||
| break; | |||
| case "ER26": | |||
| alert("发送的CRC错"); | |||
| break; | |||
| case "ER27": | |||
| alert("接收的字数不够"); | |||
| break; | |||
| case "ER28": | |||
| alert("接收的CRC错"); | |||
| break; | |||
| case "ER29": | |||
| alert("函数输入参数格式错误,请仔细查看"); | |||
| break; | |||
| default: | |||
| //读卡成功,其中ER00表示完全成功,ER01表示完全没读到卡数据,ER02表示仅读该卡的第一块成功,,ER02表示仅读该卡的第一二块成功,这是刷卡太快原因 | |||
| CardIDShower.value = "读卡成功" + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + strls + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "其中错误号为:" + errorno + "\r\n"; | |||
| CardIDShower.value = | |||
| CardIDShower.value + | |||
| "卡十六进制序列号为:" + | |||
| strls.substr(5, 8) + | |||
| "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "该区第一块十六进制数据为:" + strls.substr(14,32) + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "该区第二块十六进制数据为:" + strls.substr(46,32) + "\r\n"; | |||
| // CardIDShower.value = CardIDShower.value + "该区第三块十六进制数据为:" + strls.substr(78,32) + "\r\n"; | |||
| break; | |||
| } | |||
| }, | |||
| beep() { | |||
| IcCardReader.pcdbeep(100); //100表示响100毫秒 | |||
| }, | |||
| clears1() { | |||
| //alert("abc"); | |||
| CardIDShower.value = ""; | |||
| }, | |||
| }, | |||
| }; | |||
| </script> | |||
| <style> | |||
| </style> | |||
| @ -0,0 +1,3 @@ | |||
| %Windir%\SysWOW64\regsvr32 %Windir%\System32\IcCardReader.ocx -u | |||
| %Windir%\SysWOW64\regsvr32 %Windir%\SysWOW64\IcCardReader.ocx -u | |||
| pause | |||
| @ -0,0 +1,3 @@ | |||
| Copy %~dp0\IcCardReader.ocx %Windir%\SysWOW64\ | |||
| %Windir%\SysWOW64\regsvr32 %Windir%\SysWOW64\IcCardReader.ocx | |||
| pause | |||
| @ -0,0 +1,3 @@ | |||
| Copy %~dp0\IcCardReader.ocx %Windir%\System32\ | |||
| %Windir%\SysWOW64\regsvr32 %Windir%\System32\IcCardReader.ocx | |||
| pause | |||
| @ -0,0 +1,3 @@ | |||
| 如果你的系统是32位的x86系统,请运行 复制及注册IcCardReader控件win_32.bat | |||
| 或者 | |||
| 如果你的系统是64位的x64系统,请运行 复制及注册IcCardReader控件win10_64.bat | |||
| @ -0,0 +1,6 @@ | |||
| 1. 安装chrome最新版本 | |||
| 2. chrome 访问 chrome://extensions/ | |||
| 3. 打开开发者模式 | |||
| 4. 加载已解压的扩展程序 选中mafp_serial\plugin文件夹 | |||
| 5. chrome 访问 chrome://apps/ 打开MAFP Serial, 等待出现日志 found valid device XXX | |||
| 6. chrome 打开index.html 进行注册操作 | |||
| @ -0,0 +1,6 @@ | |||
| 1. 安装chrome最新版本 | |||
| 2. chrome 访问 chrome://extensions/ | |||
| 3. 打开开发者模式 | |||
| 4. 加载已解压的扩展程序 选中mafp_serial\plugin文件夹 | |||
| 5. chrome 访问 chrome://apps/ 打开MAFP Serial, 等待出现日志 found valid device XXX | |||
| 6. chrome 打开index.html 进行注册操作 | |||
| @ -0,0 +1,37 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | |||
| <!-- Generator: Adobe Illustrator 19.2.1, SVG Export Plug-In . SVG Version: 6.00 Build 0) --> | |||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | |||
| viewBox="0 0 180 320" style="enable-background:new 0 0 180 320;" xml:space="preserve"> | |||
| <style type="text/css"> | |||
| .demo__background-path { | |||
| fill: none; | |||
| stroke: #aaa; | |||
| stroke-linecap: round; | |||
| stroke-width: 2px; | |||
| } | |||
| </style> | |||
| <path class="demo__background-path" d="M46.1,214.3c0,0-4.7-15.6,4.1-33.3"/> | |||
| <path class="demo__background-path" d="M53.5,176.8c0,0,18.2-30.3,57.5-13.7"/> | |||
| <path class="demo__background-path" d="M115.8,166.5c0,0,19.1,8.7,19.6,38.4"/> | |||
| <path class="demo__background-path" d="M47.3,221c0,0,3.1-2.1,4.1-4.6s-5.7-20.2,7-36.7c8.5-11,22.2-19,37.9-15.3"/> | |||
| <path class="demo__background-path" d="M102.2,165.4c10.2,2.7,19.5,10.4,23.5,20.2c6.2,15.2,4.9,27.1,4.1,39.4"/> | |||
| <path class="demo__background-path" d="M51.1,226.5c3.3-2.7,5.1-6.3,5.7-10.5c0.5-4-0.3-7.7-0.3-11.7"/> | |||
| <path class="demo__background-path" d="M129.3,200.1"/> | |||
| <path class="demo__background-path" d="M56.3,197.9c3.1-16.8,17.6-29.9,35.1-28.8c17.7,1.1,30.9,14.9,32.8,32.2"/> | |||
| <path class="demo__background-path" d="M124.2,207.9c0.5,9.3,0.5,18.7-2.1,27.7"/> | |||
| <path class="demo__background-path" d="M54.2,231.1c2.1-2.6,4.6-5.1,6.3-8c4.2-6.8,0.9-14.8,1.5-22.3c0.5-7.1,3.4-16.3,10.4-19.7"/> | |||
| <path class="demo__background-path" d="M77.9,178.2c9.3-5.1,22.9-4.7,30.5,3.3"/> | |||
| <path class="demo__background-path" d="M113,186.5c0,0,13.6,18.9,1,54.8"/> | |||
| <path class="demo__background-path" d="M57.3,235.2c0,0,5.7-3.8,9-12.3"/> | |||
| <path class="demo__background-path" d="M111.7,231.5c0,0-4.1,11.5-5.7,13.6"/> | |||
| <path class="demo__background-path" d="M61.8,239.4c9.3-8.4,12.7-19.7,11.8-31.9c-0.9-12.7,3.8-20.6,18.5-21.2"/> | |||
| <path class="demo__background-path" d="M97.3,188.1c8.4,2.7,11,13,11.3,20.8c0.4,11.8-2.5,23.7-7.9,34.1c-0.1,0.1-0.1,0.2-0.2,0.3 | |||
| c-0.4,0.8-0.8,1.5-1.2,2.3c-0.5,0.8-1,1.7-1.5,2.5"/> | |||
| <path class="demo__background-path" d="M66.2,242.5c0,0,15.3-11.1,13.6-34.9"/> | |||
| <path class="demo__background-path" d="M78.7,202.5c1.5-4.6,3.8-9.4,8.9-10.6c13.5-3.2,15.7,13.3,14.6,22.1"/> | |||
| <path class="demo__background-path" d="M102.2,219.7c0,0-1.7,15.6-10.5,28.4"/> | |||
| <path class="demo__background-path" d="M72,244.9c0,0,8.8-9.9,9.9-15.7"/> | |||
| <path class="demo__background-path" d="M84.5,223c0.3-2.6,0.5-5.2,0.7-7.8c0.1-2.1,0.2-4.6-0.1-6.8c-0.3-2.2-1.1-4.3-0.9-6.5c0.5-4.4,7.2-6.9,10.1-3.1c1.7,2.2,1.7,5.3,1.9,7.9c0.4,3.8,0.3,7.6,0,11.4c-1,10.8-5.4,21-11.5,29.9"/> | |||
| <path class="demo__background-path" d="M90,201.2c0,0,4.6,28.1-11.4,45.2"/> | |||
| <path class="demo__background-path" d="M67.3,219C65,188.1,78,180.1,92.7,180.3c18.3,2,23.7,18.3,20,46.7"/> | |||
| </svg> | |||
| @ -0,0 +1,290 @@ | |||
| <!DOCTYPE html> | |||
| <head> | |||
| <link type="text/css" rel="stylesheet" href="jquery.toast.css"> | |||
| <style type="text/css"> | |||
| *, *:before, *:after { | |||
| box-sizing: border-box; | |||
| margin: 0; | |||
| padding: 0; | |||
| } | |||
| .log-text { | |||
| font-size: 14px; | |||
| padding: 10px 0 0 10px; | |||
| color: #666; | |||
| text-align: center; | |||
| margin-bottom: 30px; | |||
| } | |||
| .fprint-container { | |||
| width: 300px; | |||
| height: 252px; | |||
| margin: 0 auto; | |||
| position: relative; | |||
| } | |||
| .demo__fprint { | |||
| width: 300px; | |||
| height: 260px; | |||
| overflow: visible; | |||
| display: block; | |||
| } | |||
| .demo__fprint_bg { | |||
| position: absolute; | |||
| top: 0; | |||
| left: 0; | |||
| width: 297px; | |||
| height: 260px; | |||
| z-index: -1; | |||
| background: url(./fprintBackground.svg) no-repeat; | |||
| background-position: 78px -260px; | |||
| background-size: auto; | |||
| /* background-position: -100px 0; */ | |||
| /* background-size: cover; */ | |||
| } | |||
| .demo__fprint_bg img{position: absolute;width: 147px;top: 50%; | |||
| left: 75px; | |||
| opacity: .9; | |||
| transform: translateY(-50%);} | |||
| .demo__fprint_bg.leave { | |||
| opacity: 0.5; | |||
| } | |||
| .demo__fprint_bg.down { | |||
| opacity: 1; | |||
| } | |||
| .demo__fprint-path { | |||
| stroke-width: 2.5px; | |||
| stroke-linecap: round; | |||
| fill: none; | |||
| stroke: white; | |||
| visibility: hidden; | |||
| transition: opacity 0.5s ease; | |||
| will-change: stroke-dashoffset, stroke-dasharray; | |||
| -webkit-transform: translateZ(0); | |||
| transform: translateZ(0); | |||
| } | |||
| .demo__fprint-path--pinkish { | |||
| stroke: #a94a8c; | |||
| } | |||
| .demo__fprint-path--purplish { | |||
| stroke: #8742cc; | |||
| } | |||
| .demo__fprint-path#demo__elastic-path { | |||
| will-change: opacity; | |||
| opacity: 1; | |||
| } | |||
| .demo__hidden-path { | |||
| display: none; | |||
| } | |||
| .demo__ending-path { | |||
| fill: none; | |||
| stroke-width: 2.5px; | |||
| stroke-dasharray: 60 1000; | |||
| stroke-dashoffset: 61; | |||
| stroke-linecap: round; | |||
| will-change: stroke-dashoffset, stroke-dasharray, opacity; | |||
| -webkit-transform: translateZ(0); | |||
| transform: translateZ(0); | |||
| transition: stroke-dashoffset 1s ease, stroke-dasharray 0.5s linear, opacity 0.75s ease; | |||
| } | |||
| .demo__ending-path--pinkish { | |||
| stroke: #a94a8c; | |||
| } | |||
| .demo__ending-path--purplish { | |||
| stroke: #8742cc; | |||
| } | |||
| .buttons { | |||
| width: 500px; | |||
| display: flex; | |||
| margin: auto; | |||
| align-items: center; | |||
| justify-content: center; | |||
| } | |||
| #enroll-btn { | |||
| width: 140px; | |||
| } | |||
| </style> | |||
| </head> | |||
| <body> | |||
| <div class="fprint-container"> | |||
| <div class="demo__fprint_bg leave"><img src="./fprintBackground.svg" alt=""></div> | |||
| <svg class="demo__fprint" viewBox="0 0 180 320"> | |||
| <defs> | |||
| <linearGradient id="linear" x1="0%" y1="0%" x2="100%" y2="0%"> | |||
| <stop offset="0%" stop-color="#8742cc"></stop> | |||
| <stop offset="100%" stop-color="#a94a8c"></stop> | |||
| </linearGradient> | |||
| </defs> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--pinkish" d="M46.1,214.3c0,0-4.7-15.6,4.1-33.3" style="stroke-dasharray: 34.235, 36.235; stroke-dashoffset: 35.235px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--purplish" d="M53.5,176.8c0,0,18.2-30.3,57.5-13.7" style="stroke-dasharray: 64.5064, 66.5064; stroke-dashoffset: 65.5064px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--pinkish" d="M115.8,166.5c0,0,19.1,8.7,19.6,38.4" style="stroke-dasharray: 45.1842, 47.1842; stroke-dashoffset: 46.1842px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--pinkish" d="M47.3,221c0,0,3.1-2.1,4.1-4.6s-5.7-20.2,7-36.7c8.5-11,22.2-19,37.9-15.3" style="stroke-dasharray: 88.0272, 90.0272; stroke-dashoffset: 89.0272px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--pinkish" d="M102.2,165.4c10.2,2.7,19.5,10.4,23.5,20.2c6.2,15.2,4.9,27.1,4.1,39.4" style="stroke-dasharray: 72.1106, 74.1106; stroke-dashoffset: 73.1106px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--purplish" d="M51.1,226.5c3.3-2.7,5.1-6.3,5.7-10.5c0.5-4-0.3-7.7-0.3-11.7" style="stroke-dasharray: 23.9625, 25.9625; stroke-dashoffset: 24.9625px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--purplish" d="M56.3,197.9c3.1-16.8,17.6-29.9,35.1-28.8c17.7,1.1,30.9,14.9,32.8,32.2" style="stroke-dasharray: 99.6723, 101.672; stroke-dashoffset: 100.672px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--purplish" d="M124.2,207.9c0.5,9.3,0.5,18.7-2.1,27.7" style="stroke-dasharray: 27.9142, 29.9142; stroke-dashoffset: 28.9142px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--pinkish" d="M54.2,231.1c2.1-2.6,4.6-5.1,6.3-8c4.2-6.8,0.9-14.8,1.5-22.3c0.5-7.1,3.4-16.3,10.4-19.7" style="stroke-dasharray: 56.1771, 58.1771; stroke-dashoffset: 57.1771px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--purplish" d="M77.9,178.2c9.3-5.1,22.9-4.7,30.5,3.3" style="stroke-dasharray: 32.824, 34.824; stroke-dashoffset: 33.824px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--purplish" d="M113,186.5c0,0,13.6,18.9,1,54.8" style="stroke-dasharray: 56.5353, 58.5353; stroke-dashoffset: 57.5353px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--pinkish" d="M57.3,235.2c0,0,5.7-3.8,9-12.3" style="stroke-dasharray: 15.4365, 17.4365; stroke-dashoffset: 16.4365px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--pinkish" d="M111.7,231.5c0,0-4.1,11.5-5.7,13.6" style="stroke-dasharray: 14.7687, 16.7687; stroke-dashoffset: 15.7687px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--purplish" d="M61.8,239.4c9.3-8.4,12.7-19.7,11.8-31.9c-0.9-12.7,3.8-20.6,18.5-21.2" style="stroke-dasharray: 67.9598, 69.9598; stroke-dashoffset: 68.9598px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--pinkish" d="M97.3,188.1c8.4,2.7,11,13,11.3,20.8c0.4,11.8-2.5,23.7-7.9,34.1c-0.1,0.1-0.1,0.2-0.2,0.3 | |||
| c-0.4,0.8-0.8,1.5-1.2,2.3c-0.5,0.8-1,1.7-1.5,2.5" style="stroke-dasharray: 66.4444, 68.4444; stroke-dashoffset: 67.4444px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--purplish" d="M66.2,242.5c0,0,15.3-11.1,13.6-34.9" style="stroke-dasharray: 38.917, 40.917; stroke-dashoffset: 39.917px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--pinkish" d="M78.7,202.5c1.5-4.6,3.8-9.4,8.9-10.6c13.5-3.2,15.7,13.3,14.6,22.1" style="stroke-dasharray: 46.1932, 48.1932; stroke-dashoffset: 47.1932px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--pinkish" d="M102.2,219.7c0,0-1.7,15.6-10.5,28.4" style="stroke-dasharray: 30.5375, 32.5375; stroke-dashoffset: 31.5375px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-backwards demo__fprint-path--pinkish" d="M72,244.9c0,0,8.8-9.9,9.9-15.7" style="stroke-dasharray: 18.7134, 20.7134; stroke-dashoffset: 19.7134px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--pinkish" d="M84.5,223c0.3-2.6,0.5-5.2,0.7-7.8c0.1-2.1,0.2-4.6-0.1-6.8c-0.3-2.2-1.1-4.3-0.9-6.5c0.5-4.4,7.2-6.9,10.1-3.1 | |||
| c1.7,2.2,1.7,5.3,1.9,7.9c0.4,3.8,0.3,7.6,0,11.4c-1,10.8-5.4,21-11.5,29.9" style="stroke-dasharray: 86.4028, 88.4028; stroke-dashoffset: 87.4028px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--removes-forwards demo__fprint-path--purplish" d="M90,201.2c0,0,4.6,28.1-11.4,45.2" style="stroke-dasharray: 48.0853, 50.0853; stroke-dashoffset: 49.0853px; visibility: visible;"></path> | |||
| <path class="demo__fprint-path demo__fprint-path--pinkish" id="demo__elastic-path" d="M67.3,219C65,188.1,78,180.1,92.7,180.3c18.3,2,23.7,18.3,20,46.7" style="stroke-dasharray: 111.304, 113.304; stroke-dashoffset: 112.304px; visibility: visible;"></path> | |||
| <!-- <line id="demo__straight-path" x1="0" y1="151.3" x2="180" y2="151.3"></line> | |||
| <path class="demo__hidden-path" id="demo__arc-to-top" d="M0,148.4c62.3-13.5,122.3-13.5,180,0"></path> | |||
| <path class="demo__hidden-path" id="demo__curve" d="M0,140.2c13.1-10.5,34.7-17,48.5-4.1c5.5,5.2,7.6,12.1,9.2,19.2c2.4,10.5,4.3,21,7.2,31.4c2.4,8.6,4.3,19.6,10.4,26.7c4.3,5,17.7,13.4,23.1,4.8c5.9-9.4,6.8-22.5,9.7-33c4.9-17.8,13-14.6,15.7-14.6c1.8,0,9,2.3,15.4,5.4c6.2,3,11.9,7.7,17.9,11.2c7,4.1,16.5,9.2,22.8,6.6"></path> | |||
| <path class="demo__ending-path demo__ending-path--pinkish" d="M48.4,220c-5.8,4.2-6.9,11.5-7.6,18.1c-0.8,6.7-0.9,14.9-9.9,12.4c-9.1-2.5-14.7-5.4-19.9-13.4c-3.4-5.2-0.4-12.3,2.3-17.2c3.2-5.9,6.8-13,14.5-11.6c3.5,0.6,7.7,3.4,4.5,7.1"></path> | |||
| <path class="demo__ending-path demo__ending-path--pinkish" d="M57.3,235.2c-14.4,9.4-10.3,19.4-17.8,21.1c-5.5,1.3-8.4-7.8-13.8-4.2c-2.6,1.7-5.7,7.7-4.6,10.9c0.7,2,4.1,2,5.8,2.4c3,0.7,8.4,3,7.6,7.2c-0.6,3-5,5.3-2.4,8.7c1.8,2.2,4.7,1.1,6.9,0.3c11.7-4.3,14.5,0.8,16.5,0.9"></path> | |||
| <path class="demo__ending-path demo__ending-path--purplish" d="M79,246c-1.8,2.4-4.9,2.9-7.6,3.2c-2.7,0.3-5.8-0.8-7.7,1.6c-2.9,3.3,0.7,8.2-1.2,12c-1.5,2.8-4.5,2.4-6.9,1.3c-10.1-4.7-33.2-17.5-38.1-2.5c-1.1,3.4-1.9,7.5-1.3,11c0.6,4,5.6,7.9,7.7,2.3c0.8-2.1,3.1-8.6-1-8.9"></path> | |||
| <path class="demo__ending-path demo__ending-path--pinkish" d="M91.8,248c0,0-3.9,6.4-6.2,9.2c-3.8,4.5-7.9,8.9-11.2,13.8c-1.9,2.8-4.4,6.4-3.7,10c0.9,5.2,4.7,12.5,9.7,14.7c5.2,2.2,15.9-4.7,13.1-10.8c-1.4-3-6.3-7.9-10-7.2c-1,0.2-1.8,1-2,2"></path> | |||
| <path class="demo__ending-path demo__ending-path--purplish" d="M114.8,239.4c-2.7,6.1-8.3,12.8-7.8,19.8c0.3,4.6,3.8,7.4,7.8,9.1c8.9,3.8,19.7,0.4,28.6-1.3c8.8-1.7,19.7-3.2,23.7,6.7c2.8,6.8,6.1,14.7,4.4,22.2"></path> | |||
| <path class="demo__ending-path demo__ending-path--pinkish" d="M129.9,224.2c-0.4,7.5-3.1,18,0.7,25c2.8,5.1,14.3,6.3,19.5,7.4c3.7,0.7,8.7,2.2,12-0.5c6.7-5.4,11.1-13.7,14.1-21.6c3.1-8-4.4-12.8-11.1-14.5c-5-1.3-19.1-0.7-21-6.7c-0.9-2.8,1.8-5.9,3.4-7.9"></path> | |||
| <path class="demo__under-curve" d="M0,140.2c13.1-10.5,34.7-17,48.5-4.1c5.5,5.2,7.6,12.1,9.2,19.2c2.4,10.5,4.3,21,7.2,31.4c2.4,8.6,4.3,19.6,10.4,26.7c4.3,5,17.7,13.4,23.1,4.8c5.9-9.4,6.8-22.5,9.7-33c4.9-17.8,13-14.6,15.7-14.6c1.8,0,9,2.3,15.4,5.4c6.2,3,11.9,7.7,17.9,11.2c7,4.1,16.5,9.2,23.8,6.6l0,125.5l-181,0l0,-179.8"></path> --> | |||
| </svg> | |||
| </div> | |||
| <div class="log-text">loading...</div> | |||
| <div class="buttons"> | |||
| <button id="enroll-btn">GX1000采集指纹</button> | |||
| </div> | |||
| <audio src="./press.mp3" id="press"></audio> | |||
| <audio src="./up.mp3" id="up"></audio> | |||
| <script src='jquery.min.js'></script> | |||
| <script src='jquery.toast.min.js'></script> | |||
| <script src='index.js'></script> | |||
| <script src='mafp_sdk.js'></script> | |||
| <script type="text/javascript"> | |||
| function log(msg) { | |||
| document.querySelector(".log-text").innerHTML = msg; | |||
| } | |||
| onload = function() { | |||
| const EnrollCount = 6; | |||
| const EnrollTimeout = 30 * 1000; | |||
| setFprintProgress(0); | |||
| log("初始化..."); | |||
| mafp.init({ | |||
| enrollCount: EnrollCount, | |||
| enrollTimeout: EnrollTimeout | |||
| }).then(() => { | |||
| log("初始化成功."); | |||
| }).catch(e => { | |||
| log("初始化失败."); | |||
| }); | |||
| var up = document.getElementById("up"); | |||
| var press = document.getElementById("press"); | |||
| $("#enroll-btn").click(() => { | |||
| if (mafp.isWorking) { | |||
| mafp.cancelEnroll(); | |||
| $("#enroll-btn").text("注册"); | |||
| log("注册取消"); | |||
| }else { | |||
| log("注册准备中..."); | |||
| mafp.startEnroll((status) => { | |||
| // console.log(status) | |||
| if (status.err) { | |||
| let logText = "注册出错"; | |||
| switch(status.err) { | |||
| case ErrorReceiveDataErr: | |||
| logText = "数据包传输错误"; | |||
| break; | |||
| case ErrorEnrollTimeout: | |||
| logText = "注册超时"; | |||
| break; | |||
| case ErrorDeviceNotFound: | |||
| logText = "设备未找到"; | |||
| break; | |||
| case ErrorEmptyFail: | |||
| logText = "清空数据失败"; | |||
| break; | |||
| default: | |||
| logText = "注册出错:" + status.err; | |||
| } | |||
| log(logText); | |||
| $.toast({text: logText, position: 'top-center', icon: "error", hideAfter: 5000}) | |||
| $("#enroll-btn").text("注册"); | |||
| return; | |||
| }else { | |||
| if(status.state == STATE_WAIT_FINGER_DOWN) { | |||
| up.pause() | |||
| press.play() | |||
| log("请按手指"); | |||
| } | |||
| else if(status.state == STATE_WAIT_FINGER_LEAVE) { | |||
| up.play() | |||
| log("请抬起手指"); | |||
| } | |||
| else if(status.state == STATE_EXTRACT_TEMPLATE) { | |||
| log("提取特征点..."); | |||
| } | |||
| else if(status.state == STATE_DOWNLOAD_TEMPLATE) { | |||
| log("模板传输中..."); | |||
| } | |||
| else if (status.state == STATE_FINGER_DOWN) { | |||
| $(".demo__fprint_bg").addClass("down"); | |||
| }else if (status.state == STATE_FINGER_LEAVE) { | |||
| $(".demo__fprint_bg").removeClass("down"); | |||
| }else if (status.state >= STATE_EXTRACT_TEMPLATE_FAIL) { | |||
| let errText = "提取特征点失败"; | |||
| if (status.state == STATE_EXTRACT_TEMPLATE_DIRTY) { | |||
| errText = "提取特征点失败, 指纹图像太乱"; | |||
| }else if (status.state == STATE_EXTRACT_TEMPLATE_POINTS_FEW) { | |||
| errText = "提取特征点失败, 特征点太少"; | |||
| }else if (status.state == STATE_EXTRACT_TEMPLATE_DIRTY) { | |||
| errText = "提取特征点失败, 合并失败"; | |||
| } | |||
| $.toast({text: errText, position: 'top-center', hideAfter: 5000}) | |||
| } | |||
| } | |||
| if (status.data) { | |||
| //注册完成 模板数据 二进制数据 | |||
| //可封装成 multipart/form-data 上传服务端 | |||
| console.log(status.data,'指纹数据') | |||
| var templateData = new Uint8Array(status.data); | |||
| localStorage.setItem('fingerprint',status.data) | |||
| // console.log(status.data); | |||
| log("注册完成"); | |||
| $("#enroll-btn").text("注册"); | |||
| } | |||
| if (status.data || status.step == EnrollCount && (status.state == 5 || status.state == 6)) { | |||
| setFprintProgress(1); | |||
| }else { | |||
| setFprintProgress((status.step - 1) / EnrollCount); | |||
| } | |||
| }); | |||
| $("#enroll-btn").text("取消"); | |||
| } | |||
| }) | |||
| } | |||
| </script> | |||
| </body> | |||
| @ -0,0 +1,104 @@ | |||
| const TIME_TO_FILL_FPRINT = 2000; //ms | |||
| const $fprintPaths = document.querySelectorAll('.demo__fprint-path'); | |||
| const $endingPaths = document.querySelectorAll('.demo__ending-path'); | |||
| const fprintPathSelector = '.demo__fprint-path'; | |||
| let fprintPaths = []; | |||
| let lastRafCallTimestamp = 0; | |||
| let curFprintPathsOffset = -1; | |||
| let fprintProgressionDirection = 1; | |||
| let isFprintAnimationInProgress = false; | |||
| let isFprintAnimationOver = false; | |||
| let fprintTick = 0; | |||
| let currentFprintProgresss = 0; | |||
| function offsetAllFprintPaths(ratio) { | |||
| fprintPaths.forEach(path => path.offset(ratio)); | |||
| } | |||
| function fprintFrame(timestamp) { | |||
| if (!lastRafCallTimestamp) { | |||
| lastRafCallTimestamp = timestamp; | |||
| window.requestAnimationFrame(fprintFrame); | |||
| return; | |||
| } | |||
| let diff = timestamp - lastRafCallTimestamp; | |||
| fprintTick = (diff / TIME_TO_FILL_FPRINT) * 2; | |||
| lastRafCallTimestamp = timestamp; | |||
| curFprintPathsOffset += fprintTick * fprintProgressionDirection; | |||
| offsetAllFprintPaths(curFprintPathsOffset); | |||
| if (curFprintPathsOffset >= -1 && curFprintPathsOffset <= 1) { | |||
| window.requestAnimationFrame(fprintFrame); | |||
| } | |||
| else if (curFprintPathsOffset > 1) { | |||
| curFprintPathsOffset = curFprintPathsOffset - 2; | |||
| offsetAllFprintPaths(curFprintPathsOffset); | |||
| window.requestAnimationFrame(fprintFrame); | |||
| } | |||
| else if (curFprintPathsOffset < -1) { | |||
| curFprintPathsOffset = curFprintPathsOffset + 2; | |||
| offsetAllFprintPaths(curFprintPathsOffset); | |||
| window.requestAnimationFrame(fprintFrame); | |||
| } | |||
| } | |||
| function fprintFrameToProcess(timestamp) { | |||
| if (!lastRafCallTimestamp) { | |||
| lastRafCallTimestamp = timestamp; | |||
| window.requestAnimationFrame(fprintFrameToProcess); | |||
| return; | |||
| } | |||
| let diff = timestamp - lastRafCallTimestamp; | |||
| fprintTick = (diff / TIME_TO_FILL_FPRINT) * 2; | |||
| lastRafCallTimestamp = timestamp; | |||
| curFprintPathsOffset += fprintTick * fprintProgressionDirection; | |||
| offsetAllFprintPaths(curFprintPathsOffset); | |||
| if (curFprintPathsOffset >= -1 && curFprintPathsOffset <= currentFprintProgresss - 1) { | |||
| window.requestAnimationFrame(fprintFrameToProcess); | |||
| } | |||
| else if (curFprintPathsOffset > currentFprintProgresss - 1) { | |||
| curFprintPathsOffset = currentFprintProgresss - 1; | |||
| offsetAllFprintPaths(curFprintPathsOffset); | |||
| lastRafCallTimestamp = 0; | |||
| } | |||
| } | |||
| function setFprintProgress(progress) { | |||
| if (progress == 0) { | |||
| curFprintPathsOffset = -1; | |||
| offsetAllFprintPaths(curFprintPathsOffset); | |||
| }else { | |||
| currentFprintProgresss = progress; | |||
| window.requestAnimationFrame(fprintFrameToProcess); | |||
| } | |||
| } | |||
| class Path { | |||
| constructor(selector, index) { | |||
| this.index = index; | |||
| this.querySelection = document.querySelectorAll(selector)[index]; | |||
| this.length = this.querySelection.getTotalLength(); | |||
| this.$ = $(selector).eq(index); | |||
| this.setDasharray(); | |||
| this.removesForwards = this.$.hasClass('demo__fprint-path--removes-forwards'); | |||
| } | |||
| setDasharray() { | |||
| this.$.css('stroke-dasharray', `${this.length} ${this.length + 2}`); | |||
| return this; | |||
| } | |||
| offset(ratio) { | |||
| this.$.css('stroke-dashoffset', -this.length * ratio ); | |||
| return this; | |||
| } | |||
| makeVisible() { | |||
| this.$.css('visibility', 'visible'); | |||
| return this; | |||
| } | |||
| } | |||
| $(document).ready(() => { | |||
| for (let i = 0; i < document.querySelectorAll(fprintPathSelector).length; i++) { | |||
| fprintPaths.push(new Path(fprintPathSelector, i)); | |||
| fprintPaths[i].offset(-1).makeVisible(); | |||
| } | |||
| //window.requestAnimationFrame(fprintFrame); | |||
| }) | |||
| @ -0,0 +1,28 @@ | |||
| /** | |||
| * jQuery toast plugin created by Kamran Ahmed copyright MIT license 2014 | |||
| */ | |||
| .jq-toast-wrap { display: block; position: fixed; width: 250px; pointer-events: none !important; margin: 0; padding: 0; letter-spacing: normal; z-index: 9000 !important; } | |||
| .jq-toast-wrap * { margin: 0; padding: 0; } | |||
| .jq-toast-wrap.bottom-left { bottom: 20px; left: 20px; } | |||
| .jq-toast-wrap.bottom-right { bottom: 20px; right: 40px; } | |||
| .jq-toast-wrap.top-left { top: 20px; left: 20px; } | |||
| .jq-toast-wrap.top-right { top: 20px; right: 40px; } | |||
| .jq-toast-single { display: block; width: 100%; padding: 10px; margin: 0px 0px 5px; border-radius: 4px; font-size: 12px; font-family: arial, sans-serif; line-height: 17px; position: relative; pointer-events: all !important; background-color: #444444; color: white; } | |||
| .jq-toast-single h2 { font-family: arial, sans-serif; font-size: 14px; margin: 0px 0px 7px; background: none; color: inherit; line-height: inherit; letter-spacing: normal; } | |||
| .jq-toast-single a { color: #eee; text-decoration: none; font-weight: bold; border-bottom: 1px solid white; padding-bottom: 3px; font-size: 12px; } | |||
| .jq-toast-single ul { margin: 0px 0px 0px 15px; background: none; padding:0px; } | |||
| .jq-toast-single ul li { list-style-type: disc !important; line-height: 17px; background: none; margin: 0; padding: 0; letter-spacing: normal; } | |||
| .close-jq-toast-single { position: absolute; top: 3px; right: 7px; font-size: 14px; cursor: pointer; } | |||
| .jq-toast-loader { display: block; position: absolute; top: -2px; height: 5px; width: 0%; left: 0; border-radius: 5px; background: red; } | |||
| .jq-toast-loaded { width: 100%; } | |||
| .jq-has-icon { padding: 10px 10px 10px 50px; background-repeat: no-repeat; background-position: 10px; } | |||
| .jq-icon-info { background-image: url(''); background-color: #31708f; color: #d9edf7; border-color: #bce8f1; } | |||
| .jq-icon-warning { background-image: url(''); background-color: #8a6d3b; color: #fcf8e3; border-color: #faebcc; } | |||
| .jq-icon-error { background-image: url(''); background-color: #a94442; color: #f2dede; border-color: #ebccd1; } | |||
| .jq-icon-success { background-image: url(''); color: #dff0d8; background-color: #3c763d; border-color: #d6e9c6; } | |||
| @ -0,0 +1,107 @@ | |||
| const STATE_WAIT_FINGER_DOWN = 1; | |||
| const STATE_WAIT_FINGER_LEAVE = 2; | |||
| const STATE_FINGER_DOWN = 3; | |||
| const STATE_FINGER_LEAVE = 4; | |||
| const STATE_EXTRACT_TEMPLATE = 5; | |||
| const STATE_DOWNLOAD_TEMPLATE = 6; | |||
| const STATE_EXTRACT_TEMPLATE_FAIL = 100; | |||
| const STATE_EXTRACT_TEMPLATE_DIRTY = 101; | |||
| const STATE_EXTRACT_TEMPLATE_POINTS_FEW = 102; | |||
| const STATE_EXTRACT_TEMPLATE_MERGE_FAIL = 103; | |||
| const ErrorReceiveDataErr = 1; | |||
| const ErrorEnrollTimeout = 2; | |||
| const ErrorDeviceNotFound = 3; | |||
| const ErrorEmptyFail = 4; | |||
| var mafp = { | |||
| isConnected: false, | |||
| ws: null, | |||
| enrollCount: 6, | |||
| enrollTimeout: 30 * 1000, | |||
| callback: null, | |||
| isWorking: false, | |||
| init(conf) { | |||
| if (conf && conf.enrollCount) { | |||
| this.enrollCount = conf.enrollCount; | |||
| } | |||
| if (conf && conf.enrollTimeout) { | |||
| this.enrollTimeout = conf.enrollTimeout; | |||
| } | |||
| return new Promise((resolve,reject) => { | |||
| var that = this; | |||
| if (this.isConnected) { | |||
| resolve(); | |||
| return; | |||
| } | |||
| var timeout = setTimeout(() => { | |||
| if (!that.isConnected) { | |||
| reject("Connection timeout"); | |||
| } | |||
| }, 2000) | |||
| var port = 9897; | |||
| var address = 'ws://localhost:' + port + '/'; | |||
| that.ws = new WebSocket(address); | |||
| that.ws.addEventListener('open', function() { | |||
| that.isConnected = true; | |||
| if (timeout) { | |||
| clearTimeout(timeout); | |||
| timeout = 0; | |||
| } | |||
| resolve() | |||
| }); | |||
| that.ws.addEventListener('close', function() { | |||
| console.log('Connection lost'); | |||
| that.isConnected = false; | |||
| reject("Connection lost"); | |||
| }); | |||
| that.ws.addEventListener('message', function(e) { | |||
| let resp = JSON.parse(e.data); | |||
| if (that.callback) { | |||
| that.callback(resp); | |||
| if(resp.err || resp.data) { | |||
| that.isWorking = false; | |||
| that.callback = null; | |||
| } | |||
| } | |||
| }); | |||
| }); | |||
| }, | |||
| _sendMessage(msg) { | |||
| this.ws.send(JSON.stringify(msg)); | |||
| }, | |||
| startEnroll(callback=((status, data)=>{})) { | |||
| this.callback = callback; | |||
| this.isWorking = true; | |||
| this.init().then(() => { | |||
| this._sendMessage({ | |||
| cmd: "enrollStart", | |||
| config: { | |||
| enrollCount: this.enrollCount, | |||
| enrollTimeout: this.enrollTimeout | |||
| } | |||
| }); | |||
| }).catch(err => { | |||
| this.callback = null; | |||
| this.isWorking = false; | |||
| callback({ | |||
| err: err, | |||
| state: 0, | |||
| step:0 | |||
| }) | |||
| }) | |||
| }, | |||
| cancelEnroll() { | |||
| this.callback = null; | |||
| this.isWorking = false; | |||
| this._sendMessage({ | |||
| cmd: "enrollCancel" | |||
| }); | |||
| } | |||
| }; | |||
| @ -0,0 +1,29 @@ | |||
| var socketId = 0; | |||
| chrome.app.runtime.onLaunched.addListener(function() { | |||
| if (socketId) { | |||
| console.log("close socket " +socketId); | |||
| chrome.sockets.tcpServer.close(socketId); | |||
| socketId = 0; | |||
| } | |||
| chrome.app.window.create('main.html', { | |||
| id: "mainwin", | |||
| innerBounds: { | |||
| width: 670, | |||
| height: 350, | |||
| minWidth: 670, | |||
| minHeight: 350 | |||
| } | |||
| }); | |||
| }) | |||
| chrome.runtime.onMessage.addListener(function(request, sender, sendResponse) | |||
| { | |||
| socketId = request.socketId; | |||
| }); | |||
| chrome.runtime.onSuspend.addListener(function() { | |||
| if(socketId) { | |||
| chrome.sockets.tcpServer.close(socketId); | |||
| socketId = 0; | |||
| } | |||
| }); | |||
| @ -0,0 +1,17 @@ | |||
| var MyEvent = function(type="") { | |||
| this.type = type; | |||
| this.listeners = []; | |||
| }; | |||
| MyEvent.prototype.addListener = function(listener) { | |||
| this.listeners.push(listener); | |||
| } | |||
| MyEvent.prototype.dispatch = function() { | |||
| for (var listener of this.listeners) { | |||
| var ret = listener.apply(null, arguments); | |||
| if (!ret) { | |||
| return; | |||
| } | |||
| } | |||
| } | |||
| @ -0,0 +1,894 @@ | |||
| /** | |||
| * Copyright (c) 2013 The Chromium Authors. All rights reserved. | |||
| * Use of this source code is governed by a BSD-style license that can be | |||
| * found in the LICENSE file. | |||
| **/ | |||
| var http = function() { | |||
| if (!chrome.sockets || !chrome.sockets.tcpServer) | |||
| return {}; | |||
| // Wrap chrome.sockets.tcp socketId with a Promise API. | |||
| var PSocket = (function() { | |||
| // chrome.sockets.tcp uses a global listener for incoming data so | |||
| // use a map to dispatch to the proper instance. | |||
| var socketMap = {}; | |||
| chrome.sockets.tcp.onReceive.addListener(function(info) { | |||
| var pSocket = socketMap[info.socketId]; | |||
| if (pSocket) { | |||
| if (pSocket.handlers) { | |||
| // Fulfil the pending read. | |||
| pSocket.handlers.resolve(info.data); | |||
| delete pSocket.handlers; | |||
| } | |||
| else { | |||
| // No pending read so put data on the queue. | |||
| pSocket.readQueue.push(info); | |||
| } | |||
| } | |||
| }); | |||
| // Read errors also use a global listener. | |||
| chrome.sockets.tcp.onReceiveError.addListener(function(info) { | |||
| var pSocket = socketMap[info.socketId]; | |||
| if (pSocket) { | |||
| if (pSocket.handlers) { | |||
| // Reject the pending read. | |||
| pSocket.handlers.reject(new Error('chrome.sockets.tcp error ' + info.resultCode)); | |||
| delete pSocket.handlers; | |||
| } | |||
| else { | |||
| // No pending read so put data on the queue. | |||
| pSocket.readQueue.push(info); | |||
| } | |||
| } | |||
| }); | |||
| // PSocket constructor. | |||
| return function(socketId) { | |||
| this.socketId = socketId; | |||
| this.readQueue = []; | |||
| // Register this instance for incoming data processing. | |||
| socketMap[socketId] = this; | |||
| chrome.sockets.tcp.setPaused(socketId, false); | |||
| }; | |||
| })(); | |||
| // Returns a Promise<ArrayBuffer> with read data. | |||
| PSocket.prototype.read = function() { | |||
| var that = this; | |||
| if (this.readQueue.length) { | |||
| // Return data from the queue. | |||
| var info = this.readQueue.shift(); | |||
| if (!info.resultCode) | |||
| return Promise.resolve(info.data); | |||
| else | |||
| return Promise.reject(new Error('chrome.sockets.tcp error ' + info.resultCode)); | |||
| } | |||
| else { | |||
| // The queue is empty so install handlers. | |||
| return new Promise(function(resolve, reject) { | |||
| that.handlers = { resolve: resolve, reject: reject }; | |||
| }); | |||
| } | |||
| }; | |||
| // Returns a Promise<integer> with the number of bytes written. | |||
| PSocket.prototype.write = function(data) { | |||
| var that = this; | |||
| return new Promise(function(resolve, reject) { | |||
| chrome.sockets.tcp.send(that.socketId, data, function(info) { | |||
| if (info && info.resultCode >= 0) | |||
| resolve(info.bytesSent); | |||
| else | |||
| reject(new Error('chrome sockets.tcp error ' + (info && info.resultCode))); | |||
| }); | |||
| }); | |||
| }; | |||
| // Returns a Promise. | |||
| PSocket.prototype.close = function() { | |||
| var that = this; | |||
| return new Promise(function(resolve, reject) { | |||
| chrome.sockets.tcp.disconnect(that.socketId, function() { | |||
| chrome.sockets.tcp.close(that.socketId, resolve); | |||
| }); | |||
| }); | |||
| }; | |||
| // Http response code strings. | |||
| var responseMap = { | |||
| 200: 'OK', | |||
| 301: 'Moved Permanently', | |||
| 304: 'Not Modified', | |||
| 400: 'Bad Request', | |||
| 401: 'Unauthorized', | |||
| 403: 'Forbidden', | |||
| 404: 'Not Found', | |||
| 413: 'Request Entity Too Large', | |||
| 414: 'Request-URI Too Long', | |||
| 500: 'Internal Server Error'}; | |||
| /** | |||
| * Convert from an ArrayBuffer to a string. | |||
| * @param {ArrayBuffer} buffer The array buffer to convert. | |||
| * @return {string} The textual representation of the array. | |||
| */ | |||
| var arrayBufferToString = function(buffer) { | |||
| var array = new Uint8Array(buffer); | |||
| var str = ''; | |||
| for (var i = 0; i < array.length; ++i) { | |||
| str += String.fromCharCode(array[i]); | |||
| } | |||
| return str; | |||
| }; | |||
| /** | |||
| * Convert from an UTF-8 array to UTF-8 string. | |||
| * @param {array} UTF-8 array | |||
| * @return {string} UTF-8 string | |||
| */ | |||
| var ary2utf8 = (function() { | |||
| var patterns = [ | |||
| {pattern: '0xxxxxxx', bytes: 1}, | |||
| {pattern: '110xxxxx', bytes: 2}, | |||
| {pattern: '1110xxxx', bytes: 3}, | |||
| {pattern: '11110xxx', bytes: 4}, | |||
| {pattern: '111110xx', bytes: 5}, | |||
| {pattern: '1111110x', bytes: 6} | |||
| ]; | |||
| patterns.forEach(function(item) { | |||
| item.header = item.pattern.replace(/[^10]/g, ''); | |||
| item.pattern01 = item.pattern.replace(/[^10]/g, '0'); | |||
| item.pattern01 = parseInt(item.pattern01, 2); | |||
| item.mask_length = item.header.length; | |||
| item.data_length = 8 - item.header.length; | |||
| var mask = ''; | |||
| for (var i = 0, len = item.mask_length; i < len; i++) { | |||
| mask += '1'; | |||
| } | |||
| for (var i = 0, len = item.data_length; i < len; i++) { | |||
| mask += '0'; | |||
| } | |||
| item.mask = mask; | |||
| item.mask = parseInt(item.mask, 2); | |||
| }); | |||
| return function(ary) { | |||
| var codes = []; | |||
| var cur = 0; | |||
| while(cur < ary.length) { | |||
| var first = ary[cur]; | |||
| var pattern = null; | |||
| for (var i = 0, len = patterns.length; i < len; i++) { | |||
| if ((first & patterns[i].mask) == patterns[i].pattern01) { | |||
| pattern = patterns[i]; | |||
| break; | |||
| } | |||
| } | |||
| if (pattern == null) { | |||
| throw 'utf-8 decode error'; | |||
| } | |||
| var rest = ary.slice(cur + 1, cur + pattern.bytes); | |||
| cur += pattern.bytes; | |||
| var code = ''; | |||
| code += ('00000000' + (first & (255 ^ pattern.mask)).toString(2)).slice(-pattern.data_length); | |||
| for (var i = 0, len = rest.length; i < len; i++) { | |||
| code += ('00000000' + (rest[i] & parseInt('111111', 2)).toString(2)).slice(-6); | |||
| } | |||
| codes.push(parseInt(code, 2)); | |||
| } | |||
| return String.fromCharCode.apply(null, codes); | |||
| }; | |||
| })(); | |||
| /** | |||
| * Convert from an UTF-8 string to UTF-8 array. | |||
| * @param {string} UTF-8 string | |||
| * @return {array} UTF-8 array | |||
| */ | |||
| var utf82ary = (function() { | |||
| var patterns = [ | |||
| {pattern: '0xxxxxxx', bytes: 1}, | |||
| {pattern: '110xxxxx', bytes: 2}, | |||
| {pattern: '1110xxxx', bytes: 3}, | |||
| {pattern: '11110xxx', bytes: 4}, | |||
| {pattern: '111110xx', bytes: 5}, | |||
| {pattern: '1111110x', bytes: 6} | |||
| ]; | |||
| patterns.forEach(function(item) { | |||
| item.header = item.pattern.replace(/[^10]/g, ''); | |||
| item.mask_length = item.header.length; | |||
| item.data_length = 8 - item.header.length; | |||
| item.max_bit_length = (item.bytes - 1) * 6 + item.data_length; | |||
| }); | |||
| var code2utf8array = function(code) { | |||
| var pattern = null; | |||
| var code01 = code.toString(2); | |||
| for (var i = 0, len = patterns.length; i < len; i++) { | |||
| if (code01.length <= patterns[i].max_bit_length) { | |||
| pattern = patterns[i]; | |||
| break; | |||
| } | |||
| } | |||
| if (pattern == null) { | |||
| throw 'utf-8 encode error'; | |||
| } | |||
| var ary = []; | |||
| for (var i = 0, len = pattern.bytes - 1; i < len; i++) { | |||
| ary.unshift(parseInt('10' + ('000000' + code01.slice(-6)).slice(-6), 2)); | |||
| code01 = code01.slice(0, -6); | |||
| } | |||
| ary.unshift(parseInt(pattern.header + ('00000000' + code01).slice(-pattern.data_length), 2)); | |||
| return ary; | |||
| }; | |||
| return function(str) { | |||
| var codes = []; | |||
| for (var i = 0, len = str.length; i < len; i++) { | |||
| var code = str.charCodeAt(i); | |||
| Array.prototype.push.apply(codes, code2utf8array(code)); | |||
| } | |||
| return codes; | |||
| }; | |||
| })(); | |||
| /** | |||
| * Convert a string to an ArrayBuffer. | |||
| * @param {string} string The string to convert. | |||
| * @return {ArrayBuffer} An array buffer whose bytes correspond to the string. | |||
| */ | |||
| var stringToArrayBuffer = function(string) { | |||
| var buffer = new ArrayBuffer(string.length); | |||
| var bufferView = new Uint8Array(buffer); | |||
| for (var i = 0; i < string.length; i++) { | |||
| bufferView[i] = string.charCodeAt(i); | |||
| } | |||
| return buffer; | |||
| }; | |||
| /** | |||
| * An event source can dispatch events. These are dispatched to all of the | |||
| * functions listening for that event type with arguments. | |||
| * @constructor | |||
| */ | |||
| function EventSource() { | |||
| this.listeners_ = {}; | |||
| }; | |||
| EventSource.prototype = { | |||
| /** | |||
| * Add |callback| as a listener for |type| events. | |||
| * @param {string} type The type of the event. | |||
| * @param {function(Object|undefined): boolean} callback The function to call | |||
| * when this event type is dispatched. Arguments depend on the event | |||
| * source and type. The function returns whether the event was "handled" | |||
| * which will prevent delivery to the rest of the listeners. | |||
| */ | |||
| addEventListener: function(type, callback) { | |||
| if (!this.listeners_[type]) | |||
| this.listeners_[type] = []; | |||
| this.listeners_[type].push(callback); | |||
| }, | |||
| /** | |||
| * Remove |callback| as a listener for |type| events. | |||
| * @param {string} type The type of the event. | |||
| * @param {function(Object|undefined): boolean} callback The callback | |||
| * function to remove from the event listeners for events having type | |||
| * |type|. | |||
| */ | |||
| removeEventListener: function(type, callback) { | |||
| if (!this.listeners_[type]) | |||
| return; | |||
| for (var i = this.listeners_[type].length - 1; i >= 0; i--) { | |||
| if (this.listeners_[type][i] == callback) { | |||
| this.listeners_[type].splice(i, 1); | |||
| } | |||
| } | |||
| }, | |||
| /** | |||
| * Dispatch an event to all listeners for events of type |type|. | |||
| * @param {type} type The type of the event being dispatched. | |||
| * @param {...Object} var_args The arguments to pass when calling the | |||
| * callback function. | |||
| * @return {boolean} Returns true if the event was handled. | |||
| */ | |||
| dispatchEvent: function(type, var_args) { | |||
| if (!this.listeners_[type]) | |||
| return false; | |||
| for (var i = 0; i < this.listeners_[type].length; i++) { | |||
| if (this.listeners_[type][i].apply( | |||
| /* this */ null, | |||
| /* var_args */ Array.prototype.slice.call(arguments, 1))) { | |||
| return true; | |||
| } | |||
| } | |||
| } | |||
| }; | |||
| /** | |||
| * HttpServer provides a lightweight Http web server. Currently it only | |||
| * supports GET requests and upgrading to other protocols (i.e. WebSockets). | |||
| * @constructor | |||
| */ | |||
| function HttpServer() { | |||
| EventSource.apply(this); | |||
| this.readyState_ = 0; | |||
| } | |||
| HttpServer.prototype = { | |||
| __proto__: EventSource.prototype, | |||
| /** | |||
| * Listen for connections on |port| using the interface |host|. | |||
| * @param {number} port The port to listen for incoming connections on. | |||
| * @param {string=} opt_host The host interface to listen for connections on. | |||
| * This will default to 0.0.0.0 if not specified which will listen on | |||
| * all interfaces. | |||
| */ | |||
| listen: function(port, opt_host) { | |||
| var t = this; | |||
| chrome.sockets.tcpServer.create(function(socketInfo) { | |||
| chrome.sockets.tcpServer.onAccept.addListener(function(acceptInfo) { | |||
| if (acceptInfo.socketId === socketInfo.socketId) | |||
| t.readRequestFromSocket_(new PSocket(acceptInfo.clientSocketId)); | |||
| }); | |||
| this.socketId = socketInfo.socketId; | |||
| chrome.runtime.sendMessage({socketId: this.socketId}); | |||
| chrome.sockets.tcpServer.listen( | |||
| socketInfo.socketId, | |||
| opt_host || '0.0.0.0', | |||
| port, | |||
| 50, | |||
| function(result) { | |||
| if (!result) { | |||
| t.readyState_ = 1; | |||
| } | |||
| else { | |||
| console.log( | |||
| 'listen error ' + | |||
| chrome.runtime.lastError.message + | |||
| ' (normal if another instance is already serving requests)'); | |||
| } | |||
| }); | |||
| }); | |||
| }, | |||
| close: function() { | |||
| if (this.socketId) { | |||
| chrome.sockets.tcpServer.close(this.socketId); | |||
| this.socketId = 0; | |||
| } | |||
| }, | |||
| readRequestFromSocket_: function(pSocket) { | |||
| var t = this; | |||
| var requestData = ''; | |||
| var endIndex = 0; | |||
| var onDataRead = function(data) { | |||
| requestData += arrayBufferToString(data).replace(/\r\n/g, '\n'); | |||
| // Check for end of request. | |||
| endIndex = requestData.indexOf('\n\n', endIndex); | |||
| if (endIndex == -1) { | |||
| endIndex = requestData.length - 1; | |||
| return pSocket.read().then(onDataRead); | |||
| } | |||
| var headers = requestData.substring(0, endIndex).split('\n'); | |||
| var headerMap = {}; | |||
| // headers[0] should be the Request-Line | |||
| var requestLine = headers[0].split(' '); | |||
| headerMap['method'] = requestLine[0]; | |||
| headerMap['url'] = requestLine[1]; | |||
| headerMap['Http-Version'] = requestLine[2]; | |||
| for (var i = 1; i < headers.length; i++) { | |||
| requestLine = headers[i].split(':', 2); | |||
| if (requestLine.length == 2) | |||
| headerMap[requestLine[0]] = requestLine[1].trim(); | |||
| } | |||
| var request = new HttpRequest(headerMap, pSocket); | |||
| t.onRequest_(request); | |||
| }; | |||
| pSocket.read().then(onDataRead).catch(function(e) { | |||
| pSocket.close(); | |||
| }); | |||
| }, | |||
| onRequest_: function(request) { | |||
| var type = request.headers['Upgrade'] ? 'upgrade' : 'request'; | |||
| var keepAlive = request.headers['Connection'] == 'keep-alive'; | |||
| if (!this.dispatchEvent(type, request)) | |||
| request.close(); | |||
| else if (keepAlive) | |||
| this.readRequestFromSocket_(request.pSocket_); | |||
| }, | |||
| }; | |||
| // MIME types for common extensions. | |||
| var extensionTypes = { | |||
| 'css': 'text/css', | |||
| 'html': 'text/html', | |||
| 'htm': 'text/html', | |||
| 'jpg': 'image/jpeg', | |||
| 'jpeg': 'image/jpeg', | |||
| 'js': 'text/javascript', | |||
| 'png': 'image/png', | |||
| 'svg': 'image/svg+xml', | |||
| 'txt': 'text/plain'}; | |||
| /** | |||
| * Constructs an HttpRequest object which tracks all of the request headers and | |||
| * socket for an active Http request. | |||
| * @param {Object} headers The HTTP request headers. | |||
| * @param {Object} pSocket The socket to use for the response. | |||
| * @constructor | |||
| */ | |||
| function HttpRequest(headers, pSocket) { | |||
| this.version = 'HTTP/1.1'; | |||
| this.headers = headers; | |||
| this.responseHeaders_ = {}; | |||
| this.headersSent = false; | |||
| this.pSocket_ = pSocket; | |||
| this.writes_ = 0; | |||
| this.bytesRemaining = 0; | |||
| this.finished_ = false; | |||
| this.readyState = 1; | |||
| } | |||
| HttpRequest.prototype = { | |||
| __proto__: EventSource.prototype, | |||
| /** | |||
| * Closes the Http request. | |||
| */ | |||
| close: function() { | |||
| // The socket for keep alive connections will be re-used by the server. | |||
| // Just stop referencing or using the socket in this HttpRequest. | |||
| if (this.headers['Connection'] != 'keep-alive') | |||
| pSocket.close(); | |||
| this.pSocket_ = null; | |||
| this.readyState = 3; | |||
| }, | |||
| /** | |||
| * Write the provided headers as a response to the request. | |||
| * @param {int} responseCode The HTTP status code to respond with. | |||
| * @param {Object} responseHeaders The response headers describing the | |||
| * response. | |||
| */ | |||
| writeHead: function(responseCode, responseHeaders) { | |||
| var headerString = this.version + ' ' + responseCode + ' ' + | |||
| (responseMap[responseCode] || 'Unknown'); | |||
| this.responseHeaders_ = responseHeaders; | |||
| if (this.headers['Connection'] == 'keep-alive') | |||
| responseHeaders['Connection'] = 'keep-alive'; | |||
| if (!responseHeaders['Content-Length'] && responseHeaders['Connection'] == 'keep-alive') | |||
| responseHeaders['Transfer-Encoding'] = 'chunked'; | |||
| for (var i in responseHeaders) { | |||
| headerString += '\r\n' + i + ': ' + responseHeaders[i]; | |||
| } | |||
| headerString += '\r\n\r\n'; | |||
| this.write_(stringToArrayBuffer(headerString)); | |||
| }, | |||
| /** | |||
| * Writes data to the response stream. | |||
| * @param {string|ArrayBuffer} data The data to write to the stream. | |||
| */ | |||
| write: function(data) { | |||
| if (this.responseHeaders_['Transfer-Encoding'] == 'chunked') { | |||
| var newline = '\r\n'; | |||
| var byteLength = (data instanceof ArrayBuffer) ? data.byteLength : data.length; | |||
| var chunkLength = byteLength.toString(16).toUpperCase() + newline; | |||
| var buffer = new ArrayBuffer(chunkLength.length + byteLength + newline.length); | |||
| var bufferView = new Uint8Array(buffer); | |||
| for (var i = 0; i < chunkLength.length; i++) | |||
| bufferView[i] = chunkLength.charCodeAt(i); | |||
| if (data instanceof ArrayBuffer) { | |||
| bufferView.set(new Uint8Array(data), chunkLength.length); | |||
| } else { | |||
| for (var i = 0; i < data.length; i++) | |||
| bufferView[chunkLength.length + i] = data.charCodeAt(i); | |||
| } | |||
| for (var i = 0; i < newline.length; i++) | |||
| bufferView[chunkLength.length + byteLength + i] = newline.charCodeAt(i); | |||
| data = buffer; | |||
| } else if (!(data instanceof ArrayBuffer)) { | |||
| data = stringToArrayBuffer(data); | |||
| } | |||
| this.write_(data); | |||
| }, | |||
| /** | |||
| * Finishes the HTTP response writing |data| before closing. | |||
| * @param {string|ArrayBuffer=} opt_data Optional data to write to the stream | |||
| * before closing it. | |||
| */ | |||
| end: function(opt_data) { | |||
| if (opt_data) | |||
| this.write(opt_data); | |||
| if (this.responseHeaders_['Transfer-Encoding'] == 'chunked') | |||
| this.write(''); | |||
| this.finished_ = true; | |||
| this.checkFinished_(); | |||
| }, | |||
| /** | |||
| * Automatically serve the given |url| request. | |||
| * @param {string} url The URL to fetch the file to be served from. This is | |||
| * retrieved via an XmlHttpRequest and served as the response to the | |||
| * request. | |||
| */ | |||
| serveUrl: function(url) { | |||
| var t = this; | |||
| var xhr = new XMLHttpRequest(); | |||
| xhr.onloadend = function() { | |||
| var type = 'text/plain'; | |||
| if (this.getResponseHeader('Content-Type')) { | |||
| type = this.getResponseHeader('Content-Type'); | |||
| } else if (url.indexOf('.') != -1) { | |||
| var extension = url.substr(url.indexOf('.') + 1); | |||
| type = extensionTypes[extension] || type; | |||
| } | |||
| console.log('Served ' + url); | |||
| var contentLength = this.getResponseHeader('Content-Length'); | |||
| if (xhr.status == 200) | |||
| contentLength = (this.response && this.response.byteLength) || 0; | |||
| t.writeHead(this.status, { | |||
| 'Content-Type': type, | |||
| 'Content-Length': contentLength}); | |||
| t.end(this.response); | |||
| }; | |||
| xhr.open('GET', url, true); | |||
| xhr.responseType = 'arraybuffer'; | |||
| xhr.send(); | |||
| }, | |||
| write_: function(array) { | |||
| var t = this; | |||
| this.bytesRemaining += array.byteLength; | |||
| this.pSocket_.write(array).then(function(bytesWritten) { | |||
| t.bytesRemaining -= bytesWritten; | |||
| t.checkFinished_(); | |||
| }).catch(function(e) { | |||
| console.error(e.message); | |||
| return; | |||
| }); | |||
| }, | |||
| checkFinished_: function() { | |||
| if (!this.finished_ || this.bytesRemaining > 0) | |||
| return; | |||
| this.close(); | |||
| } | |||
| }; | |||
| /** | |||
| * Constructs a server which is capable of accepting WebSocket connections. | |||
| * @param {HttpServer} httpServer The Http Server to listen and handle | |||
| * WebSocket upgrade requests on. | |||
| * @constructor | |||
| */ | |||
| function WebSocketServer(httpServer) { | |||
| EventSource.apply(this); | |||
| httpServer.addEventListener('upgrade', this.upgradeToWebSocket_.bind(this)); | |||
| } | |||
| WebSocketServer.prototype = { | |||
| __proto__: EventSource.prototype, | |||
| upgradeToWebSocket_: function(request) { | |||
| if (request.headers['Upgrade'] != 'websocket' || | |||
| !request.headers['Sec-WebSocket-Key']) { | |||
| return false; | |||
| } | |||
| if (this.dispatchEvent('request', new WebSocketRequest(request))) { | |||
| if (request.pSocket_) | |||
| request.reject(); | |||
| return true; | |||
| } | |||
| return false; | |||
| } | |||
| }; | |||
| /** | |||
| * Constructs a WebSocket request object from an Http request. This invalidates | |||
| * the Http request's socket and offers accept and reject methods for accepting | |||
| * and rejecting the WebSocket upgrade request. | |||
| * @param {HttpRequest} httpRequest The HTTP request to upgrade. | |||
| */ | |||
| function WebSocketRequest(httpRequest) { | |||
| // We'll assume control of the socket for this request. | |||
| HttpRequest.apply(this, [httpRequest.headers, httpRequest.pSocket_]); | |||
| httpRequest.pSocket_ = null; | |||
| } | |||
| WebSocketRequest.prototype = { | |||
| __proto__: HttpRequest.prototype, | |||
| /** | |||
| * Accepts the WebSocket request. | |||
| * @return {WebSocketServerSocket} The websocket for the accepted request. | |||
| */ | |||
| accept: function() { | |||
| // Construct WebSocket response key. | |||
| var clientKey = this.headers['Sec-WebSocket-Key']; | |||
| var toArray = function(str) { | |||
| var a = []; | |||
| for (var i = 0; i < str.length; i++) { | |||
| a.push(str.charCodeAt(i)); | |||
| } | |||
| return a; | |||
| } | |||
| var toString = function(a) { | |||
| var str = ''; | |||
| for (var i = 0; i < a.length; i++) { | |||
| str += String.fromCharCode(a[i]); | |||
| } | |||
| return str; | |||
| } | |||
| // Magic string used for websocket connection key hashing: | |||
| // http://en.wikipedia.org/wiki/WebSocket | |||
| var magicStr = '258EAFA5-E914-47DA-95CA-C5AB0DC85B11'; | |||
| // clientKey is base64 encoded key. | |||
| clientKey += magicStr; | |||
| var sha1 = new Sha1(); | |||
| sha1.reset(); | |||
| sha1.update(toArray(clientKey)); | |||
| var responseKey = btoa(toString(sha1.digest())); | |||
| var responseHeader = { | |||
| 'Upgrade': 'websocket', | |||
| 'Connection': 'Upgrade', | |||
| 'Sec-WebSocket-Accept': responseKey}; | |||
| if (this.headers['Sec-WebSocket-Protocol']) | |||
| responseHeader['Sec-WebSocket-Protocol'] = this.headers['Sec-WebSocket-Protocol']; | |||
| this.writeHead(101, responseHeader); | |||
| var socket = new WebSocketServerSocket(this.pSocket_); | |||
| // Detach the socket so that we don't use it anymore. | |||
| this.pSocket_ = 0; | |||
| return socket; | |||
| }, | |||
| /** | |||
| * Rejects the WebSocket request, closing the connection. | |||
| */ | |||
| reject: function() { | |||
| this.close(); | |||
| } | |||
| } | |||
| /** | |||
| * Constructs a WebSocketServerSocket using the given socketId. This should be | |||
| * a socket which has already been upgraded from an Http request. | |||
| * @param {number} socketId The socket id with an active websocket connection. | |||
| */ | |||
| function WebSocketServerSocket(pSocket) { | |||
| this.pSocket_ = pSocket; | |||
| this.readyState = 1; | |||
| EventSource.apply(this); | |||
| this.readFromSocket_(); | |||
| } | |||
| WebSocketServerSocket.prototype = { | |||
| __proto__: EventSource.prototype, | |||
| /** | |||
| * Send |data| on the WebSocket. | |||
| * @param {string|Array.<number>|ArrayBuffer} data The data to send over the WebSocket. | |||
| */ | |||
| send: function(data) { | |||
| // WebSocket must specify opcode when send frame. | |||
| // The opcode for data frame is 1(text) or 2(binary). | |||
| if (typeof data == 'string' || data instanceof String) { | |||
| this.sendFrame_(1, data); | |||
| } else { | |||
| this.sendFrame_(2, data); | |||
| } | |||
| }, | |||
| /** | |||
| * Begin closing the WebSocket. Note that the WebSocket protocol uses a | |||
| * handshake to close the connection, so this call will begin the closing | |||
| * process. | |||
| */ | |||
| close: function() { | |||
| if (this.readyState === 1) { | |||
| this.sendFrame_(8); | |||
| this.readyState = 2; | |||
| } | |||
| }, | |||
| readFromSocket_: function() { | |||
| var t = this; | |||
| var data = []; | |||
| var message = ''; | |||
| var fragmentedOp = 0; | |||
| var fragmentedMessages = []; | |||
| var onDataRead = function(dataBuffer) { | |||
| var a = new Uint8Array(dataBuffer); | |||
| for (var i = 0; i < a.length; i++) | |||
| data.push(a[i]); | |||
| while (data.length) { | |||
| var length_code = -1; | |||
| var data_start = 6; | |||
| var mask; | |||
| var fin = (data[0] & 128) >> 7; | |||
| var op = data[0] & 15; | |||
| if (data.length > 1) | |||
| length_code = data[1] & 127; | |||
| if (length_code > 125) { | |||
| if ((length_code == 126 && data.length > 7) || | |||
| (length_code == 127 && data.length > 14)) { | |||
| if (length_code == 126) { | |||
| length_code = data[2] * 256 + data[3]; | |||
| mask = data.slice(4, 8); | |||
| data_start = 8; | |||
| } else if (length_code == 127) { | |||
| length_code = 0; | |||
| for (var i = 0; i < 8; i++) { | |||
| length_code = length_code * 256 + data[2 + i]; | |||
| } | |||
| mask = data.slice(10, 14); | |||
| data_start = 14; | |||
| } | |||
| } else { | |||
| length_code = -1; // Insufficient data to compute length | |||
| } | |||
| } else { | |||
| if (data.length > 5) | |||
| mask = data.slice(2, 6); | |||
| } | |||
| if (length_code > -1 && data.length >= data_start + length_code) { | |||
| var decoded = data.slice(data_start, data_start + length_code).map(function(byte, index) { | |||
| return byte ^ mask[index % 4]; | |||
| }); | |||
| if (op == 1) { | |||
| decoded = ary2utf8(decoded); | |||
| } | |||
| data = data.slice(data_start + length_code); | |||
| if (fin && op > 0) { | |||
| // Unfragmented message. | |||
| if (!t.onFrame_(op, decoded)) | |||
| return; | |||
| } else { | |||
| // Fragmented message. | |||
| fragmentedOp = fragmentedOp || op; | |||
| fragmentedMessages.push(decoded); | |||
| if (fin) { | |||
| var joinMessage = null; | |||
| if (op == 1) { | |||
| joinMessage = fragmentedMessagess.join(''); | |||
| } else { | |||
| joinMessage = fragmentedMessages.reduce(function(pre, cur) { | |||
| return Array.prototype.push.apply(pre, cur); | |||
| }, []); | |||
| } | |||
| if (!t.onFrame_(fragmentedOp, joinMessage)) | |||
| return; | |||
| fragmentedOp = 0; | |||
| fragmentedMessages = []; | |||
| } | |||
| } | |||
| } else { | |||
| break; // Insufficient data, wait for more. | |||
| } | |||
| } | |||
| return t.pSocket_.read().then(onDataRead); | |||
| }; | |||
| this.pSocket_.read().then(function(data) { | |||
| return onDataRead(data); | |||
| }).catch(function(e) { | |||
| t.close_(); | |||
| }); | |||
| }, | |||
| onFrame_: function(op, data) { | |||
| if (op == 1 || op == 2) { | |||
| if (typeof data == 'string' || data instanceof String) { | |||
| // Don't do anything. | |||
| } else if (Array.isArray(data)) { | |||
| data = new Uint8Array(data).buffer; | |||
| } else if (data instanceof ArrayBuffer) { | |||
| // Don't do anything. | |||
| } else { | |||
| data = data.buffer; | |||
| } | |||
| this.dispatchEvent('message', {'data': data}); | |||
| } else if (op == 8) { | |||
| // A close message must be confirmed before the websocket is closed. | |||
| if (this.readyState === 1) { | |||
| this.sendFrame_(8); | |||
| this.readyState = 2; | |||
| } else { | |||
| this.close_(); | |||
| return false; | |||
| } | |||
| } | |||
| return true; | |||
| }, | |||
| sendFrame_: function(op, data) { | |||
| var t = this; | |||
| var WebsocketFrameData = function(op, data) { | |||
| var ary = data; | |||
| if (typeof data == 'string' || data instanceof String) { | |||
| ary = utf82ary(data); | |||
| } | |||
| if (Array.isArray(ary)) { | |||
| ary = new Uint8Array(ary); | |||
| } | |||
| if (ary instanceof ArrayBuffer) { | |||
| ary = new Uint8Array(ary); | |||
| } | |||
| ary = new Uint8Array(ary.buffer); | |||
| var length = ary.length; | |||
| if (ary.length > 65535) | |||
| length += 10; | |||
| else if (ary.length > 125) | |||
| length += 4; | |||
| else | |||
| length += 2; | |||
| var lengthBytes = 0; | |||
| var buffer = new ArrayBuffer(length); | |||
| var bv = new Uint8Array(buffer); | |||
| bv[0] = 128 | (op & 15); // Fin and type text. | |||
| bv[1] = ary.length > 65535 ? 127 : | |||
| (ary.length > 125 ? 126 : ary.length); | |||
| if (ary.length > 65535) | |||
| lengthBytes = 8; | |||
| else if (ary.length > 125) | |||
| lengthBytes = 2; | |||
| var len = ary.length; | |||
| for (var i = lengthBytes - 1; i >= 0; i--) { | |||
| bv[2 + i] = len & 255; | |||
| len = len >> 8; | |||
| } | |||
| var dataStart = lengthBytes + 2; | |||
| for (var i = 0; i < ary.length; i++) { | |||
| bv[dataStart + i] = ary[i]; | |||
| } | |||
| return buffer; | |||
| } | |||
| var array = WebsocketFrameData(op, data || ''); | |||
| this.pSocket_.write(array).then(function(bytesWritten) { | |||
| if (bytesWritten !== array.byteLength) | |||
| throw new Error('insufficient write'); | |||
| }).catch(function(e) { | |||
| t.close_(); | |||
| }); | |||
| }, | |||
| close_: function() { | |||
| if (this.readyState !== 3) { | |||
| this.pSocket_.close(); | |||
| this.readyState = 3; | |||
| this.dispatchEvent('close'); | |||
| } | |||
| } | |||
| }; | |||
| return { | |||
| 'Server': HttpServer, | |||
| 'WebSocketServer': WebSocketServer, | |||
| }; | |||
| }(); | |||
| @ -0,0 +1,27 @@ | |||
| <!DOCTYPE html> | |||
| <html> | |||
| <head> | |||
| <script src="event.js"></script> | |||
| <script src="sha1.js"></script> | |||
| <script src="http.js"></script> | |||
| <script src="main.js"></script> | |||
| <link rel="stylesheet" href="style.css"> | |||
| </head> | |||
| <body> | |||
| <div class="msg"> | |||
| <div id="buffer"> </div> | |||
| <div id="msg_end"></div> | |||
| </div> | |||
| <div class="buttons"> | |||
| <input type="checkbox" id="show-log"> Debug | |||
| <button id="enroll">Enroll</button> | |||
| <button id="down">Down</button> | |||
| <button id="match">Match</button> | |||
| <button id="refresh">Refesh Device</button> | |||
| <button id="export">Export</button> | |||
| <button id="import">Import&Down</button> | |||
| <input type="file" id="import-file" style="display:none"/> | |||
| </div> | |||
| </body> | |||
| </html> | |||
| @ -0,0 +1,883 @@ | |||
| const serial = chrome.serial; | |||
| var showDebugLog = false; | |||
| var SerialConnection = function() { | |||
| this.connectionId = -1; | |||
| this.boundOnReceive = this.onReceive.bind(this); | |||
| this.boundOnReceiveError = this.onReceiveError.bind(this); | |||
| this.onConnect = new MyEvent(); | |||
| this.onReceive = new MyEvent(); | |||
| this.onError = new MyEvent(); | |||
| this.recvBuffer = new Uint8Array(8*1024); | |||
| this.recvView = new DataView(this.recvBuffer.buffer); | |||
| this.recvCursor = 0; | |||
| this.bitrate = 57600; | |||
| }; | |||
| SerialConnection.prototype.onConnectComplete = function(connectionInfo) { | |||
| if (!connectionInfo) { | |||
| if (chrome.runtime.lastError != undefined) { | |||
| logln('chrome.serial.connect error: ' + chrome.runtime.lastError.message); | |||
| } | |||
| return; | |||
| } | |||
| this.connectionId = connectionInfo.connectionId; | |||
| chrome.serial.onReceive.addListener(this.boundOnReceive); | |||
| chrome.serial.onReceiveError.addListener(this.boundOnReceiveError); | |||
| this.onConnect.dispatch(); | |||
| }; | |||
| SerialConnection.prototype.onReceive = function(receiveInfo) { | |||
| if (receiveInfo.connectionId !== this.connectionId) { | |||
| return; | |||
| } | |||
| this.recvBuffer.set(new Uint8Array(receiveInfo.data), this.recvCursor); | |||
| this.recvCursor += receiveInfo.data.byteLength; | |||
| //console.log(buf2hex(receiveInfo.data)); | |||
| if (this.recvCursor < 6) { | |||
| return; | |||
| } | |||
| this._dispathReceiveData(); | |||
| }; | |||
| SerialConnection.prototype.clearRecvData = function() { | |||
| this.recvCursor = 0; | |||
| this.recvBuffer.fill(0); | |||
| } | |||
| SerialConnection.prototype._dispathReceiveData = function() { | |||
| var dLen = this.recvView.getUint16(7); | |||
| if (this.recvCursor < dLen + 9) { | |||
| return; | |||
| } | |||
| var dataBuffer = new Uint8Array(dLen + 9); | |||
| dataBuffer.set(this.recvBuffer.subarray(0, dLen + 9)); | |||
| if (showDebugLog) { | |||
| logln("recv: " + buf2hex(dataBuffer.buffer)); | |||
| } | |||
| var realCrc = calcCRC(dataBuffer, 6, dLen + 7); | |||
| var crc = dataBuffer[dLen+7] * 256 + dataBuffer[dLen+8]; | |||
| if (crc != realCrc) { | |||
| logln("invalid crc " + crc + " ,real= " + realCrc); | |||
| }else { | |||
| var packet = new Packet().fromDataBuffer(dataBuffer); | |||
| this.onReceive.dispatch(packet); | |||
| } | |||
| if (this.recvCursor > dLen + 9) { | |||
| dataBuffer = new Uint8Array(this.recvCursor - dLen - 6); | |||
| dataBuffer.set(this.recvBuffer.subarray(dLen + 9, this.recvCursor)); | |||
| this.recvBuffer.fill(0); | |||
| this.recvBuffer.set(dataBuffer); | |||
| this.recvCursor -= dLen + 9; | |||
| this._dispathReceiveData(); | |||
| }else { | |||
| this.recvCursor = 0; | |||
| this.recvBuffer.fill(0); | |||
| } | |||
| } | |||
| SerialConnection.prototype.onReceiveError = function(errorInfo) { | |||
| if (errorInfo.connectionId === this.connectionId) { | |||
| this.onError.dispatch(errorInfo.error); | |||
| } | |||
| }; | |||
| SerialConnection.prototype.update = function(conf, cb) { | |||
| if (!this.connectionId) {return;} | |||
| serial.update(this.connectionId, conf, cb); | |||
| } | |||
| SerialConnection.prototype.connect = function(path, bitrate=57600) { | |||
| this.clearRecvData(); | |||
| this.bitrate = bitrate; | |||
| serial.connect(path, { bitrate: this.bitrate },this.onConnectComplete.bind(this)) | |||
| }; | |||
| SerialConnection.prototype.send = function(packet) { | |||
| if (this.connectionId < 0) { | |||
| throw 'Invalid connection'; | |||
| } | |||
| var data = packet.getDataBuffer(); | |||
| if (showDebugLog) { | |||
| logln("send: " + buf2hex(data)); | |||
| } | |||
| serial.send(this.connectionId, data, function() {}); | |||
| }; | |||
| SerialConnection.prototype.disconnect = function() { | |||
| if (this.connectionId < 0) { | |||
| throw 'Invalid connection'; | |||
| } | |||
| serial.disconnect(this.connectionId, function() {}); | |||
| }; | |||
| //////////////////////////////////////////////////////// | |||
| //////////////////////////////////////////////////////// | |||
| function logln(msg) { | |||
| var buffer = document.querySelector('#buffer'); | |||
| buffer.innerHTML += msg + '<br/>'; | |||
| var msgEnd = document.querySelector('#msg_end'); | |||
| msgEnd.scrollIntoView(); | |||
| } | |||
| function log(msg) { | |||
| var buffer = document.querySelector('#buffer'); | |||
| buffer.innerHTML += msg; | |||
| var msgEnd = document.querySelector('#msg_end'); | |||
| msgEnd.scrollIntoView(); | |||
| } | |||
| function buf2hex(buffer) { | |||
| return '0x' + Array.prototype.map.call(new Uint8Array(buffer), x => ('0x00' + x.toString(16)).slice(-2)).join(' 0x'); | |||
| } | |||
| function calcCRC(buffer, start, end) { | |||
| var crc = 0; | |||
| for (var i = start; i < end; i++) { | |||
| crc += buffer[i] & 0xff; | |||
| } | |||
| return crc & 0xffff; | |||
| } | |||
| const PacketTypeCmd = 0x01; | |||
| const PacketTypeData = 0x02; | |||
| const PacketTypeDataEnd = 0x08; | |||
| const PacketTypeCmdResp = 0x07; | |||
| const PacketTypeDataResp = 0x09; | |||
| var Packet = function(data=new Uint8Array([0x35]), dataLen=1, type=PacketTypeCmd) { | |||
| this.dataBuffer = new Uint8Array(512); | |||
| this.dataBuffer.set([0xEF, 0x01, 0xFF, 0xFF, 0xFF, 0xFF]); | |||
| this.type = type; | |||
| this.dataLen = dataLen; | |||
| this.data = data; | |||
| this.result = 0; | |||
| } | |||
| Packet.prototype.setCmd = function(cmd) { | |||
| this.type = PacketTypeCmd; | |||
| this.data = Uint8Array.of(cmd); | |||
| this.dataLen = 1; | |||
| } | |||
| Packet.prototype.getDataBuffer = function () { | |||
| var dataView = new DataView(this.dataBuffer.buffer); | |||
| dataView.setUint8(6, this.type); | |||
| var len = this.dataLen + 2; | |||
| dataView.setUint16(7, len); | |||
| this.dataBuffer.set(this.data, 9); | |||
| var crc = calcCRC(this.dataBuffer, 6, this.dataLen + 9); | |||
| dataView.setUint16(9 + this.dataLen, crc); | |||
| return new Uint8Array(this.dataBuffer.buffer, 0, this.dataLen + 11); | |||
| } | |||
| Packet.prototype.fromDataBuffer = function(buffer) { | |||
| this.dataBuffer.set(buffer); | |||
| var dataView = new DataView(this.dataBuffer.buffer); | |||
| this.type = dataView.getUint8(6); | |||
| var len = dataView.getUint16(7); | |||
| this.dataLen = len - 2; | |||
| this.data = new Uint8Array(buffer.buffer, 9, this.dataLen); | |||
| if (this.type == PacketTypeCmdResp) { | |||
| this.result = this.data[0]; | |||
| } | |||
| return this; | |||
| } | |||
| //////////////////////////////////////////////////////// | |||
| //////////////////////////////////////////////////////// | |||
| var curDevice = null; | |||
| function tryHandshake(device) { | |||
| return new Promise((resolve, reject) =>{ | |||
| logln("try handshake with " + device.path); | |||
| var connection = new SerialConnection(); | |||
| var isValidDevice = 0; | |||
| device.bitrate = 57600; | |||
| connection.onConnect.addListener(function() { | |||
| logln("device " + device.path + " connected"); | |||
| connection.send(new Packet()); | |||
| }); | |||
| connection.onReceive.addListener(function(packet) { | |||
| if (packet.type == PacketTypeCmdResp && packet.result == 0) { | |||
| isValidDevice = 1; | |||
| connection.disconnect(); | |||
| resolve(1) | |||
| } | |||
| }); | |||
| connection.connect(device.path, device.bitrate); | |||
| setTimeout(() => { | |||
| if (isValidDevice) { | |||
| }else { | |||
| device.bitrate = 115200; | |||
| connection.update({ | |||
| bitrate: device.bitrate | |||
| }, (result) => { | |||
| if (result) { | |||
| connection.send(new Packet()); | |||
| setTimeout(() => { | |||
| connection.disconnect(); | |||
| if (!isValidDevice) { | |||
| resolve(0); | |||
| } | |||
| }, 500); | |||
| }else { | |||
| connection.disconnect(); | |||
| resolve(0); | |||
| } | |||
| }) | |||
| } | |||
| }, 500); | |||
| }) | |||
| } | |||
| async function checkDevices(devices) { | |||
| for (var device of devices) { | |||
| logln("found device, path = " + device.path); | |||
| var res = await tryHandshake(device); | |||
| if (res) { | |||
| curDevice = device; | |||
| logln("found valid device " + device.path); | |||
| break; | |||
| } | |||
| } | |||
| } | |||
| //////////////////////////////////////////////////////// | |||
| var EnrollSchedule = function(device, enrollCount=6, callback=((err=null, step=0, state=0, data=null)=>{}), timeout = 60 * 1000) { | |||
| this.step = 0; | |||
| this.enrollCount = enrollCount; | |||
| this.callback = callback; | |||
| this.device = device; | |||
| this.connection = null; | |||
| this.timeout = timeout; | |||
| this.timeoutFlag = 0; | |||
| this.responseTime = 1000; | |||
| this.responseTimeout = 0; | |||
| this.responseCallback = null; | |||
| this.canceled = false; | |||
| } | |||
| const STATE_WAIT_FINGER_DOWN = 1; | |||
| const STATE_WAIT_FINGER_LEAVE = 2; | |||
| const STATE_FINGER_DOWN = 3; | |||
| const STATE_FINGER_LEAVE = 4; | |||
| const STATE_EXTRACT_TEMPLATE = 5; | |||
| const STATE_DOWNLOAD_TEMPLATE = 6; | |||
| const STATE_EXTRACT_TEMPLATE_FAIL = 100; | |||
| const STATE_EXTRACT_TEMPLATE_DIRTY = 101; | |||
| const STATE_EXTRACT_TEMPLATE_POINTS_FEW = 102; | |||
| const STATE_EXTRACT_TEMPLATE_MERGE_FAIL = 103; | |||
| const ErrorReceiveDataErr = 1; | |||
| const ErrorEnrollTimeout = 2; | |||
| const ErrorDeviceNotFound = 3; | |||
| const ErrorEmptyFail = 4; | |||
| function printError(err) { | |||
| logln( "enroll err with code: " + err); | |||
| } | |||
| EnrollSchedule.prototype.start = async function() { | |||
| this.step = 0; | |||
| this.canceled = false; | |||
| try { | |||
| await this._connect(); | |||
| if(this.canceled) { return this._disConnect(); } | |||
| var ret = await this._sendAndWaitRecvCmd(0x0d); | |||
| if (ret == 0x11) { | |||
| throw ErrorEmptyFail; | |||
| } | |||
| while(this.step < this.enrollCount) { | |||
| this.step += 1; | |||
| ret = await this._enrollOne(this.step); | |||
| if (ret) { | |||
| let stateErr = STATE_EXTRACT_TEMPLATE_FAIL; | |||
| if(ret == 0x06) { | |||
| stateErr = STATE_EXTRACT_TEMPLATE_DIRTY; | |||
| }else if (ret == 0x07) { | |||
| stateErr = STATE_EXTRACT_TEMPLATE_POINTS_FEW; | |||
| }else if (ret == 0x0a) { | |||
| stateErr = STATE_EXTRACT_TEMPLATE_MERGE_FAIL; | |||
| } | |||
| this.callback(null, this.step, stateErr, null); | |||
| this.step -= 1; | |||
| } | |||
| if(this.canceled) { return this._disConnect(); } | |||
| } | |||
| logln("receive tempalte"); | |||
| this.callback(null, this.step, STATE_DOWNLOAD_TEMPLATE, null); | |||
| var ret = await this._mergeTemplate(); | |||
| if(this.canceled) { return this._disConnect(); } | |||
| var recvBuffer = await this._receiveTemplate(); | |||
| if(this.canceled) { return this._disConnect(); } | |||
| this.callback(null, this.step, 0, recvBuffer); | |||
| }catch(e) { | |||
| this.callback(e, this.step, 0, null); | |||
| } | |||
| this._disConnect(); | |||
| } | |||
| EnrollSchedule.prototype._mergeTemplate = function() { | |||
| return this._sendAndWaitRecvCmd(0x05); | |||
| } | |||
| EnrollSchedule.prototype._receiveTemplate = function() { | |||
| return new Promise((resolve, reject) => { | |||
| var that = this; | |||
| var data = Uint8Array.of(0x08, 0x01); | |||
| var packet = new Packet(data, 2); | |||
| var recvDataLen = 4 * 1024 * 1024; | |||
| var recvData = new Uint8Array(recvDataLen); | |||
| var recvDataCursor = 0; | |||
| var resetTimeoutCheck = function() { | |||
| if (that.responseTimeout) { | |||
| clearTimeout(that.responseTimeout); | |||
| } | |||
| that.responseTimeout = setTimeout(() => { | |||
| var dummy = new Packet(); | |||
| that.connection.send(dummy); | |||
| that.responseTimeout = setTimeout(() => { | |||
| reject(ErrorReceiveDataErr); | |||
| }, that.responseTime); | |||
| }, that.responseTime); | |||
| } | |||
| this.responseCallback = (packet) => { | |||
| resetTimeoutCheck(); | |||
| if (packet.type == PacketTypeCmdResp) { | |||
| }else if (packet.type == PacketTypeData || packet.type == PacketTypeDataEnd) { | |||
| if (recvDataCursor + packet.dataLen < recvDataLen) { | |||
| recvData.set(packet.data, recvDataCursor); | |||
| recvDataCursor += packet.dataLen; | |||
| }else { | |||
| if (that.responseTimeout) { | |||
| clearTimeout(that.responseTimeout); | |||
| } | |||
| reject("recv buffer full"); | |||
| } | |||
| } | |||
| if (packet.type == PacketTypeDataEnd) { | |||
| if (that.responseTimeout) { | |||
| clearTimeout(that.responseTimeout); | |||
| } | |||
| resolve(recvData.slice(0, recvDataCursor)); | |||
| } | |||
| } | |||
| this.connection.send(packet); | |||
| resetTimeoutCheck(); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype._sendAndWaitRecvCmd = function(cmd) { | |||
| return new Promise((resolve, reject) =>{ | |||
| var packet = new Packet(); | |||
| packet.setCmd(cmd); | |||
| this._sendAndWaitRecv(packet).then(packet => { | |||
| if (packet.type == PacketTypeCmdResp) { | |||
| resolve(packet.result); | |||
| } | |||
| }).catch(err => { | |||
| reject(err); | |||
| }); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype._enrollOne = async function(step) { | |||
| var down = false; | |||
| var leave = false; | |||
| var isTimeout = false; | |||
| this.timeoutFlag = setTimeout(() => { | |||
| if (!down) { | |||
| isTimeout = true; | |||
| } | |||
| }, this.timeout); | |||
| //wait finger leave | |||
| logln("wait leave") | |||
| this.callback(null, this.step, STATE_WAIT_FINGER_LEAVE, null); | |||
| while (leave == false && !isTimeout) { | |||
| if(this.canceled) { return; } | |||
| leave = !(await this.captureOne()); | |||
| } | |||
| this.callback(null, this.step, STATE_FINGER_LEAVE, null); | |||
| if (isTimeout) { | |||
| throw ErrorEnrollTimeout; | |||
| } | |||
| this.callback(null, this.step, STATE_WAIT_FINGER_DOWN, null); | |||
| //wait finger down | |||
| logln("wait down") | |||
| while (!down && !isTimeout) { | |||
| if(this.canceled) { return; } | |||
| down = await this.captureOne(); | |||
| } | |||
| if (isTimeout) { | |||
| throw ErrorEnrollTimeout; | |||
| } | |||
| this.callback(null, this.step, STATE_FINGER_DOWN, null); | |||
| logln("finger down step = " + step); | |||
| if(this.canceled) { return; } | |||
| this.callback(null, this.step, STATE_EXTRACT_TEMPLATE, null); | |||
| var ret = await this.extractOne(step); | |||
| if (ret != 0) { | |||
| logln("extract tempate err " + ret); | |||
| return ret; | |||
| } | |||
| return 0; | |||
| } | |||
| EnrollSchedule.prototype.extractOne = function(step) { | |||
| return new Promise((resolve, reject) =>{ | |||
| var data = Uint8Array.of(0x02, step); | |||
| var packet = new Packet(data, 2); | |||
| this._sendAndWaitRecv(packet).then(packet => { | |||
| if (packet.type == PacketTypeCmdResp) { | |||
| resolve(packet.result); | |||
| } | |||
| }).catch(err => { | |||
| reject(err); | |||
| }); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype.captureOne = function() { | |||
| return new Promise((resolve, reject) => { | |||
| var packet = new Packet(); | |||
| packet.setCmd(0x01); | |||
| this._sendAndWaitRecv(packet).then(packet => { | |||
| if (packet.type == PacketTypeCmdResp) { | |||
| resolve(packet.result == 0); | |||
| } | |||
| }).catch(err => { | |||
| reject(err); | |||
| }); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype._sendAndWaitRecv = function(packet) { | |||
| return new Promise((resolve, reject) => { | |||
| var that = this; | |||
| this.responseCallback = (packet) => { | |||
| if (packet.type == PacketTypeCmdResp) { | |||
| if (that.responseTimeout) { | |||
| clearTimeout(that.responseTimeout); | |||
| that.responseTimeout = 0; | |||
| } | |||
| if (packet.result == 0x01) { | |||
| reject(ErrorReceiveDataErr); | |||
| }else { | |||
| resolve(packet); | |||
| } | |||
| } | |||
| } | |||
| this.connection.send(packet); | |||
| this.responseTimeout = setTimeout(() => { | |||
| that.connection.send(packet); | |||
| this.responseTimeout = setTimeout(() => { | |||
| reject(ErrorReceiveDataErr); | |||
| }, this.responseTime); | |||
| }, this.responseTime); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype._resolvePacket = function(packet) { | |||
| if (this.responseCallback) { | |||
| this.responseCallback(packet); | |||
| } | |||
| } | |||
| EnrollSchedule.prototype._connect = function() { | |||
| return new Promise((resolve, reject) => { | |||
| var that = this; | |||
| this.connection = new SerialConnection(); | |||
| this.connection.onConnect.addListener(function() { | |||
| logln("device " + that.device.path + " connected"); | |||
| resolve(); | |||
| }); | |||
| this.connection.onReceive.addListener(function(packet) { | |||
| that._resolvePacket(packet); | |||
| }); | |||
| this.connection.connect(this.device.path, this.device.bitrate); | |||
| setTimeout(() => { | |||
| reject("connect timeout"); | |||
| }, 500); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype._disConnect = function() { | |||
| if (this.connection) { | |||
| this.connection.disconnect(); | |||
| this.connection = null; | |||
| } | |||
| if (this.responseTimeout) { | |||
| clearTimeout(this.responseTimeout); | |||
| this.responseTimeout = 0; | |||
| } | |||
| if (this.timeoutFlag) { | |||
| clearTimeout(this.timeoutFlag); | |||
| this.timeoutFlag = 0; | |||
| } | |||
| } | |||
| EnrollSchedule.prototype.stop = function() { | |||
| this.canceled = true; | |||
| } | |||
| var enrollSchedule = null; | |||
| var tempData = null; | |||
| function startEnroll(enrollCount=6, timeout=10*1000, cb) { | |||
| if (!curDevice) { | |||
| logln("device not found"); | |||
| cb.call(null, { | |||
| err: ErrorDeviceNotFound | |||
| }, 0, null); | |||
| return; | |||
| } | |||
| enrollSchedule = new EnrollSchedule(curDevice, enrollCount, (err, step, state, data) => { | |||
| if (err) { | |||
| printError(status.err); | |||
| } | |||
| if(cb) { | |||
| cb.call(null, { | |||
| err: err, | |||
| state: state, | |||
| step: step, | |||
| data: data | |||
| }); | |||
| } | |||
| if (data) { | |||
| tempData = data; | |||
| } | |||
| }, timeout); | |||
| logln("start enroll"); | |||
| enrollSchedule.start(); | |||
| } | |||
| function stopEnroll() { | |||
| if (enrollSchedule) { | |||
| enrollSchedule.stop(); | |||
| enrollSchedule = null; | |||
| } | |||
| } | |||
| //////////////////////////////////////////////////////// | |||
| //test | |||
| EnrollSchedule.prototype.sleep = function(time) { | |||
| return new Promise((resolve, reject) => { | |||
| setTimeout(() => { | |||
| resolve(); | |||
| }, time); | |||
| }); | |||
| } | |||
| EnrollSchedule.prototype.downloadTemplate = async function(data) { | |||
| var cmdData = Uint8Array.of(0x09, 0x02); | |||
| var packet = new Packet(cmdData, 2); | |||
| this.connection.send(packet); | |||
| await this.sleep(1000); | |||
| var dataLength = data.length; | |||
| var packetLen = 256; | |||
| var sendCursor = 0; | |||
| var sendLen = 0; | |||
| var sendType = PacketTypeData; | |||
| logln("downloading"); | |||
| while (sendCursor < dataLength) { | |||
| sendLen = (dataLength-sendCursor > packetLen) ? packetLen : (dataLength-sendCursor); | |||
| sendType = (dataLength-sendCursor > packetLen) ? PacketTypeData : PacketTypeDataEnd; | |||
| var sendData = new Uint8Array(data.buffer, sendCursor, sendLen); | |||
| var dataPack = new Packet(sendData, sendLen, sendType); | |||
| this.connection.send(dataPack); | |||
| console.log("send "+ sendLen) | |||
| log('.'); | |||
| sendCursor += sendLen; | |||
| await this.sleep(100); | |||
| } | |||
| cmdData = Uint8Array.of(0x06, 0x02, 0x00, 0x03); | |||
| packet = new Packet(cmdData, 4); | |||
| this.connection.send(packet); | |||
| logln('.'); | |||
| await this.sleep(500); | |||
| } | |||
| async function downloadTemplate() { | |||
| if (!curDevice) { | |||
| var err = "device not found"; | |||
| logln(err); | |||
| return; | |||
| } | |||
| if (!tempData) { | |||
| var err = "please enroll"; | |||
| logln(err); | |||
| return; | |||
| } | |||
| var enroll = new EnrollSchedule(curDevice); | |||
| try { | |||
| await enroll._connect(); | |||
| await enroll.downloadTemplate(tempData); | |||
| logln("download complete"); | |||
| }catch(err) { | |||
| logln("download err: " + err); | |||
| } | |||
| await enroll._disConnect(); | |||
| } | |||
| async function matchTemplate() { | |||
| if (!curDevice) { | |||
| var err = "device not found"; | |||
| logln(err); | |||
| return; | |||
| } | |||
| var enroll = new EnrollSchedule(curDevice); | |||
| try { | |||
| await enroll._connect(); | |||
| await enroll._enrollOne(1); | |||
| var packData = Uint8Array.of(0x04, 0x01, 0x00, 0x03, 0x00, 0x03); | |||
| var packet = new Packet(packData, 6); | |||
| await new Promise((resolve, reject) =>{ | |||
| enroll._sendAndWaitRecv(packet).then(packet => { | |||
| if (packet.type == PacketTypeCmdResp) { | |||
| if(packet.result) { | |||
| logln("match fail"); | |||
| }else { | |||
| logln("match success score " + (packet.data[3] * 256 + packet.data[4])); | |||
| } | |||
| resolve(); | |||
| } | |||
| }).catch(err => { | |||
| reject("match err: " + err); | |||
| }); | |||
| }) | |||
| }catch(err) { | |||
| logln("download err: " + err); | |||
| } | |||
| await enroll._disConnect(); | |||
| } | |||
| //////////////////////////////////////////////////////// | |||
| //////////////////////////////////////////////////////// | |||
| function disableAll() { | |||
| document.querySelector("#enroll").disabled = true; | |||
| document.querySelector("#down").disabled = true; | |||
| document.querySelector("#match").disabled = true; | |||
| document.querySelector("#refresh").disabled = true; | |||
| document.querySelector("#export").disabled = true; | |||
| document.querySelector("#import").disabled = true; | |||
| } | |||
| function enableAll() { | |||
| document.querySelector("#enroll").disabled = false; | |||
| document.querySelector("#down").disabled = false; | |||
| document.querySelector("#match").disabled = false; | |||
| document.querySelector("#refresh").disabled = false; | |||
| document.querySelector("#export").disabled = false; | |||
| document.querySelector("#import").disabled = false; | |||
| } | |||
| var server = null; | |||
| onload = function() { | |||
| document.querySelector('#show-log').checked = showDebugLog; | |||
| document.querySelector('#show-log').onchange = function(e) { | |||
| showDebugLog = e.target.checked; | |||
| } | |||
| document.querySelector('#enroll').onclick = function(e) { | |||
| if (e.currentTarget.innerText == "Enroll") { | |||
| e.currentTarget.innerText = "Stop"; | |||
| startEnroll(6, 10* 1000, (status) => { | |||
| console.log(status.step); | |||
| if (status.err) { | |||
| console.log(status.err); | |||
| } | |||
| if (status.err || status.data) { | |||
| document.querySelector('#enroll').innerText = "Enroll"; | |||
| } | |||
| if (status.data) { | |||
| // console.log(data); | |||
| // console.log(buf2hex(data.buffer)); | |||
| } | |||
| }); | |||
| }else { | |||
| e.currentTarget.innerText = "Enroll"; | |||
| stopEnroll(); | |||
| } | |||
| } | |||
| document.querySelector('.msg').style="height:" + (window.innerHeight - 70) + "px"; | |||
| chrome.serial.getDevices((devices) => { | |||
| checkDevices(devices); | |||
| }); | |||
| document.querySelector("#down").onclick = async function(e) { | |||
| disableAll(); | |||
| try { | |||
| await downloadTemplate(); | |||
| }catch(err) { | |||
| logln("download err: " + err); | |||
| } | |||
| enableAll(); | |||
| } | |||
| document.querySelector("#match").onclick = async function(e) { | |||
| disableAll(); | |||
| try { | |||
| await matchTemplate(); | |||
| }catch(err) { | |||
| logln("match err: " + err); | |||
| } | |||
| enableAll(); | |||
| } | |||
| document.querySelector("#refresh").onclick = function(e) { | |||
| disableAll(); | |||
| chrome.serial.getDevices(async function(devices) { | |||
| try { | |||
| await checkDevices(devices); | |||
| }catch (e) { | |||
| console.log(e) | |||
| } | |||
| enableAll(); | |||
| }); | |||
| } | |||
| document.querySelector("#export").onclick = function(e) { | |||
| if (!tempData) { | |||
| logln("please enroll"); | |||
| return; | |||
| } | |||
| let reader = new FileReader(); | |||
| reader.onload = function(eve) { | |||
| if (eve.target.readyState == FileReader.DONE) | |||
| { | |||
| let a = document.createElement('a'); | |||
| a.download = "mafp_template.bin"; | |||
| a.href = this.result; | |||
| a.click(); | |||
| } | |||
| } | |||
| reader.readAsDataURL(new Blob([tempData])); | |||
| } | |||
| document.querySelector("#import").onclick = function(e) { | |||
| let fileInput = document.querySelector("#import-file"); | |||
| fileInput.value = ""; | |||
| fileInput.click(); | |||
| } | |||
| document.querySelector("#import-file").onchange = function(e) { | |||
| let files = e.target.files; | |||
| if (files && files.length) { | |||
| let reader = new FileReader(); | |||
| reader.onload = async function(ev) { | |||
| if (ev.total == 4096) { | |||
| tempData = new Uint8Array(this.result); | |||
| disableAll(); | |||
| try { | |||
| await downloadTemplate(); | |||
| }catch(err) { | |||
| logln("download err: " + err); | |||
| } | |||
| enableAll(); | |||
| }else { | |||
| logln("invalid file length " + ev.total); | |||
| } | |||
| } | |||
| reader.readAsArrayBuffer(files[0]); | |||
| } | |||
| } | |||
| const port = 9897; | |||
| if (http.Server && http.WebSocketServer) { | |||
| server = new http.Server(); | |||
| var wsServer = new http.WebSocketServer(server); | |||
| server.listen(port); | |||
| logln("ws socket server listen at " + port); | |||
| var connectedSocket = null; | |||
| wsServer.addEventListener('request', function(req) { | |||
| if (connectedSocket) { | |||
| req.reject(); | |||
| return; | |||
| } | |||
| console.log('Client connected'); | |||
| var socket = req.accept(); | |||
| connectedSocket = socket; | |||
| socket.addEventListener('message', function(e) { | |||
| var reqData = JSON.parse(e.data); | |||
| if (reqData.cmd == "enrollStart") { | |||
| startEnroll(reqData.config.enrollCount, | |||
| reqData.config.enrollTimeout, | |||
| (status) =>{ | |||
| var resp = { | |||
| err: status.err ? status.err : "", | |||
| state: status.state ? status.state : 0, | |||
| step: status.step | |||
| } | |||
| if (status.data) { | |||
| resp.data = Array.from(status.data); | |||
| } | |||
| connectedSocket.send(JSON.stringify(resp)); | |||
| }); | |||
| }else if (reqData.cmd == "enrollCancel") { | |||
| stopEnroll(); | |||
| } | |||
| }); | |||
| socket.addEventListener('close', function() { | |||
| connectedSocket = null; | |||
| stopEnroll(); | |||
| console.log('Client disconnected'); | |||
| if (chrome.runtime.lastError != undefined) { | |||
| console.log(chrome.runtime.lastError.message); | |||
| } | |||
| }); | |||
| return true; | |||
| }); | |||
| } | |||
| }; | |||
| onresize = function(e) { | |||
| document.querySelector('.msg').style="height:" + (e.currentTarget.innerHeight - 70) + "px"; | |||
| document.querySelector('#msg_end').scrollIntoView(); | |||
| } | |||
| @ -0,0 +1,30 @@ | |||
| { | |||
| "name": "MAFP Serial", | |||
| "version": "1.0", | |||
| "manifest_version": 2, | |||
| "minimum_chrome_version": "23", | |||
| "description": "microarray fingerprint serial control.", | |||
| "icons": | |||
| { | |||
| "48": "logo.png", | |||
| "128": "logo.png" | |||
| }, | |||
| "app": { | |||
| "background": { | |||
| "scripts": [ "background.js" ] | |||
| } | |||
| }, | |||
| "sockets": { | |||
| "tcp": { | |||
| "connect": "*" | |||
| }, | |||
| "tcpServer": { | |||
| "listen": "*" | |||
| } | |||
| }, | |||
| "permissions": [ | |||
| "serial" | |||
| ] | |||
| } | |||
| @ -0,0 +1,232 @@ | |||
| // Licensed under the Apache License, Version 2.0 (the "License"); | |||
| // you may not use this file except in compliance with the License. | |||
| // You may obtain a copy of the License at | |||
| // | |||
| // http://www.apache.org/licenses/LICENSE-2.0 | |||
| // | |||
| // Unless required by applicable law or agreed to in writing, software | |||
| // distributed under the License is distributed on an "AS IS" BASIS, | |||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |||
| // See the License for the specific language governing permissions and | |||
| // limitations under the License. | |||
| // Copyright 2005 Google Inc. All Rights Reserved. | |||
| /** | |||
| * @fileoverview SHA-1 cryptographic hash. | |||
| * Variable names follow the notation in FIPS PUB 180-3: | |||
| * http://csrc.nist.gov/publications/fips/fips180-3/fips180-3_final.pdf. | |||
| * | |||
| * Usage: | |||
| * var sha1 = new goog.crypt.sha1(); | |||
| * sha1.update(bytes); | |||
| * var hash = sha1.digest(); | |||
| * | |||
| */ | |||
| /** | |||
| * SHA-1 cryptographic hash constructor. | |||
| * | |||
| * The properties declared here are discussed in the above algorithm document. | |||
| * @constructor | |||
| */ | |||
| var Sha1 = function() { | |||
| /** | |||
| * Holds the previous values of accumulated variables a-e in the compress_ | |||
| * function. | |||
| * @type {Array.<number>} | |||
| * @private | |||
| */ | |||
| this.chain_ = []; | |||
| /** | |||
| * A buffer holding the partially computed hash result. | |||
| * @type {Array.<number>} | |||
| * @private | |||
| */ | |||
| this.buf_ = []; | |||
| /** | |||
| * An array of 80 bytes, each a part of the message to be hashed. Referred to | |||
| * as the message schedule in the docs. | |||
| * @type {Array.<number>} | |||
| * @private | |||
| */ | |||
| this.W_ = []; | |||
| /** | |||
| * Contains data needed to pad messages less than 64 bytes. | |||
| * @type {Array.<number>} | |||
| * @private | |||
| */ | |||
| this.pad_ = []; | |||
| this.pad_[0] = 128; | |||
| for (var i = 1; i < 64; ++i) { | |||
| this.pad_[i] = 0; | |||
| } | |||
| this.reset(); | |||
| }; | |||
| /** | |||
| * Resets the internal accumulator. | |||
| */ | |||
| Sha1.prototype.reset = function() { | |||
| this.chain_[0] = 0x67452301; | |||
| this.chain_[1] = 0xefcdab89; | |||
| this.chain_[2] = 0x98badcfe; | |||
| this.chain_[3] = 0x10325476; | |||
| this.chain_[4] = 0xc3d2e1f0; | |||
| this.inbuf_ = 0; | |||
| this.total_ = 0; | |||
| }; | |||
| /** | |||
| * Internal helper performing 32 bit left rotate. | |||
| * @return {number} w rotated left by r bits. | |||
| * @private | |||
| */ | |||
| Sha1.prototype.rotl_ = function(w, r) { | |||
| return ((w << r) | (w >>> (32 - r))) & 0xffffffff; | |||
| }; | |||
| /** | |||
| * Internal compress helper function. | |||
| * @param {Array} buf containing block to compress. | |||
| * @private | |||
| */ | |||
| Sha1.prototype.compress_ = function(buf) { | |||
| var W = this.W_; | |||
| // get 16 big endian words | |||
| for (var i = 0; i < 64; i += 4) { | |||
| var w = (buf[i] << 24) | | |||
| (buf[i + 1] << 16) | | |||
| (buf[i + 2] << 8) | | |||
| (buf[i + 3]); | |||
| W[i / 4] = w; | |||
| } | |||
| // expand to 80 words | |||
| for (var i = 16; i < 80; i++) { | |||
| W[i] = this.rotl_(W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16], 1); | |||
| } | |||
| var a = this.chain_[0]; | |||
| var b = this.chain_[1]; | |||
| var c = this.chain_[2]; | |||
| var d = this.chain_[3]; | |||
| var e = this.chain_[4]; | |||
| var f, k; | |||
| for (var i = 0; i < 80; i++) { | |||
| if (i < 40) { | |||
| if (i < 20) { | |||
| f = d ^ (b & (c ^ d)); | |||
| k = 0x5a827999; | |||
| } else { | |||
| f = b ^ c ^ d; | |||
| k = 0x6ed9eba1; | |||
| } | |||
| } else { | |||
| if (i < 60) { | |||
| f = (b & c) | (d & (b | c)); | |||
| k = 0x8f1bbcdc; | |||
| } else { | |||
| f = b ^ c ^ d; | |||
| k = 0xca62c1d6; | |||
| } | |||
| } | |||
| var t = (this.rotl_(a, 5) + f + e + k + W[i]) & 0xffffffff; | |||
| e = d; | |||
| d = c; | |||
| c = this.rotl_(b, 30); | |||
| b = a; | |||
| a = t; | |||
| } | |||
| this.chain_[0] = (this.chain_[0] + a) & 0xffffffff; | |||
| this.chain_[1] = (this.chain_[1] + b) & 0xffffffff; | |||
| this.chain_[2] = (this.chain_[2] + c) & 0xffffffff; | |||
| this.chain_[3] = (this.chain_[3] + d) & 0xffffffff; | |||
| this.chain_[4] = (this.chain_[4] + e) & 0xffffffff; | |||
| }; | |||
| /** | |||
| * Adds a byte array to internal accumulator. | |||
| * @param {Array.<number>} bytes to add to digest. | |||
| * @param {number} opt_length is # of bytes to compress. | |||
| */ | |||
| Sha1.prototype.update = function(bytes, opt_length) { | |||
| if (!opt_length) { | |||
| opt_length = bytes.length; | |||
| } | |||
| var n = 0; | |||
| // Optimize for 64 byte chunks at 64 byte boundaries. | |||
| if (this.inbuf_ == 0) { | |||
| while (n + 64 < opt_length) { | |||
| this.compress_(bytes.slice(n, n + 64)); | |||
| n += 64; | |||
| this.total_ += 64; | |||
| } | |||
| } | |||
| while (n < opt_length) { | |||
| this.buf_[this.inbuf_++] = bytes[n++]; | |||
| this.total_++; | |||
| if (this.inbuf_ == 64) { | |||
| this.inbuf_ = 0; | |||
| this.compress_(this.buf_); | |||
| // Pick up 64 byte chunks. | |||
| while (n + 64 < opt_length) { | |||
| this.compress_(bytes.slice(n, n + 64)); | |||
| n += 64; | |||
| this.total_ += 64; | |||
| } | |||
| } | |||
| } | |||
| }; | |||
| /** | |||
| * @return {Array} byte[20] containing finalized hash. | |||
| */ | |||
| Sha1.prototype.digest = function() { | |||
| var digest = []; | |||
| var totalBits = this.total_ * 8; | |||
| // Add pad 0x80 0x00*. | |||
| if (this.inbuf_ < 56) { | |||
| this.update(this.pad_, 56 - this.inbuf_); | |||
| } else { | |||
| this.update(this.pad_, 64 - (this.inbuf_ - 56)); | |||
| } | |||
| // Add # bits. | |||
| for (var i = 63; i >= 56; i--) { | |||
| this.buf_[i] = totalBits & 255; | |||
| totalBits >>>= 8; | |||
| } | |||
| this.compress_(this.buf_); | |||
| var n = 0; | |||
| for (var i = 0; i < 5; i++) { | |||
| for (var j = 24; j >= 0; j -= 8) { | |||
| digest[n++] = (this.chain_[i] >> j) & 255; | |||
| } | |||
| } | |||
| return digest; | |||
| }; | |||
| @ -0,0 +1,32 @@ | |||
| body { | |||
| font-family: "Helvetica Neue", Helvetica, sans-serif; | |||
| font-size: 13px; | |||
| width: 100%; | |||
| height: 100%;; | |||
| margin: 0; | |||
| } | |||
| html { | |||
| width: 100%; | |||
| height: 100%;; | |||
| } | |||
| #buffer { | |||
| line-height: 20px; | |||
| } | |||
| .msg { | |||
| overflow: auto; | |||
| padding: 20px; | |||
| } | |||
| .buttons { | |||
| position: absolute; | |||
| bottom: 0; | |||
| display: flex; | |||
| align-items: center; | |||
| justify-content: flex-start; | |||
| height: 50px; | |||
| } | |||
| button { | |||
| margin-left: 20px; | |||
| } | |||
| @ -0,0 +1,2 @@ | |||
| <br /> | |||
| <b>Warning</b>: readfile(): Filename cannot be empty in <b>/www/wwwroot/51tool.com/ico/download.php</b> on line <b>17</b><br /> | |||
| @ -0,0 +1,3 @@ | |||
| { | |||
| "tenantId": "200,300,400" | |||
| } | |||