├── .babelrc ├── .editorconfig ├── .gitignore ├── .postcssrc.js ├── README.md ├── build ├── build.js ├── check-versions.js ├── logo.png ├── utils.js ├── vue-loader.conf.js ├── webpack.base.conf.js ├── webpack.dev.conf.js └── webpack.prod.conf.js ├── config ├── dev.env.js ├── index.js └── prod.env.js ├── dist ├── favicon.ico ├── index.html └── static │ ├── css │ ├── app.1d365fa20d80077303afd74b3d5ba528.css │ ├── app.1d365fa20d80077303afd74b3d5ba528.css.map │ └── reset.css │ ├── fonts │ ├── element-icons.535877f.woff │ ├── element-icons.732389d.ttf │ ├── fontawesome-webfont.674f50d.eot │ ├── fontawesome-webfont.af7ae50.woff2 │ ├── fontawesome-webfont.b06871f.ttf │ └── fontawesome-webfont.fee66e7.woff │ ├── img │ ├── fontawesome-webfont.912ec66.svg │ └── icons.2704f3c.png │ └── js │ ├── app.d963b6f8b5c668253b76.js │ ├── app.d963b6f8b5c668253b76.js.map │ ├── manifest.c33f93f8098cff3fbb0d.js │ ├── manifest.c33f93f8098cff3fbb0d.js.map │ ├── vendor.4b8eb37f85a468e06321.js │ └── vendor.4b8eb37f85a468e06321.js.map ├── favicon.ico ├── index.html ├── package.json ├── src ├── App.vue ├── assets │ ├── logo.png │ └── topo │ │ ├── application.png │ │ ├── canvas_bg.jpg │ │ ├── cloud.png │ │ ├── compute.png │ │ ├── container.png │ │ ├── database.png │ │ ├── dbms.png │ │ ├── filesystem.png │ │ ├── icons.png │ │ ├── message.png │ │ ├── network.png │ │ ├── port.png │ │ ├── pull.svg │ │ ├── push.svg │ │ ├── root.png │ │ ├── router.png │ │ ├── sercurity.png │ │ ├── storage.png │ │ ├── subnet.png │ │ ├── virtualip.png │ │ ├── volume.png │ │ └── webserver.png ├── components │ ├── vFooter.vue │ ├── vHeader.vue │ └── vTopo │ │ ├── components │ │ ├── vShapebar.vue │ │ └── vTopoAttrPanel.vue │ │ └── vTopo.vue ├── config │ └── connectorRules.js ├── data │ ├── toolbarNodeData.js │ ├── topoData1.js │ ├── topoData2.js │ └── topoJson.js ├── less │ ├── index.css │ └── index.less ├── main.js ├── nlComponentLib │ ├── components │ │ ├── nlButton.vue │ │ └── nlPageTitle.vue │ └── index.js └── router │ └── index.js ├── static ├── .gitkeep └── css │ └── reset.css └── topoJson.json /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | ["env", { 4 | "modules": false 5 | }], 6 | "stage-2" 7 | ], 8 | "plugins": ["transform-runtime"] 9 | } 10 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | charset = utf-8 5 | indent_style = space 6 | indent_size = 2 7 | end_of_line = lf 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules/ 3 | npm-debug.log* 4 | yarn-debug.log* 5 | yarn-error.log* 6 | 7 | # Editor directories and files 8 | .idea 9 | .vscode 10 | *.suo 11 | *.ntvs* 12 | *.njsproj 13 | *.sln 14 | -------------------------------------------------------------------------------- /.postcssrc.js: -------------------------------------------------------------------------------- 1 | // https://github.com/michael-ciniawsky/postcss-load-config 2 | 3 | module.exports = { 4 | "plugins": { 5 | // to edit target browsers: use "browserslist" field in package.json 6 | "postcss-import": {}, 7 | "autoprefixer": {} 8 | } 9 | } 10 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | 996.icu 2 | # vue2.0拓扑组件 3 | ## demo演示地址 4 | https://mirror198829.github.io/vue-topo/dist/index.html 5 | ## 序 6 | 2016年9月接到html5替换flash的拓扑图任务,历时3月整,使用jquery+SVG技术完成第一款拓扑图组件,涵盖拖拽、缩放、删除、选中区域、连线、导入、保存图片、多种布局显示等功能,但当node数高达百位时,因dom的操作,出现了非常可怕的性能问题。
7 | 2017年10月20日,因新公司项目需要,果断决定结合当下流行的VUE2.0框架 + SVG技术 重写拓扑组件,并将此开源,希望大家多提意见 8 | ## 技术选型 9 | * VUE2.0 10 | * SVG 11 | * Element-UI 12 | ## 功能分析及技术点 13 | | 功能 | 技术点 | 14 | |-------------|-----| 15 | |toolbar拖拽|记录鼠标位置;
拖拽物体的显示隐藏;| 16 | |连接关系:创建连线|绘制时:使用svg的 ``,需正确获取起始和终止点;
结束绘制:使用``路径绘制连线形态,根据两者节点位置不同,共计4种连线形态展示;| 17 | |包含关系:单层嵌套|依据最终的位置判断是否为包含关系(这里的算法有待优化);
重新计算父节点宽高: 父结点宽高为子节点宽高+边距
重新计算子节点的位置信息:父节点位置信息 - 边距| 18 | |包含关系:多层嵌套|通过递归的方式完成嵌套布局| 19 | |包含关系:并列嵌套|父节点宽高:内部子节点宽度之和 + 边距,父节点高度:最高子节点高度 + 边距;
每个子节点位置:当前节点位置 = 前一个节点位置 + 前一个节点宽度 + 间隔边距| 20 | |删除:单结点|删除节点同时删除关联连线(source&target| 21 | |删除:包含关系|删除包含关系同时恢复父节点的宽高同时刷新并列节点位置| 22 | |删除:连线|刷新原本有连线的连接点样式| 23 | |删除:多节点|使用for循环,注意i--| 24 | |框选|使用``,根据鼠标移动位移的正负数,选择不同的x和y值| 25 | |svg的viewbox的移动功能 |记录鼠标移动的位移,修改viewbox的视图位置| 26 | 27 | ## Usage 28 | #### NPM 29 | ``` 30 | npm i vue-topo -save 31 | ``` 32 | ``` 33 | import vTopo from 'vue-topo' 34 | import 'vue-topo/dist/vue-topo.min.css' 35 | Vue.use(vTopo) 36 | ``` 37 | #### github源码使用 38 | ``` 39 | npm install 40 | ``` 41 | ``` 42 | npm run dev //启动服务 43 | ``` 44 | #### 引入组件 45 | ``` 46 | // xxx.vue 47 | 50 | 51 | ``` 52 | #### Attributes 53 | | 参数 | 说明 | 类型 | 可选值 | 默认值 | 必填 | 54 | | --- | --- | --- | --- | --- | --- | 55 | | editable | 是否可编辑 | boolean | true/false | true | — | 56 | | topo-data | topo初始值 | object | — | {nodes:[],connectors:[]} | 是 | 57 | ## 说明 58 | 代码开源,欢迎码农们提出宝贵意见,bug请提issues,本人将及时修改。最后著作权归一步工程师,请大家注明代码来源:https://github.com/Mirror198829/vue-topo 59 | -------------------------------------------------------------------------------- /build/build.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | require('./check-versions')() 3 | 4 | process.env.NODE_ENV = 'production' 5 | 6 | const ora = require('ora') 7 | const rm = require('rimraf') 8 | const path = require('path') 9 | const chalk = require('chalk') 10 | const webpack = require('webpack') 11 | const config = require('../config') 12 | const webpackConfig = require('./webpack.prod.conf') 13 | 14 | const spinner = ora('building for production...') 15 | spinner.start() 16 | 17 | rm(path.join(config.build.assetsRoot, config.build.assetsSubDirectory), err => { 18 | if (err) throw err 19 | webpack(webpackConfig, function (err, stats) { 20 | spinner.stop() 21 | if (err) throw err 22 | process.stdout.write(stats.toString({ 23 | colors: true, 24 | modules: false, 25 | children: false, 26 | chunks: false, 27 | chunkModules: false 28 | }) + '\n\n') 29 | 30 | if (stats.hasErrors()) { 31 | console.log(chalk.red(' Build failed with errors.\n')) 32 | process.exit(1) 33 | } 34 | 35 | console.log(chalk.cyan(' Build complete.\n')) 36 | console.log(chalk.yellow( 37 | ' Tip: built files are meant to be served over an HTTP server.\n' + 38 | ' Opening index.html over file:// won\'t work.\n' 39 | )) 40 | }) 41 | }) 42 | -------------------------------------------------------------------------------- /build/check-versions.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const chalk = require('chalk') 3 | const semver = require('semver') 4 | const packageConfig = require('../package.json') 5 | const shell = require('shelljs') 6 | function exec (cmd) { 7 | return require('child_process').execSync(cmd).toString().trim() 8 | } 9 | 10 | const versionRequirements = [ 11 | { 12 | name: 'node', 13 | currentVersion: semver.clean(process.version), 14 | versionRequirement: packageConfig.engines.node 15 | } 16 | ] 17 | 18 | if (shell.which('npm')) { 19 | versionRequirements.push({ 20 | name: 'npm', 21 | currentVersion: exec('npm --version'), 22 | versionRequirement: packageConfig.engines.npm 23 | }) 24 | } 25 | 26 | module.exports = function () { 27 | const warnings = [] 28 | for (let i = 0; i < versionRequirements.length; i++) { 29 | const mod = versionRequirements[i] 30 | if (!semver.satisfies(mod.currentVersion, mod.versionRequirement)) { 31 | warnings.push(mod.name + ': ' + 32 | chalk.red(mod.currentVersion) + ' should be ' + 33 | chalk.green(mod.versionRequirement) 34 | ) 35 | } 36 | } 37 | 38 | if (warnings.length) { 39 | console.log('') 40 | console.log(chalk.yellow('To use this template, you must update following to modules:')) 41 | console.log() 42 | for (let i = 0; i < warnings.length; i++) { 43 | const warning = warnings[i] 44 | console.log(' ' + warning) 45 | } 46 | console.log() 47 | process.exit(1) 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /build/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/build/logo.png -------------------------------------------------------------------------------- /build/utils.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const path = require('path') 3 | const config = require('../config') 4 | const ExtractTextPlugin = require('extract-text-webpack-plugin') 5 | const pkg = require('../package.json') 6 | 7 | exports.assetsPath = function (_path) { 8 | const assetsSubDirectory = process.env.NODE_ENV === 'production' 9 | ? config.build.assetsSubDirectory 10 | : config.dev.assetsSubDirectory 11 | return path.posix.join(assetsSubDirectory, _path) 12 | } 13 | 14 | exports.cssLoaders = function (options) { 15 | options = options || {} 16 | 17 | const cssLoader = { 18 | loader: 'css-loader', 19 | options: { 20 | sourceMap: options.sourceMap 21 | } 22 | } 23 | 24 | var postcssLoader = { 25 | loader: 'postcss-loader', 26 | options: { 27 | sourceMap: options.sourceMap 28 | } 29 | } 30 | 31 | // generate loader string to be used with extract text plugin 32 | function generateLoaders (loader, loaderOptions) { 33 | const loaders = options.usePostCSS ? [cssLoader, postcssLoader] : [cssLoader] 34 | if (loader) { 35 | loaders.push({ 36 | loader: loader + '-loader', 37 | options: Object.assign({}, loaderOptions, { 38 | sourceMap: options.sourceMap 39 | }) 40 | }) 41 | } 42 | 43 | // Extract CSS when that option is specified 44 | // (which is the case during production build) 45 | if (options.extract) { 46 | return ExtractTextPlugin.extract({ 47 | use: loaders, 48 | fallback: 'vue-style-loader' 49 | }) 50 | } else { 51 | return ['vue-style-loader'].concat(loaders) 52 | } 53 | } 54 | 55 | // https://vue-loader.vuejs.org/en/configurations/extract-css.html 56 | return { 57 | css: generateLoaders(), 58 | postcss: generateLoaders(), 59 | less: generateLoaders('less').concat({ 60 | loader: 'sass-resources-loader', 61 | options: { 62 | resources: path.resolve(__dirname, '../src/less/index.less') 63 | } 64 | }), 65 | sass: generateLoaders('sass', { indentedSyntax: true }), 66 | scss: generateLoaders('sass'), 67 | stylus: generateLoaders('stylus'), 68 | styl: generateLoaders('stylus') 69 | } 70 | } 71 | 72 | // Generate loaders for standalone style files (outside of .vue) 73 | exports.styleLoaders = function (options) { 74 | const output = [] 75 | const loaders = exports.cssLoaders(options) 76 | for (const extension in loaders) { 77 | const loader = loaders[extension] 78 | output.push({ 79 | test: new RegExp('\\.' + extension + '$'), 80 | use: loader 81 | }) 82 | } 83 | return output 84 | } 85 | 86 | exports.createNotifierCallback = function () { 87 | const notifier = require('node-notifier') 88 | 89 | return (severity, errors) => { 90 | if (severity !== 'error') { 91 | return 92 | } 93 | const error = errors[0] 94 | 95 | const filename = error.file && error.file.split('!').pop() 96 | notifier.notify({ 97 | title: pkg.name, 98 | message: severity + ': ' + error.name, 99 | subtitle: filename || '', 100 | icon: path.join(__dirname, 'logo.png') 101 | }) 102 | } 103 | } 104 | -------------------------------------------------------------------------------- /build/vue-loader.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const utils = require('./utils') 3 | const config = require('../config') 4 | const isProduction = process.env.NODE_ENV === 'production' 5 | const sourceMapEnabled = isProduction 6 | ? config.build.productionSourceMap 7 | : config.dev.cssSourceMap 8 | 9 | 10 | module.exports = { 11 | loaders: utils.cssLoaders({ 12 | sourceMap: sourceMapEnabled, 13 | extract: isProduction 14 | }), 15 | cssSourceMap: sourceMapEnabled, 16 | cacheBusting: config.dev.cacheBusting, 17 | transformToRequire: { 18 | video: 'src', 19 | source: 'src', 20 | img: 'src', 21 | image: 'xlink:href' 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /build/webpack.base.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const path = require('path') 3 | const utils = require('./utils') 4 | const config = require('../config') 5 | const vueLoaderConfig = require('./vue-loader.conf') 6 | const webpack = require("webpack") 7 | 8 | function resolve (dir) { 9 | return path.join(__dirname, '..', dir) 10 | } 11 | 12 | module.exports = { 13 | context: path.resolve(__dirname, '../'), 14 | entry: { 15 | app: './src/main.js' 16 | }, 17 | output: { 18 | path: config.build.assetsRoot, 19 | filename: '[name].js', 20 | publicPath: process.env.NODE_ENV === 'production' 21 | ? config.build.assetsPublicPath 22 | : config.dev.assetsPublicPath 23 | }, 24 | resolve: { 25 | extensions: ['.js', '.vue', '.json'], 26 | alias: { 27 | 'vue$': 'vue/dist/vue.esm.js', 28 | '@': resolve('src'), 29 | } 30 | }, 31 | plugins: [ 32 | new webpack.ProvidePlugin({ 33 | jQuery: "jquery", 34 | $: "jquery" 35 | }) 36 | ], 37 | module: { 38 | rules: [ 39 | { 40 | test: /\.vue$/, 41 | loader: 'vue-loader', 42 | options: vueLoaderConfig 43 | }, 44 | { 45 | test: /\.js$/, 46 | loader: 'babel-loader', 47 | include: [resolve('src'), resolve('test')] 48 | }, 49 | { 50 | test: /\.(png|jpe?g|gif|svg)(\?.*)?$/, 51 | loader: 'url-loader', 52 | options: { 53 | limit: 10000, 54 | name: utils.assetsPath('img/[name].[hash:7].[ext]') 55 | } 56 | }, 57 | { 58 | test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/, 59 | loader: 'url-loader', 60 | options: { 61 | limit: 10000, 62 | name: utils.assetsPath('media/[name].[hash:7].[ext]') 63 | } 64 | }, 65 | { 66 | test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/, 67 | loader: 'url-loader', 68 | options: { 69 | limit: 10000, 70 | name: utils.assetsPath('fonts/[name].[hash:7].[ext]') 71 | } 72 | } 73 | ] 74 | } 75 | } 76 | -------------------------------------------------------------------------------- /build/webpack.dev.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const utils = require('./utils') 3 | const webpack = require('webpack') 4 | const config = require('../config') 5 | const merge = require('webpack-merge') 6 | const baseWebpackConfig = require('./webpack.base.conf') 7 | const HtmlWebpackPlugin = require('html-webpack-plugin') 8 | const FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin') 9 | const portfinder = require('portfinder') 10 | 11 | const devWebpackConfig = merge(baseWebpackConfig, { 12 | module: { 13 | rules: utils.styleLoaders({ sourceMap: config.dev.cssSourceMap, usePostCSS: true }) 14 | }, 15 | // cheap-module-eval-source-map is faster for development 16 | devtool: config.dev.devtool, 17 | 18 | // these devServer options should be customized in /config/index.js 19 | devServer: { 20 | clientLogLevel: 'warning', 21 | historyApiFallback: true, 22 | hot: true, 23 | compress: true, 24 | host: process.env.HOST || config.dev.host, 25 | port: process.env.PORT || config.dev.port, 26 | open: config.dev.autoOpenBrowser, 27 | overlay: config.dev.errorOverlay ? { 28 | warnings: false, 29 | errors: true, 30 | } : false, 31 | publicPath: config.dev.assetsPublicPath, 32 | proxy: config.dev.proxyTable, 33 | quiet: true, // necessary for FriendlyErrorsPlugin 34 | watchOptions: { 35 | poll: config.dev.poll, 36 | } 37 | }, 38 | plugins: [ 39 | new webpack.DefinePlugin({ 40 | 'process.env': require('../config/dev.env') 41 | }), 42 | new webpack.HotModuleReplacementPlugin(), 43 | new webpack.NamedModulesPlugin(), // HMR shows correct file names in console on update. 44 | new webpack.NoEmitOnErrorsPlugin(), 45 | // https://github.com/ampedandwired/html-webpack-plugin 46 | new HtmlWebpackPlugin({ 47 | filename: 'index.html', 48 | template: 'index.html', 49 | inject: true 50 | }), 51 | ] 52 | }) 53 | 54 | module.exports = new Promise((resolve, reject) => { 55 | portfinder.basePort = process.env.PORT || config.dev.port 56 | portfinder.getPort((err, port) => { 57 | if (err) { 58 | reject(err) 59 | } else { 60 | // publish the new Port, necessary for e2e tests 61 | process.env.PORT = port 62 | // add port to devServer config 63 | devWebpackConfig.devServer.port = port 64 | 65 | // Add FriendlyErrorsPlugin 66 | devWebpackConfig.plugins.push(new FriendlyErrorsPlugin({ 67 | compilationSuccessInfo: { 68 | messages: [`Your application is running here: http://${config.dev.host}:${port}`], 69 | }, 70 | onErrors: config.dev.notifyOnErrors 71 | ? utils.createNotifierCallback() 72 | : undefined 73 | })) 74 | 75 | resolve(devWebpackConfig) 76 | } 77 | }) 78 | }) 79 | -------------------------------------------------------------------------------- /build/webpack.prod.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const path = require('path') 3 | const utils = require('./utils') 4 | const webpack = require('webpack') 5 | const config = require('../config') 6 | const merge = require('webpack-merge') 7 | const baseWebpackConfig = require('./webpack.base.conf') 8 | const CopyWebpackPlugin = require('copy-webpack-plugin') 9 | const HtmlWebpackPlugin = require('html-webpack-plugin') 10 | const ExtractTextPlugin = require('extract-text-webpack-plugin') 11 | const OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin') 12 | 13 | const env = require('../config/prod.env') 14 | 15 | const webpackConfig = merge(baseWebpackConfig, { 16 | module: { 17 | rules: utils.styleLoaders({ 18 | sourceMap: config.build.productionSourceMap, 19 | extract: true, 20 | usePostCSS: true 21 | }) 22 | }, 23 | devtool: config.build.productionSourceMap ? config.build.devtool : false, 24 | output: { 25 | path: config.build.assetsRoot, 26 | filename: utils.assetsPath('js/[name].[chunkhash].js'), 27 | chunkFilename: utils.assetsPath('js/[id].[chunkhash].js') 28 | }, 29 | plugins: [ 30 | // http://vuejs.github.io/vue-loader/en/workflow/production.html 31 | new webpack.DefinePlugin({ 32 | 'process.env': env 33 | }), 34 | // UglifyJs do not support ES6+, you can also use babel-minify for better treeshaking: https://github.com/babel/minify 35 | new webpack.optimize.UglifyJsPlugin({ 36 | compress: { 37 | warnings: false 38 | }, 39 | sourceMap: config.build.productionSourceMap, 40 | parallel: true 41 | }), 42 | // extract css into its own file 43 | new ExtractTextPlugin({ 44 | filename: utils.assetsPath('css/[name].[contenthash].css'), 45 | // set the following option to `true` if you want to extract CSS from 46 | // codesplit chunks into this main css file as well. 47 | // This will result in *all* of your app's CSS being loaded upfront. 48 | allChunks: false, 49 | }), 50 | // Compress extracted CSS. We are using this plugin so that possible 51 | // duplicated CSS from different components can be deduped. 52 | new OptimizeCSSPlugin({ 53 | cssProcessorOptions: config.build.productionSourceMap 54 | ? { safe: true, map: { inline: false } } 55 | : { safe: true } 56 | }), 57 | // generate dist index.html with correct asset hash for caching. 58 | // you can customize output by editing /index.html 59 | // see https://github.com/ampedandwired/html-webpack-plugin 60 | new HtmlWebpackPlugin({ 61 | filename: config.build.index, 62 | template: 'index.html', 63 | favicon: 'favicon.ico', 64 | inject: true, 65 | minify: { 66 | removeComments: true, 67 | collapseWhitespace: true, 68 | removeAttributeQuotes: true 69 | // more options: 70 | // https://github.com/kangax/html-minifier#options-quick-reference 71 | }, 72 | // necessary to consistently work with multiple chunks via CommonsChunkPlugin 73 | chunksSortMode: 'dependency' 74 | }), 75 | // keep module.id stable when vender modules does not change 76 | new webpack.HashedModuleIdsPlugin(), 77 | // enable scope hoisting 78 | new webpack.optimize.ModuleConcatenationPlugin(), 79 | // split vendor js into its own file 80 | new webpack.optimize.CommonsChunkPlugin({ 81 | name: 'vendor', 82 | minChunks: function (module) { 83 | // any required modules inside node_modules are extracted to vendor 84 | return ( 85 | module.resource && 86 | /\.js$/.test(module.resource) && 87 | module.resource.indexOf( 88 | path.join(__dirname, '../node_modules') 89 | ) === 0 90 | ) 91 | } 92 | }), 93 | // extract webpack runtime and module manifest to its own file in order to 94 | // prevent vendor hash from being updated whenever app bundle is updated 95 | new webpack.optimize.CommonsChunkPlugin({ 96 | name: 'manifest', 97 | minChunks: Infinity 98 | }), 99 | // This instance extracts shared chunks from code splitted chunks and bundles them 100 | // in a separate chunk, similar to the vendor chunk 101 | // see: https://webpack.js.org/plugins/commons-chunk-plugin/#extra-async-commons-chunk 102 | new webpack.optimize.CommonsChunkPlugin({ 103 | name: 'app', 104 | async: 'vendor-async', 105 | children: true, 106 | minChunks: 3 107 | }), 108 | 109 | // copy custom static assets 110 | new CopyWebpackPlugin([ 111 | { 112 | from: path.resolve(__dirname, '../static'), 113 | to: config.build.assetsSubDirectory, 114 | ignore: ['.*'] 115 | } 116 | ]) 117 | ] 118 | }) 119 | 120 | if (config.build.productionGzip) { 121 | const CompressionWebpackPlugin = require('compression-webpack-plugin') 122 | 123 | webpackConfig.plugins.push( 124 | new CompressionWebpackPlugin({ 125 | asset: '[path].gz[query]', 126 | algorithm: 'gzip', 127 | test: new RegExp( 128 | '\\.(' + 129 | config.build.productionGzipExtensions.join('|') + 130 | ')$' 131 | ), 132 | threshold: 10240, 133 | minRatio: 0.8 134 | }) 135 | ) 136 | } 137 | 138 | if (config.build.bundleAnalyzerReport) { 139 | const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin 140 | webpackConfig.plugins.push(new BundleAnalyzerPlugin()) 141 | } 142 | 143 | module.exports = webpackConfig 144 | -------------------------------------------------------------------------------- /config/dev.env.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const merge = require('webpack-merge') 3 | const prodEnv = require('./prod.env') 4 | 5 | module.exports = merge(prodEnv, { 6 | NODE_ENV: '"development"' 7 | }) 8 | -------------------------------------------------------------------------------- /config/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | // Template version: 1.2.4 3 | // see http://vuejs-templates.github.io/webpack for documentation. 4 | 5 | const path = require('path') 6 | 7 | module.exports = { 8 | dev: { 9 | 10 | // Paths 11 | assetsSubDirectory: 'static', 12 | assetsPublicPath: '/', 13 | proxyTable: {}, 14 | 15 | // Various Dev Server settings 16 | host: 'localhost', // can be overwritten by process.env.HOST 17 | port: 8086, // can be overwritten by process.env.PORT, if port is in use, a free one will be determined 18 | autoOpenBrowser: true, 19 | errorOverlay: true, 20 | notifyOnErrors: true, 21 | poll: false, // https://webpack.js.org/configuration/dev-server/#devserver-watchoptions- 22 | 23 | // Use Eslint Loader? 24 | // If true, your code will be linted during bundling and 25 | // linting errors and warnings will be shown in the console. 26 | useEslint: true, 27 | // If true, eslint errors and warnings will also be shown in the error overlay 28 | // in the browser. 29 | showEslintErrorsInOverlay: false, 30 | 31 | /** 32 | * Source Maps 33 | */ 34 | 35 | // https://webpack.js.org/configuration/devtool/#development 36 | devtool: 'eval-source-map', 37 | 38 | // If you have problems debugging vue-files in devtools, 39 | // set this to false - it *may* help 40 | // https://vue-loader.vuejs.org/en/options.html#cachebusting 41 | cacheBusting: true, 42 | 43 | // CSS Sourcemaps off by default because relative paths are "buggy" 44 | // with this option, according to the CSS-Loader README 45 | // (https://github.com/webpack/css-loader#sourcemaps) 46 | // In our experience, they generally work as expected, 47 | // just be aware of this issue when enabling this option. 48 | cssSourceMap: false, 49 | }, 50 | 51 | build: { 52 | // Template for index.html 53 | index: path.resolve(__dirname, '../dist/index.html'), 54 | 55 | // Paths 56 | assetsRoot: path.resolve(__dirname, '../dist'), 57 | assetsSubDirectory: 'static', 58 | assetsPublicPath: '/vue-topo/dist/', 59 | //assetsPublicPath: '/', 60 | 61 | /** 62 | * Source Maps 63 | */ 64 | 65 | productionSourceMap: true, 66 | // https://webpack.js.org/configuration/devtool/#production 67 | devtool: '#source-map', 68 | 69 | // Gzip off by default as many popular static hosts such as 70 | // Surge or Netlify already gzip all static assets for you. 71 | // Before setting to `true`, make sure to: 72 | // npm install --save-dev compression-webpack-plugin 73 | productionGzip: false, 74 | productionGzipExtensions: ['js', 'css'], 75 | 76 | // Run the build command with an extra argument to 77 | // View the bundle analyzer report after build finishes: 78 | // `npm run build --report` 79 | // Set to `true` or `false` to always turn it on or off 80 | bundleAnalyzerReport: process.env.npm_config_report 81 | } 82 | } 83 | -------------------------------------------------------------------------------- /config/prod.env.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | module.exports = { 3 | NODE_ENV: '"production"' 4 | } 5 | -------------------------------------------------------------------------------- /dist/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/favicon.ico -------------------------------------------------------------------------------- /dist/index.html: -------------------------------------------------------------------------------- 1 | vueTopo
-------------------------------------------------------------------------------- /dist/static/css/reset.css: -------------------------------------------------------------------------------- 1 | html, 2 | body { 3 | height: 100%; 4 | width: 100%; 5 | } 6 | 7 | body { 8 | font-family: arial, helvetica, sans-serif; 9 | font-size: 13px; 10 | font-size-adjust: none; 11 | font-stretch: normal; 12 | font-style: normal; 13 | font-variant: normal; 14 | font-weight: normal; 15 | line-height: 1.4; 16 | text-align: center; 17 | } 18 | 19 | body, 20 | ul, 21 | ol, 22 | dl, 23 | dd, 24 | h1, 25 | h2, 26 | h3, 27 | h4, 28 | h5, 29 | h6, 30 | p, 31 | form, 32 | fieldset, 33 | legend, 34 | input, 35 | textarea, 36 | select, 37 | button, 38 | th, 39 | td { 40 | margin: 0; 41 | padding: 0; 42 | } 43 | 44 | h1, 45 | h2, 46 | h3, 47 | h4, 48 | h5, 49 | h6 { 50 | font-size: 100%; 51 | font-weight: normal; 52 | } 53 | 54 | table { 55 | font-size: inherit; 56 | } 57 | 58 | input, 59 | select { 60 | font-family: arial, helvetica, clean, sans-serif; 61 | font-size: 100%; 62 | font-size-adjust: none; 63 | font-stretch: normal; 64 | font-style: normal; 65 | font-variant: normal; 66 | font-weight: normal; 67 | line-height: normal; 68 | } 69 | 70 | button { 71 | overflow: visible; 72 | } 73 | 74 | th, 75 | em, 76 | strong, 77 | b, 78 | address, 79 | cite { 80 | font-style: normal; 81 | font-weight: normal; 82 | } 83 | 84 | li { 85 | list-style-image: none; 86 | list-style-position: outside; 87 | list-style-type: none; 88 | } 89 | 90 | img, 91 | fieldset { 92 | border: 0 none; 93 | } 94 | 95 | ins { 96 | text-decoration: none; 97 | } 98 | 99 | .clearfix { 100 | zoom: 1; 101 | } 102 | 103 | .clearfix:after { 104 | content: ''; 105 | display: block; 106 | clear: both; 107 | } 108 | -------------------------------------------------------------------------------- /dist/static/fonts/element-icons.535877f.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/fonts/element-icons.535877f.woff -------------------------------------------------------------------------------- /dist/static/fonts/element-icons.732389d.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/fonts/element-icons.732389d.ttf -------------------------------------------------------------------------------- /dist/static/fonts/fontawesome-webfont.674f50d.eot: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/fonts/fontawesome-webfont.674f50d.eot -------------------------------------------------------------------------------- /dist/static/fonts/fontawesome-webfont.af7ae50.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/fonts/fontawesome-webfont.af7ae50.woff2 -------------------------------------------------------------------------------- /dist/static/fonts/fontawesome-webfont.b06871f.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/fonts/fontawesome-webfont.b06871f.ttf -------------------------------------------------------------------------------- /dist/static/fonts/fontawesome-webfont.fee66e7.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/fonts/fontawesome-webfont.fee66e7.woff -------------------------------------------------------------------------------- /dist/static/img/icons.2704f3c.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/dist/static/img/icons.2704f3c.png -------------------------------------------------------------------------------- /dist/static/js/manifest.c33f93f8098cff3fbb0d.js: -------------------------------------------------------------------------------- 1 | !function(r){function n(o){if(t[o])return t[o].exports;var e=t[o]={i:o,l:!1,exports:{}};return r[o].call(e.exports,e,e.exports,n),e.l=!0,e.exports}var o=window.webpackJsonp;window.webpackJsonp=function(t,u,c){for(var f,i,p,a=0,l=[];a 2 | 3 | 4 | 5 | vueTopo 6 | 7 | 8 | 9 |
10 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vue-topo", 3 | "version": "1.0.0", 4 | "description": "A Vue.js project", 5 | "author": "'Mirror198829' <'412366756@qq.com'>", 6 | "private": true, 7 | "scripts": { 8 | "dev": "webpack-dev-server --inline --progress --config build/webpack.dev.conf.js", 9 | "start": "npm run dev", 10 | "build": "node build/build.js" 11 | }, 12 | "dependencies": { 13 | "vue": "^2.5.2", 14 | "vue-router": "^3.0.1" 15 | }, 16 | "devDependencies": { 17 | "autoprefixer": "^7.1.2", 18 | "babel-core": "^6.22.1", 19 | "babel-loader": "^7.1.1", 20 | "babel-plugin-transform-runtime": "^6.22.0", 21 | "babel-preset-env": "^1.3.2", 22 | "babel-preset-stage-2": "^6.22.0", 23 | "babel-register": "^6.22.0", 24 | "chalk": "^2.0.1", 25 | "copy-webpack-plugin": "^4.0.1", 26 | "css-loader": "^0.28.0", 27 | "element-ui": "^2.0.5", 28 | "eventsource-polyfill": "^0.9.6", 29 | "extract-text-webpack-plugin": "^3.0.0", 30 | "file-loader": "^1.1.4", 31 | "font-awesome": "^4.7.0", 32 | "friendly-errors-webpack-plugin": "^1.6.1", 33 | "html-webpack-plugin": "^2.30.1", 34 | "jquery": "^3.2.1", 35 | "less": "^2.7.3", 36 | "less-loader": "^4.0.5", 37 | "node-notifier": "^5.1.2", 38 | "optimize-css-assets-webpack-plugin": "^3.2.0", 39 | "ora": "^1.2.0", 40 | "portfinder": "^1.0.13", 41 | "postcss-import": "^11.0.0", 42 | "postcss-loader": "^2.0.8", 43 | "rimraf": "^2.6.0", 44 | "sass-resources-loader": "^2.0.0", 45 | "save-svg-as-png": "^1.2.0", 46 | "semver": "^5.3.0", 47 | "shelljs": "^0.7.6", 48 | "url-loader": "^0.5.8", 49 | "vue-loader": "^13.3.0", 50 | "vue-style-loader": "^3.0.1", 51 | "vue-template-compiler": "^2.5.2", 52 | "webpack": "^3.6.0", 53 | "webpack-bundle-analyzer": "^3.9.0", 54 | "webpack-dev-server": "^2.9.1", 55 | "webpack-merge": "^4.1.0" 56 | }, 57 | "engines": { 58 | "node": ">= 4.0.0", 59 | "npm": ">= 3.0.0" 60 | }, 61 | "browserslist": [ 62 | "> 1%", 63 | "last 2 versions", 64 | "not ie <= 8" 65 | ] 66 | } 67 | -------------------------------------------------------------------------------- /src/App.vue: -------------------------------------------------------------------------------- 1 | 7 | 25 | 26 | 76 | 77 | 82 | -------------------------------------------------------------------------------- /src/assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/logo.png -------------------------------------------------------------------------------- /src/assets/topo/application.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/application.png -------------------------------------------------------------------------------- /src/assets/topo/canvas_bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/canvas_bg.jpg -------------------------------------------------------------------------------- /src/assets/topo/cloud.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/cloud.png -------------------------------------------------------------------------------- /src/assets/topo/compute.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/compute.png -------------------------------------------------------------------------------- /src/assets/topo/container.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/container.png -------------------------------------------------------------------------------- /src/assets/topo/database.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/database.png -------------------------------------------------------------------------------- /src/assets/topo/dbms.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/dbms.png -------------------------------------------------------------------------------- /src/assets/topo/filesystem.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/filesystem.png -------------------------------------------------------------------------------- /src/assets/topo/icons.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/icons.png -------------------------------------------------------------------------------- /src/assets/topo/message.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/message.png -------------------------------------------------------------------------------- /src/assets/topo/network.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/network.png -------------------------------------------------------------------------------- /src/assets/topo/port.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/port.png -------------------------------------------------------------------------------- /src/assets/topo/pull.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Group 9 Copy 2 5 | Created with Sketch. 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /src/assets/topo/push.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Group 9 5 | Created with Sketch. 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /src/assets/topo/root.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/root.png -------------------------------------------------------------------------------- /src/assets/topo/router.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/router.png -------------------------------------------------------------------------------- /src/assets/topo/sercurity.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/sercurity.png -------------------------------------------------------------------------------- /src/assets/topo/storage.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/storage.png -------------------------------------------------------------------------------- /src/assets/topo/subnet.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/subnet.png -------------------------------------------------------------------------------- /src/assets/topo/virtualip.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/virtualip.png -------------------------------------------------------------------------------- /src/assets/topo/volume.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/volume.png -------------------------------------------------------------------------------- /src/assets/topo/webserver.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/assets/topo/webserver.png -------------------------------------------------------------------------------- /src/components/vFooter.vue: -------------------------------------------------------------------------------- 1 | 7 | 21 | 22 | 32 | -------------------------------------------------------------------------------- /src/components/vHeader.vue: -------------------------------------------------------------------------------- 1 | 7 | 8 | 13 | 14 | 25 | 26 | 29 | -------------------------------------------------------------------------------- /src/components/vTopo/components/vShapebar.vue: -------------------------------------------------------------------------------- 1 | 7 | 24 | 25 | 57 | -------------------------------------------------------------------------------- /src/components/vTopo/components/vTopoAttrPanel.vue: -------------------------------------------------------------------------------- 1 | 7 | 46 | 47 | 80 | 81 | 84 | -------------------------------------------------------------------------------- /src/components/vTopo/vTopo.vue: -------------------------------------------------------------------------------- 1 | 7 | 227 | 1221 | 1292 | 1295 | -------------------------------------------------------------------------------- /src/config/connectorRules.js: -------------------------------------------------------------------------------- 1 | let connectorRules = [ 2 | { 3 | type:'Service', 4 | canBeContainedType:['Service'], 5 | canLinkToType:['Service'] 6 | }, 7 | { 8 | type:'Root', 9 | canBeContainedType:['Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'], 10 | canLinkToType:['Root','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'] 11 | }, 12 | { 13 | type:'Compute', 14 | canBeContainedType:[], 15 | canLinkToType:['Compute','Root','Container','Volume','FileSystem','ObjectStorage','Network','VirtualIP','SercurityGroup','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','ApplicationModule','CloudifyManager'] 16 | }, 17 | { 18 | type:'Container', 19 | canBeContainedType:['Container','Compute'], 20 | canLinkToType:['Container','Root','Volume','FileSystem','ObjectStorage','Network','VirtualIP','SercurityGroup','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','ApplicationModule','CloudifyManager'] 21 | }, 22 | { 23 | type:'Volume', 24 | canBeContainedType:['Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'], 25 | canLinkToType:['Volume','Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'] 26 | }, 27 | { 28 | type:'FileSystem', 29 | canBeContainedType:['FileSystem','Root','Compute','Container','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'], 30 | canLinkToType:['FileSystem','Root','Compute','Container','Volume','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','ApplicationModule','CloudifyManager'] 31 | }, 32 | { 33 | type:'ObjectStorage', 34 | canBeContainedType:['ObjectStorage','Root','Compute','Container','FileSystem','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'], 35 | canLinkToType:['ObjectStorage','Root','Compute','Container','Volume','FileSystem','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','ApplicationModule','CloudifyManager'] 36 | }, 37 | { 38 | type:'Network', 39 | canBeContainedType:[], 40 | canLinkToType:[] 41 | }, 42 | { 43 | type:'Router', 44 | canBeContainedType:[], 45 | canLinkToType:[] 46 | }, 47 | { 48 | type:'LoadBalancer', 49 | canBeContainedType:[], 50 | canLinkToType:[] 51 | }, 52 | { 53 | type:'VirtualIP', 54 | canBeContainedType:[], 55 | canLinkToType:[] 56 | }, 57 | { 58 | type:'SercurityGroup', 59 | canBeContainedType:[], 60 | canLinkToType:[] 61 | }, 62 | { 63 | type:'DBMS', 64 | canBeContainedType:['DBMS','Root','Compute','Container','FileSystem','ObjectStorage','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'], 65 | canLinkToType:['DBMS','Root','Compute','Container','Volume','FileSystem','ObjectStorage','DataBase','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'] 66 | }, 67 | { 68 | type:'DataBase', 69 | canBeContainedType:['DataBase','Root','Compute','Container','FileSystem','ObjectStorage','DBMS','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'], 70 | canLinkToType:['DataBase','Root','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','WebServer','ApplicationServer','MessageBusServer','CloudifyManager'] 71 | }, 72 | { 73 | type:'WebServer', 74 | canBeContainedType:['WebServer','Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','ApplicationServer','CloudifyManager'], 75 | canLinkToType:['WebServer','Root','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','DataBase','ApplicationServer','CloudifyManager'] 76 | }, 77 | { 78 | type:'ApplicationServer', 79 | canBeContainedType:['ApplicationServer','Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','CloudifyManager'], 80 | canLinkToType:['ApplicationServer','Root','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','CloudifyManager'] 81 | }, 82 | { 83 | type:'MessageBusServer', 84 | canBeContainedType:['ApplicationServer','Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','CloudifyManager'], 85 | canLinkToType:['ApplicationServer','Root','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','CloudifyManager'] 86 | }, 87 | { 88 | type:'ApplicationModule', 89 | canBeContainedType:['Root','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','CloudifyManager'], 90 | canLinkToType:['ApplicationModule','Root','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','CloudifyManager'] 91 | }, 92 | { 93 | type:'CloudifyManager', 94 | canBeContainedType:['CloudifyManager','Compute','Container','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer'], 95 | canLinkToType:['CloudifyManager','Compute','Container','Volume','FileSystem','ObjectStorage','DBMS','DataBase','WebServer','ApplicationServer','MessageBusServer','ApplicationModule'] 96 | } 97 | ] 98 | export default connectorRules -------------------------------------------------------------------------------- /src/data/toolbarNodeData.js: -------------------------------------------------------------------------------- 1 | const toolbarNodeData = [ 2 | {type:'Service',icon:require('../assets/topo/root.png'),width:140,height:80,num:1,classType:'T1'}, 3 | {type:'Router',icon:require('../assets/topo/router.png'),width:50,height:50,num:1,classType:'T2'}, 4 | {type:'Container',icon:require('../assets/topo/container.png'),width:140,height:80,num:1,classType:'T1'}, 5 | {type:'Pod',icon:require('../assets/topo/container.png'),width:140,height:80,num:1,classType:'T1'}, 6 | {type:'LoadBalancer',icon:require('../assets/topo/router.png'),width:115,height:60,num:1,classType:'T1'}, 7 | {type:'Compute',icon:require('../assets/topo/compute.png'),width:150,height:100,num:1,classType:'T1'}, 8 | {type:'Volume',icon:require('../assets/topo/volume.png'),width:50,height:50,num:1,classType:'T2'}, 9 | {type:'FileSystem',icon:require('../assets/topo/filesystem.png'),width:150,height:100,num:1,classType:'T1'}, 10 | {type:'ObjectStorage',icon:require('../assets/topo/storage.png'),width:150,height:100,num:1,classType:'T1'}, 11 | {type:'Network',icon:require('../assets/topo/network.png'),width:130,height:80,num:1,classType:'T1'}, 12 | {type:'Subnet',icon:require('../assets/topo/subnet.png'),width:150,height:100,num:1,classType:'T1'}, 13 | {type:'Port',icon:require('../assets/topo/port.png'),width:150,height:100,num:1,classType:'T1'}, 14 | {type:'LoadBalancer',icon:require('../assets/topo/router.png'),width:150,height:100,num:1,classType:'T1'}, 15 | {type:'VirtualIP',icon:require('../assets/topo/virtualip.png'),width:50,height:50,num:1,classType:'T2'}, 16 | {type:'SercurityGroup',icon:require('../assets/topo/sercurity.png'),width:50,height:50,num:1,classType:'T2'}, 17 | {type:'DBMS',icon:require('../assets/topo/dbms.png'),width:150,height:100,num:1,classType:'T1'}, 18 | {type:'DataBase',icon:require('../assets/topo/database.png'),width:150,height:100,num:1,classType:'T1'}, 19 | {type:'WebServer',icon:require('../assets/topo/webserver.png'),width:150,height:100,num:1,classType:'T1'}, 20 | {type:'ApplicationServer',icon:require('../assets/topo/application.png'),width:150,height:100,num:1,classType:'T1'}, 21 | {type:'MessageBusServer',icon:require('../assets/topo/message.png'),width:150,height:100,num:1,classType:'T1'}, 22 | {type:'ApplicationModule',icon:require('../assets/topo/application.png'),width:50,height:50,num:1,classType:'T2'}, 23 | {type:'CloudifyManager',icon:require('../assets/topo/cloud.png'),width:150,height:100,num:1,classType:'T1'} 24 | ] 25 | export default toolbarNodeData -------------------------------------------------------------------------------- /src/data/topoData1.js: -------------------------------------------------------------------------------- 1 | let topoJson = { 2 | "nodes": [{ 3 | "name": "ApplicationModule_1", 4 | "type": "ApplicationModule", 5 | "id": "qehkqs6mazk", 6 | "x": 460, 7 | "y": 340, 8 | "icon": "", 9 | "width": 50, 10 | "height": 50, 11 | "initW": 50, 12 | "initH": 50, 13 | "classType": "T2", 14 | "isLeftConnectShow": true, 15 | "isRightConnectShow": false, 16 | "containNodes": [], 17 | "attrs": [], 18 | "isSelect": false 19 | }, { 20 | "name": "Container_1", 21 | "type": "Container", 22 | "id": "4trq8hpmv7g0", 23 | "x": 100, 24 | "y": 160, 25 | "icon": "", 26 | "width": 140, 27 | "height": 80, 28 | "initW": 140, 29 | "initH": 80, 30 | "classType": "T1", 31 | "isLeftConnectShow": false, 32 | "isRightConnectShow": true, 33 | "containNodes": [], 34 | "attrs": [], 35 | "isSelect": false 36 | }, { 37 | "name": "ObjectStorage_1", 38 | "type": "ObjectStorage", 39 | "id": "168oyhimq4m8", 40 | "x": 500, 41 | "y": 80, 42 | "icon": "", 43 | "width": 150, 44 | "height": 100, 45 | "initW": 150, 46 | "initH": 100, 47 | "classType": "T1", 48 | "isLeftConnectShow": true, 49 | "isRightConnectShow": false, 50 | "containNodes": [], 51 | "attrs": [], 52 | "isSelect": false 53 | }, { 54 | "name": "Container_2", 55 | "type": "Container", 56 | "id": "49dz3mbgaas0", 57 | "x": 80, 58 | "y": 360, 59 | "icon": "", 60 | "width": 140, 61 | "height": 80, 62 | "initW": 140, 63 | "initH": 80, 64 | "classType": "T1", 65 | "isLeftConnectShow": true, 66 | "isRightConnectShow": true, 67 | "containNodes": [], 68 | "attrs": [], 69 | "isSelect": false 70 | }, { 71 | "name": "Container_3", 72 | "type": "Container", 73 | "id": "6gle6rqk2m00", 74 | "x": 100, 75 | "y": 520, 76 | "icon": "", 77 | "width": 346, 78 | "height": 120, 79 | "initW": 140, 80 | "initH": 80, 81 | "classType": "T1", 82 | "isLeftConnectShow": false, 83 | "isRightConnectShow": false, 84 | "containNodes": ["5jkxlx7tpy00", "4aoxxa2ejva0"], 85 | "attrs": [], 86 | "isSelect": false 87 | }, { 88 | "name": "Pod_1", 89 | "type": "Pod", 90 | "id": "5jkxlx7tpy00", 91 | "x": 122, 92 | "y": 550, 93 | "icon": "", 94 | "width": 140, 95 | "height": 80, 96 | "initW": 140, 97 | "initH": 80, 98 | "classType": "T1", 99 | "isLeftConnectShow": false, 100 | "isRightConnectShow": false, 101 | "containNodes": [], 102 | "attrs": [], 103 | "isSelect": false 104 | }, { 105 | "name": "Pod_2", 106 | "type": "Pod", 107 | "id": "4aoxxa2ejva0", 108 | "x": 284, 109 | "y": 550, 110 | "icon": "", 111 | "width": 140, 112 | "height": 80, 113 | "initW": 140, 114 | "initH": 80, 115 | "classType": "T1", 116 | "isLeftConnectShow": false, 117 | "isRightConnectShow": false, 118 | "containNodes": [], 119 | "attrs": [], 120 | "isSelect": false 121 | }], 122 | "connectors": [{ 123 | "id": "2owjakk3cj0", 124 | "type": "Line", 125 | "strokeW": 10, 126 | "color": "#9AC0EE", 127 | "targetNode": { 128 | "x": 460, 129 | "y": 340, 130 | "id": "qehkqs6mazk", 131 | "width": 50, 132 | "height": 50 133 | }, 134 | "sourceNode": { 135 | "x": 100, 136 | "y": 160, 137 | "id": "4trq8hpmv7g0", 138 | "width": 140, 139 | "height": 80 140 | }, 141 | "isSelect": false 142 | }, { 143 | "id": "ucbdfy4baf", 144 | "type": "Line", 145 | "strokeW": 6, 146 | "color": "#ED565B", 147 | "targetNode": { 148 | "x": 500, 149 | "y": 80, 150 | "id": "168oyhimq4m8", 151 | "width": 150, 152 | "height": 100 153 | }, 154 | "sourceNode": { 155 | "x": 100, 156 | "y": 160, 157 | "id": "4trq8hpmv7g0", 158 | "width": 140, 159 | "height": 80 160 | }, 161 | "isSelect": false 162 | }, { 163 | "id": "w0kqd4vs0l", 164 | "type": "Line", 165 | "strokeW": 12, 166 | "color": "#3CD37B", 167 | "targetNode": { 168 | "x": 80, 169 | "y": 360, 170 | "id": "49dz3mbgaas0", 171 | "width": 140, 172 | "height": 80 173 | }, 174 | "sourceNode": { 175 | "x": 80, 176 | "y": 360, 177 | "id": "49dz3mbgaas0", 178 | "width": 140, 179 | "height": 80 180 | }, 181 | "isSelect": false 182 | }, { 183 | "id": "1tvycjqq70u", 184 | "type": "Contain", 185 | "sourceNode": { 186 | "id": "5jkxlx7tpy00", 187 | "width": 140, 188 | "height": 80, 189 | "x": 122, 190 | "y": 550 191 | }, 192 | "targetNode": { 193 | "id": "6gle6rqk2m00", 194 | "width": 346, 195 | "height": 120, 196 | "x": 100, 197 | "y": 520 198 | }, 199 | "isSelect": false 200 | }, { 201 | "id": "1ltadqyi6pj", 202 | "type": "Contain", 203 | "sourceNode": { 204 | "id": "4aoxxa2ejva0", 205 | "width": 140, 206 | "height": 80, 207 | "x": 284, 208 | "y": 550 209 | }, 210 | "targetNode": { 211 | "id": "6gle6rqk2m00", 212 | "width": 346, 213 | "height": 120, 214 | "x": 100, 215 | "y": 520 216 | }, 217 | "isSelect": false 218 | }] 219 | } 220 | export default topoJson 221 | -------------------------------------------------------------------------------- /src/data/topoData2.js: -------------------------------------------------------------------------------- 1 | let topoJson = { 2 | "nodes": [{ 3 | "name": "LoadBalancer_1", 4 | "type": "LoadBalancer", 5 | "id": "1hokg0mgnmww", 6 | "x": 840, 7 | "y": 180, 8 | "icon": "", 9 | "width": 115, 10 | "height": 60, 11 | "initW": 115, 12 | "initH": 60, 13 | "classType": "T1", 14 | "isLeftConnectShow": true, 15 | "isRightConnectShow": false, 16 | "containNodes": [], 17 | "attrs": [], 18 | "isSelect": false 19 | }, { 20 | "name": "Router_1", 21 | "type": "Router", 22 | "id": "dbfp8krc8ow", 23 | "x": 680, 24 | "y": 220, 25 | "icon": "", 26 | "width": 50, 27 | "height": 50, 28 | "initW": 50, 29 | "initH": 50, 30 | "classType": "T2", 31 | "isLeftConnectShow": true, 32 | "isRightConnectShow": true, 33 | "containNodes": [], 34 | "attrs": [], 35 | "isSelect": false 36 | }, { 37 | "name": "Service_1", 38 | "type": "Service", 39 | "id": "1orsbg8m30zk", 40 | "x": 60, 41 | "y": 40, 42 | "icon": "", 43 | "width": 552, 44 | "height": 160, 45 | "initW": 140, 46 | "initH": 80, 47 | "classType": "T1", 48 | "isLeftConnectShow": false, 49 | "isRightConnectShow": true, 50 | "containNodes": ["jdturwsp9xc", "6isr7j0hiec0"], 51 | "attrs": [], 52 | "isSelect": false 53 | }, { 54 | "name": "Pod_1", 55 | "type": "Pod", 56 | "id": "jdturwsp9xc", 57 | "x": 82, 58 | "y": 70, 59 | "icon": "", 60 | "width": 346, 61 | "height": 120, 62 | "initW": 140, 63 | "initH": 80, 64 | "classType": "T1", 65 | "isLeftConnectShow": false, 66 | "isRightConnectShow": false, 67 | "containNodes": [66, 88], 68 | "attrs": [], 69 | "isSelect": false 70 | }, { 71 | "x": 266, 72 | "y": 100, 73 | "width": 140, 74 | "height": 80, 75 | "id": 88, 76 | "isLeftConnectShow": false, 77 | "isRightConnectShow": false, 78 | "name": "Container_c", 79 | "isSelect": false, 80 | "initW": 140, 81 | "initH": 80, 82 | "icon": "", 83 | "classType": "T1", 84 | "containNodes": [], 85 | "attrs": [], 86 | "type": "Container" 87 | }, { 88 | "name": "Pod_2", 89 | "type": "Pod", 90 | "id": "6isr7j0hiec0", 91 | "x": 450, 92 | "y": 70, 93 | "icon": "", 94 | "width": 140, 95 | "height": 80, 96 | "initW": 140, 97 | "initH": 80, 98 | "classType": "T1", 99 | "isLeftConnectShow": false, 100 | "isRightConnectShow": false, 101 | "containNodes": [], 102 | "attrs": [], 103 | "isSelect": false 104 | }, { 105 | "x": 104, 106 | "y": 100, 107 | "width": 140, 108 | "height": 80, 109 | "id": 66, 110 | "isLeftConnectShow": false, 111 | "isRightConnectShow": false, 112 | "name": "Container_a", 113 | "isSelect": false, 114 | "initW": 140, 115 | "initH": 80, 116 | "icon": "", 117 | "classType": "T1", 118 | "containNodes": [], 119 | "type": "Container", 120 | "attrs": [{ 121 | "type": "input", 122 | "name": "portId", 123 | "value": "2222141", 124 | "placeholder": "请输入portId", 125 | "rules": [{ 126 | "required": true, 127 | "message": "请输入活动名称", 128 | "trigger": "blur" 129 | }], 130 | "disabled": true 131 | }, { 132 | "type": "select", 133 | "name": "server", 134 | "value": "", 135 | "placeholder": "请选择服务器", 136 | "options": [{ 137 | "label": "上海服务器", 138 | "value": "shagnhai" 139 | }, { 140 | "label": "北京服务器", 141 | "value": "beijing" 142 | }], 143 | "disabled": false 144 | }, { 145 | "type": "checkbox", 146 | "name": "数据库类型", 147 | "value": [], 148 | "options": [{ 149 | "label": "SQL server" 150 | }, { 151 | "label": "Access" 152 | }, { 153 | "label": "mySQL" 154 | }, { 155 | "label": "Oracle" 156 | }], 157 | "disabled": false 158 | }, { 159 | "type": "textarea", 160 | "name": "数据库", 161 | "value": "", 162 | "rules": [], 163 | "disabled": false 164 | }, { 165 | "type": "radio", 166 | "name": "数据类型", 167 | "value": "", 168 | "options": [{ 169 | "label": "sql" 170 | }, { 171 | "label": "oracle" 172 | }], 173 | "disabled": true 174 | }, { 175 | "type": "keyVal", 176 | "name": "service", 177 | "value": "dbms" 178 | }, { 179 | "type": "keyVal", 180 | "name": "sql", 181 | "value": "sq" 182 | }] 183 | }], 184 | "connectors": [{ 185 | "type": "Contain", 186 | "sourceNode": { 187 | "id": 66, 188 | "width": 140, 189 | "height": 80, 190 | "x": 104, 191 | "y": 100 192 | }, 193 | "targetNode": { 194 | "id": "jdturwsp9xc", 195 | "width": 346, 196 | "height": 120, 197 | "x": 82, 198 | "y": 70 199 | }, 200 | "isSelect": false 201 | }, { 202 | "type": "Contain", 203 | "sourceNode": { 204 | "id": "jdturwsp9xc", 205 | "width": 346, 206 | "height": 120, 207 | "x": 82, 208 | "y": 70 209 | }, 210 | "targetNode": { 211 | "id": "1orsbg8m30zk", 212 | "width": 552, 213 | "height": 160, 214 | "x": 60, 215 | "y": 40 216 | }, 217 | "isSelect": false 218 | }, { 219 | "type": "Line", 220 | "targetNode": { 221 | "x": 840, 222 | "y": 180, 223 | "id": "1hokg0mgnmww", 224 | "width": 115, 225 | "height": 60 226 | }, 227 | "sourceNode": { 228 | "x": 680, 229 | "y": 220, 230 | "id": "dbfp8krc8ow", 231 | "width": 50, 232 | "height": 50 233 | }, 234 | "isSelect": false 235 | }, { 236 | "type": "Line", 237 | "targetNode": { 238 | "x": 680, 239 | "y": 220, 240 | "id": "dbfp8krc8ow", 241 | "width": 50, 242 | "height": 50 243 | }, 244 | "sourceNode": { 245 | "x": 60, 246 | "y": 40, 247 | "id": "1orsbg8m30zk", 248 | "width": 552, 249 | "height": 160 250 | }, 251 | "isSelect": false 252 | }, { 253 | "type": "Contain", 254 | "sourceNode": { 255 | "id": "6isr7j0hiec0", 256 | "width": 140, 257 | "height": 80, 258 | "x": 450, 259 | "y": 70 260 | }, 261 | "targetNode": { 262 | "id": "1orsbg8m30zk", 263 | "width": 552, 264 | "height": 160, 265 | "x": 60, 266 | "y": 40 267 | }, 268 | "isSelect": false 269 | }, { 270 | "type": "Contain", 271 | "sourceNode": { 272 | "id": 88, 273 | "width": 140, 274 | "height": 80, 275 | "x": 266, 276 | "y": 100 277 | }, 278 | "targetNode": { 279 | "id": "jdturwsp9xc", 280 | "width": 346, 281 | "height": 120, 282 | "x": 82, 283 | "y": 70 284 | }, 285 | "isSelect": false 286 | }] 287 | } 288 | 289 | export default topoJson 290 | -------------------------------------------------------------------------------- /src/data/topoJson.js: -------------------------------------------------------------------------------- 1 | // let topoJson = { 2 | // "nodes": [{ 3 | // "name": "LoadBalancer_1", 4 | // "type": "LoadBalancer", 5 | // "id": "1hokg0mgnmww", 6 | // "x": 840, 7 | // "y": 180, 8 | // "icon": "", 9 | // "width": 115, 10 | // "height": 60, 11 | // "initW": 115, 12 | // "initH": 60, 13 | // "classType": "T1", 14 | // "isLeftConnectShow": true, 15 | // "isRightConnectShow": false, 16 | // "containNodes": [], 17 | // "attrs": [], 18 | // "isSelect": false 19 | // }, { 20 | // "name": "Router_1", 21 | // "type": "Router", 22 | // "id": "dbfp8krc8ow", 23 | // "x": 680, 24 | // "y": 220, 25 | // "icon": "", 26 | // "width": 50, 27 | // "height": 50, 28 | // "initW": 50, 29 | // "initH": 50, 30 | // "classType": "T2", 31 | // "isLeftConnectShow": true, 32 | // "isRightConnectShow": true, 33 | // "containNodes": [], 34 | // "attrs": [], 35 | // "isSelect": false 36 | // }, { 37 | // "name": "Service_1", 38 | // "type": "Service", 39 | // "id": "1orsbg8m30zk", 40 | // "x": 60, 41 | // "y": 40, 42 | // "icon": "", 43 | // "width": 552, 44 | // "height": 160, 45 | // "initW": 140, 46 | // "initH": 80, 47 | // "classType": "T1", 48 | // "isLeftConnectShow": false, 49 | // "isRightConnectShow": true, 50 | // "containNodes": ["jdturwsp9xc", "6isr7j0hiec0"], 51 | // "attrs": [], 52 | // "isSelect": false 53 | // }, { 54 | // "name": "Pod_1", 55 | // "type": "Pod", 56 | // "id": "jdturwsp9xc", 57 | // "x": 82, 58 | // "y": 70, 59 | // "icon": "", 60 | // "width": 346, 61 | // "height": 120, 62 | // "initW": 140, 63 | // "initH": 80, 64 | // "classType": "T1", 65 | // "isLeftConnectShow": false, 66 | // "isRightConnectShow": false, 67 | // "containNodes": [66, 88], 68 | // "attrs": [], 69 | // "isSelect": false 70 | // }, { 71 | // "x": 266, 72 | // "y": 100, 73 | // "width": 140, 74 | // "height": 80, 75 | // "id": 88, 76 | // "isLeftConnectShow": false, 77 | // "isRightConnectShow": false, 78 | // "name": "Container_c", 79 | // "isSelect": false, 80 | // "initW": 140, 81 | // "initH": 80, 82 | // "icon": "", 83 | // "classType": "T1", 84 | // "containNodes": [], 85 | // "attrs": [], 86 | // "type": "Container" 87 | // }, { 88 | // "name": "Pod_2", 89 | // "type": "Pod", 90 | // "id": "6isr7j0hiec0", 91 | // "x": 450, 92 | // "y": 70, 93 | // "icon": "", 94 | // "width": 140, 95 | // "height": 80, 96 | // "initW": 140, 97 | // "initH": 80, 98 | // "classType": "T1", 99 | // "isLeftConnectShow": false, 100 | // "isRightConnectShow": false, 101 | // "containNodes": [], 102 | // "attrs": [], 103 | // "isSelect": false 104 | // }, { 105 | // "x": 104, 106 | // "y": 100, 107 | // "width": 140, 108 | // "height": 80, 109 | // "id": 66, 110 | // "isLeftConnectShow": false, 111 | // "isRightConnectShow": false, 112 | // "name": "Container_a", 113 | // "isSelect": false, 114 | // "initW": 140, 115 | // "initH": 80, 116 | // "icon": "", 117 | // "classType": "T1", 118 | // "containNodes": [], 119 | // "type": "Container", 120 | // "attrs": [{ 121 | // "type": "input", 122 | // "name": "portId", 123 | // "value": "2222141", 124 | // "placeholder": "请输入portId", 125 | // "rules": [{ 126 | // "required": true, 127 | // "message": "请输入活动名称", 128 | // "trigger": "blur" 129 | // }], 130 | // "disabled": true 131 | // }, { 132 | // "type": "select", 133 | // "name": "server", 134 | // "value": "", 135 | // "placeholder": "请选择服务器", 136 | // "options": [{ 137 | // "label": "上海服务器", 138 | // "value": "shagnhai" 139 | // }, { 140 | // "label": "北京服务器", 141 | // "value": "beijing" 142 | // }], 143 | // "disabled": false 144 | // }, { 145 | // "type": "checkbox", 146 | // "name": "数据库类型", 147 | // "value": [], 148 | // "options": [{ 149 | // "label": "SQL server" 150 | // }, { 151 | // "label": "Access" 152 | // }, { 153 | // "label": "mySQL" 154 | // }, { 155 | // "label": "Oracle" 156 | // }], 157 | // "disabled": false 158 | // }, { 159 | // "type": "textarea", 160 | // "name": "数据库", 161 | // "value": "", 162 | // "rules": [], 163 | // "disabled": false 164 | // }, { 165 | // "type": "radio", 166 | // "name": "数据类型", 167 | // "value": "", 168 | // "options": [{ 169 | // "label": "sql" 170 | // }, { 171 | // "label": "oracle" 172 | // }], 173 | // "disabled": true 174 | // }, { 175 | // "type": "keyVal", 176 | // "name": "service", 177 | // "value": "dbms" 178 | // }, { 179 | // "type": "keyVal", 180 | // "name": "sql", 181 | // "value": "sq" 182 | // }] 183 | // }], 184 | // "connectors": [{ 185 | // "type": "Contain", 186 | // "sourceNode": { 187 | // "id": 66, 188 | // "width": 140, 189 | // "height": 80, 190 | // "x": 104, 191 | // "y": 100 192 | // }, 193 | // "targetNode": { 194 | // "id": "jdturwsp9xc", 195 | // "width": 346, 196 | // "height": 120, 197 | // "x": 82, 198 | // "y": 70 199 | // }, 200 | // "isSelect": false 201 | // }, { 202 | // "type": "Contain", 203 | // "sourceNode": { 204 | // "id": "jdturwsp9xc", 205 | // "width": 346, 206 | // "height": 120, 207 | // "x": 82, 208 | // "y": 70 209 | // }, 210 | // "targetNode": { 211 | // "id": "1orsbg8m30zk", 212 | // "width": 552, 213 | // "height": 160, 214 | // "x": 60, 215 | // "y": 40 216 | // }, 217 | // "isSelect": false 218 | // }, { 219 | // "type": "Line", 220 | // "targetNode": { 221 | // "x": 840, 222 | // "y": 180, 223 | // "id": "1hokg0mgnmww", 224 | // "width": 115, 225 | // "height": 60 226 | // }, 227 | // "sourceNode": { 228 | // "x": 680, 229 | // "y": 220, 230 | // "id": "dbfp8krc8ow", 231 | // "width": 50, 232 | // "height": 50 233 | // }, 234 | // "isSelect": false 235 | // }, { 236 | // "type": "Line", 237 | // "targetNode": { 238 | // "x": 680, 239 | // "y": 220, 240 | // "id": "dbfp8krc8ow", 241 | // "width": 50, 242 | // "height": 50 243 | // }, 244 | // "sourceNode": { 245 | // "x": 60, 246 | // "y": 40, 247 | // "id": "1orsbg8m30zk", 248 | // "width": 552, 249 | // "height": 160 250 | // }, 251 | // "isSelect": false 252 | // }, { 253 | // "type": "Contain", 254 | // "sourceNode": { 255 | // "id": "6isr7j0hiec0", 256 | // "width": 140, 257 | // "height": 80, 258 | // "x": 450, 259 | // "y": 70 260 | // }, 261 | // "targetNode": { 262 | // "id": "1orsbg8m30zk", 263 | // "width": 552, 264 | // "height": 160, 265 | // "x": 60, 266 | // "y": 40 267 | // }, 268 | // "isSelect": false 269 | // }, { 270 | // "type": "Contain", 271 | // "sourceNode": { 272 | // "id": 88, 273 | // "width": 140, 274 | // "height": 80, 275 | // "x": 266, 276 | // "y": 100 277 | // }, 278 | // "targetNode": { 279 | // "id": "jdturwsp9xc", 280 | // "width": 346, 281 | // "height": 120, 282 | // "x": 82, 283 | // "y": 70 284 | // }, 285 | // "isSelect": false 286 | // }] 287 | // } 288 | 289 | let topoJson = { 290 | "nodes": [{ 291 | "name": "Router_1", 292 | "type": "Router", 293 | "id": "3e9gd7rqais0", 294 | "x": 460, 295 | "y": 180, 296 | "icon": "", 297 | "width": 50, 298 | "height": 50, 299 | "initW": 50, 300 | "initH": 50, 301 | "classType": "T2", 302 | "isLeftConnectShow": true, 303 | "isRightConnectShow": false, 304 | "containNodes": [], 305 | "attrs": [], 306 | "isSelect": false 307 | }, { 308 | "name": "Compute_2", 309 | "type": "Compute", 310 | "id": "5xcsqlke4ck0", 311 | "x": 140, 312 | "y": 260, 313 | "icon": "", 314 | "width": 150, 315 | "height": 100, 316 | "initW": 150, 317 | "initH": 100, 318 | "classType": "T1", 319 | "isLeftConnectShow": true, 320 | "isRightConnectShow": true, 321 | "containNodes": [], 322 | "attrs": [], 323 | "isSelect": false 324 | }, { 325 | "name": "Compute_1", 326 | "type": "Compute", 327 | "id": "4he2wkhejti0", 328 | "x": 100, 329 | "y": 60, 330 | "icon": "", 331 | "width": 150, 332 | "height": 100, 333 | "initW": 150, 334 | "initH": 100, 335 | "classType": "T1", 336 | "isLeftConnectShow": false, 337 | "isRightConnectShow": true, 338 | "containNodes": [], 339 | "attrs": [], 340 | "isSelect": false 341 | }], 342 | "connectors": [{ 343 | "type": "Line", 344 | "strokeW":10, 345 | "color":"#DE4516", 346 | "targetNode": { 347 | "x": 460, 348 | "y": 180, 349 | "id": "3e9gd7rqais0", 350 | "width": 50, 351 | "height": 50 352 | }, 353 | "sourceNode": { 354 | "x": 100, 355 | "y": 60, 356 | "id": "4he2wkhejti0", 357 | "width": 150, 358 | "height": 100 359 | }, 360 | "isSelect": false 361 | }, { 362 | "type": "Line", 363 | "strokeW":5, 364 | "color":"#3399ff", 365 | "targetNode": { 366 | "x": 140, 367 | "y": 260, 368 | "id": "5xcsqlke4ck0", 369 | "width": 150, 370 | "height": 100 371 | }, 372 | "sourceNode": { 373 | "x": 140, 374 | "y": 260, 375 | "id": "5xcsqlke4ck0", 376 | "width": 150, 377 | "height": 100 378 | }, 379 | "isSelect": false 380 | }] 381 | } 382 | export default topoJson 383 | -------------------------------------------------------------------------------- /src/less/index.css: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/src/less/index.css -------------------------------------------------------------------------------- /src/less/index.less: -------------------------------------------------------------------------------- 1 | @theme-color:#f3f3f3; 2 | @theme-font-color:#525252; -------------------------------------------------------------------------------- /src/main.js: -------------------------------------------------------------------------------- 1 | // The Vue build version to load with the `import` command 2 | // (runtime-only or standalone) has been set in webpack.base.conf with an alias. 3 | import Vue from 'vue' 4 | import App from './App' 5 | import router from './router' 6 | import ElementUI from 'element-ui' 7 | import $ from 'jquery' 8 | import nlComponentLib from './nlComponentLib' 9 | 10 | import 'element-ui/lib/theme-chalk/index.css' 11 | import 'element-ui/lib/theme-chalk/display.css' 12 | import 'font-awesome/css/font-awesome.css' 13 | import '../static/css/reset.css' //rest样式引入 14 | 15 | 16 | Vue.config.productionTip = false 17 | 18 | Vue.use(ElementUI) 19 | Vue.use(nlComponentLib) 20 | 21 | /* eslint-disable no-new */ 22 | new Vue({ 23 | el: '#app', 24 | router, 25 | template: '', 26 | components: { App } 27 | }) 28 | -------------------------------------------------------------------------------- /src/nlComponentLib/components/nlButton.vue: -------------------------------------------------------------------------------- 1 | /* 2 | * @Author: caojing 3 | * @Date: 2018-10-23 11:36:54 4 | * @LastEditors: caojing 5 | * @LastEditTime: 2018-11-13 16:18:52 6 | */ 7 | 12 | 13 | 45 | 46 | 65 | -------------------------------------------------------------------------------- /src/nlComponentLib/components/nlPageTitle.vue: -------------------------------------------------------------------------------- 1 | /* 2 | * @Author: caojing 3 | * @Date: 2018-10-23 11:26:05 4 | * @LastEditors: caojing 5 | * @LastEditTime: 2018-11-13 16:19:15 6 | */ 7 | 15 | 16 | 33 | 38 | -------------------------------------------------------------------------------- /src/nlComponentLib/index.js: -------------------------------------------------------------------------------- 1 | /* 2 | * @Author: caojing 3 | * @Date: 2018-10-23 11:25:17 4 | * @Last Modified by: caojing 5 | * @Last Modified time: 2018-10-23 11:38:20 6 | */ 7 | import Vue from 'vue' 8 | import nlPageTitle from './components/nlPageTitle'; 9 | import nlButton from './components/nlButton'; 10 | export default (Vue)=>{ 11 | Vue.component("nlPageTitle",nlPageTitle); 12 | Vue.component("nlButton",nlButton) 13 | } -------------------------------------------------------------------------------- /src/router/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import Router from 'vue-router' 3 | // import HelloWorld from '@/components/HelloWorld' 4 | 5 | Vue.use(Router) 6 | 7 | export default new Router({ 8 | routes: [ 9 | { 10 | path: '/', 11 | // name: 'Hello', 12 | // component: HelloWorld 13 | } 14 | ] 15 | }) 16 | -------------------------------------------------------------------------------- /static/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/1StepEngineer/vue-topo/08a7e478717e15e4e5236f1ec4bbfbafaaba56a9/static/.gitkeep -------------------------------------------------------------------------------- /static/css/reset.css: -------------------------------------------------------------------------------- 1 | html, 2 | body { 3 | height: 100%; 4 | width: 100%; 5 | } 6 | 7 | body { 8 | font-family: arial, helvetica, sans-serif; 9 | font-size: 13px; 10 | font-size-adjust: none; 11 | font-stretch: normal; 12 | font-style: normal; 13 | font-variant: normal; 14 | font-weight: normal; 15 | line-height: 1.4; 16 | text-align: center; 17 | } 18 | 19 | body, 20 | ul, 21 | ol, 22 | dl, 23 | dd, 24 | h1, 25 | h2, 26 | h3, 27 | h4, 28 | h5, 29 | h6, 30 | p, 31 | form, 32 | fieldset, 33 | legend, 34 | input, 35 | textarea, 36 | select, 37 | button, 38 | th, 39 | td { 40 | margin: 0; 41 | padding: 0; 42 | } 43 | 44 | h1, 45 | h2, 46 | h3, 47 | h4, 48 | h5, 49 | h6 { 50 | font-size: 100%; 51 | font-weight: normal; 52 | } 53 | 54 | table { 55 | font-size: inherit; 56 | } 57 | 58 | input, 59 | select { 60 | font-family: arial, helvetica, clean, sans-serif; 61 | font-size: 100%; 62 | font-size-adjust: none; 63 | font-stretch: normal; 64 | font-style: normal; 65 | font-variant: normal; 66 | font-weight: normal; 67 | line-height: normal; 68 | } 69 | 70 | button { 71 | overflow: visible; 72 | } 73 | 74 | th, 75 | em, 76 | strong, 77 | b, 78 | address, 79 | cite { 80 | font-style: normal; 81 | font-weight: normal; 82 | } 83 | 84 | li { 85 | list-style-image: none; 86 | list-style-position: outside; 87 | list-style-type: none; 88 | } 89 | 90 | img, 91 | fieldset { 92 | border: 0 none; 93 | } 94 | 95 | ins { 96 | text-decoration: none; 97 | } 98 | 99 | .clearfix { 100 | zoom: 1; 101 | } 102 | 103 | .clearfix:after { 104 | content: ''; 105 | display: block; 106 | clear: both; 107 | } 108 | -------------------------------------------------------------------------------- /topoJson.json: -------------------------------------------------------------------------------- 1 | { 2 | "nodes": [{ 3 | "name": "New_Cloud_2", 4 | "type": "T1", 5 | "id": "34ls07hv9zw0", 6 | "x": 620, 7 | "y": 40, 8 | "icon": "", 9 | "width": 150, 10 | "height": 100, 11 | "initW": 150, 12 | "initH": 100, 13 | "isLeftConnectShow": true, 14 | "isRightConnectShow": false, 15 | "containNodes": [], 16 | "attrs": [], 17 | "isSelect": false 18 | }, { 19 | "name": "New_Sercurity_4", 20 | "type": "T2", 21 | "id": "6mcyevkii4g0", 22 | "x": 720, 23 | "y": 220, 24 | "icon": "", 25 | "width": 50, 26 | "height": 50, 27 | "initW": 50, 28 | "initH": 50, 29 | "isLeftConnectShow": true, 30 | "isRightConnectShow": false, 31 | "containNodes": [], 32 | "attrs": [], 33 | "isSelect": false 34 | }, { 35 | "name": "New_message_2", 36 | "type": "T1", 37 | "id": "6we4dyy14ng0", 38 | "x": 60, 39 | "y": 60, 40 | "icon": "", 41 | "width": 425, 42 | "height": 180, 43 | "initW": 150, 44 | "initH": 100, 45 | "isLeftConnectShow": true, 46 | "isRightConnectShow": true, 47 | "containNodes": ["2b6spnktzphc", "4up0c1sqs040"], 48 | "attrs": [], 49 | "isSelect": false 50 | }, { 51 | "name": "New_Compute_2", 52 | "type": "T1", 53 | "id": "2b6spnktzphc", 54 | "x": 85, 55 | "y": 90, 56 | "icon": "", 57 | "width": 200, 58 | "height": 140, 59 | "initW": 150, 60 | "initH": 100, 61 | "isLeftConnectShow": false, 62 | "isRightConnectShow": false, 63 | "containNodes": ["6zj25w7bblc0"], 64 | "attrs": [], 65 | "isSelect": false 66 | }, { 67 | "name": "New_Database_2", 68 | "type": "T1", 69 | "id": "6zj25w7bblc0", 70 | "x": 110, 71 | "y": 120, 72 | "icon": "", 73 | "width": 150, 74 | "height": 100, 75 | "initW": 150, 76 | "initH": 100, 77 | "isLeftConnectShow": false, 78 | "isRightConnectShow": false, 79 | "containNodes": [], 80 | "attrs": [], 81 | "isSelect": false 82 | }, { 83 | "name": "New_Network_3", 84 | "type": "T1", 85 | "id": "4up0c1sqs040", 86 | "x": 310, 87 | "y": 90, 88 | "icon": "", 89 | "width": 150, 90 | "height": 100, 91 | "initW": 150, 92 | "initH": 100, 93 | "isLeftConnectShow": false, 94 | "isRightConnectShow": false, 95 | "containNodes": [], 96 | "attrs": [], 97 | "isSelect": false 98 | }], 99 | "connectors": [{ 100 | "type": "Line", 101 | "targetNode": { 102 | "x": 620, 103 | "y": 40, 104 | "id": "34ls07hv9zw0", 105 | "width": 150, 106 | "height": 100 107 | }, 108 | "sourceNode": { 109 | "x": 60, 110 | "y": 60, 111 | "id": "6we4dyy14ng0", 112 | "width": 425, 113 | "height": 180 114 | }, 115 | "isSelect": false 116 | }, { 117 | "type": "Contain", 118 | "sourceNode": { 119 | "id": "2b6spnktzphc", 120 | "width": 200, 121 | "height": 140, 122 | "x": 85, 123 | "y": 90 124 | }, 125 | "targetNode": { 126 | "id": "6we4dyy14ng0", 127 | "width": 425, 128 | "height": 180, 129 | "x": 60, 130 | "y": 60 131 | }, 132 | "isSelect": false 133 | }, { 134 | "type": "Line", 135 | "targetNode": { 136 | "x": 60, 137 | "y": 60, 138 | "id": "6we4dyy14ng0", 139 | "width": 425, 140 | "height": 180 141 | }, 142 | "sourceNode": { 143 | "x": 60, 144 | "y": 60, 145 | "id": "6we4dyy14ng0", 146 | "width": 425, 147 | "height": 180 148 | }, 149 | "isSelect": false 150 | }, { 151 | "type": "Contain", 152 | "sourceNode": { 153 | "id": "4up0c1sqs040", 154 | "width": 150, 155 | "height": 100, 156 | "x": 310, 157 | "y": 90 158 | }, 159 | "targetNode": { 160 | "id": "6we4dyy14ng0", 161 | "width": 425, 162 | "height": 180, 163 | "x": 60, 164 | "y": 60 165 | }, 166 | "isSelect": false 167 | }, { 168 | "type": "Contain", 169 | "sourceNode": { 170 | "id": "6zj25w7bblc0", 171 | "width": 150, 172 | "height": 100, 173 | "x": 110, 174 | "y": 120 175 | }, 176 | "targetNode": { 177 | "id": "2b6spnktzphc", 178 | "width": 200, 179 | "height": 140, 180 | "x": 85, 181 | "y": 90 182 | }, 183 | "isSelect": false 184 | }, { 185 | "type": "Line", 186 | "targetNode": { 187 | "x": 720, 188 | "y": 220, 189 | "id": "6mcyevkii4g0", 190 | "width": 50, 191 | "height": 50 192 | }, 193 | "sourceNode": { 194 | "x": 60, 195 | "y": 60, 196 | "id": "6we4dyy14ng0", 197 | "width": 425, 198 | "height": 180 199 | }, 200 | "isSelect": false 201 | }] 202 | } --------------------------------------------------------------------------------