forked from Gitlink/forgeplus-react
Merge pull request '并入独立版本' (#202) from jasder/forgeplus-react:standalone into standalone
This commit is contained in:
commit
925d5ea356
|
@ -0,0 +1,2 @@
|
||||||
|
{
|
||||||
|
}
|
|
@ -19,7 +19,8 @@ const getClientEnvironment = require("./env");
|
||||||
|
|
||||||
let publicPath = "/react/build/";
|
let publicPath = "/react/build/";
|
||||||
const publicUrl = publicPath.slice(0, -1);
|
const publicUrl = publicPath.slice(0, -1);
|
||||||
const shouldUseSourceMap = process.env.GENERATE_SOURCEMAP !== "false";
|
// const shouldUseSourceMap = process.env.GENERATE_SOURCEMAP !== "false";
|
||||||
|
const shouldUseSourceMap = process.env.NODE_ENV !== "production";
|
||||||
const env = getClientEnvironment(publicPath);
|
const env = getClientEnvironment(publicPath);
|
||||||
|
|
||||||
// This is the production configuration.
|
// This is the production configuration.
|
||||||
|
@ -54,7 +55,8 @@ module.exports = {
|
||||||
},
|
},
|
||||||
bail: true,
|
bail: true,
|
||||||
mode: "production",
|
mode: "production",
|
||||||
devtool: false, //测试版
|
// devtool: false, //测试版
|
||||||
|
devtool: shouldUseSourceMap?'source-map':false,
|
||||||
entry: [require.resolve("./polyfills"), paths.appIndexJs],
|
entry: [require.resolve("./polyfills"), paths.appIndexJs],
|
||||||
output: {
|
output: {
|
||||||
path: paths.appBuild,
|
path: paths.appBuild,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "educoder",
|
"name": "forge",
|
||||||
"version": "0.1.0",
|
"version": "3.0.0",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -990,6 +990,18 @@
|
||||||
"react-lifecycles-compat": "^3.0.4",
|
"react-lifecycles-compat": "^3.0.4",
|
||||||
"warning": "^4.0.3"
|
"warning": "^4.0.3"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"react-slick": {
|
||||||
|
"version": "0.25.2",
|
||||||
|
"resolved": "https://registry.nlark.com/react-slick/download/react-slick-0.25.2.tgz",
|
||||||
|
"integrity": "sha1-VjMbZ9R9i8/i3OtqyrHI/VvR9rw=",
|
||||||
|
"requires": {
|
||||||
|
"classnames": "^2.2.5",
|
||||||
|
"enquire.js": "^2.1.6",
|
||||||
|
"json2mq": "^0.2.0",
|
||||||
|
"lodash.debounce": "^4.0.8",
|
||||||
|
"resize-observer-polyfill": "^1.5.0"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -1261,7 +1273,7 @@
|
||||||
},
|
},
|
||||||
"babel-cli": {
|
"babel-cli": {
|
||||||
"version": "6.26.0",
|
"version": "6.26.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-cli/download/babel-cli-6.26.0.tgz",
|
"resolved": "https://registry.npmjs.org/babel-cli/-/babel-cli-6.26.0.tgz",
|
||||||
"integrity": "sha1-UCq1SHTX24itALiHoGODzgPQAvE=",
|
"integrity": "sha1-UCq1SHTX24itALiHoGODzgPQAvE=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -1284,7 +1296,7 @@
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"chokidar": {
|
"chokidar": {
|
||||||
"version": "1.7.0",
|
"version": "1.7.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/chokidar/download/chokidar-1.7.0.tgz?cache=0&sync_timestamp=1602585438968&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fchokidar%2Fdownload%2Fchokidar-1.7.0.tgz",
|
"resolved": "https://registry.npmjs.org/chokidar/-/chokidar-1.7.0.tgz",
|
||||||
"integrity": "sha1-eY5ol3gVHIB2tLNg5e3SjNortGg=",
|
"integrity": "sha1-eY5ol3gVHIB2tLNg5e3SjNortGg=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"optional": true,
|
"optional": true,
|
||||||
|
@ -1302,7 +1314,7 @@
|
||||||
},
|
},
|
||||||
"source-map": {
|
"source-map": {
|
||||||
"version": "0.5.7",
|
"version": "0.5.7",
|
||||||
"resolved": "https://registry.npm.taobao.org/source-map/download/source-map-0.5.7.tgz",
|
"resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz",
|
||||||
"integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=",
|
"integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=",
|
||||||
"dev": true
|
"dev": true
|
||||||
}
|
}
|
||||||
|
@ -1401,7 +1413,7 @@
|
||||||
},
|
},
|
||||||
"babel-helper-bindify-decorators": {
|
"babel-helper-bindify-decorators": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-helper-bindify-decorators/download/babel-helper-bindify-decorators-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-helper-bindify-decorators/-/babel-helper-bindify-decorators-6.24.1.tgz",
|
||||||
"integrity": "sha1-FMGeXxQte0fxmlJDHlKxzLxAozA=",
|
"integrity": "sha1-FMGeXxQte0fxmlJDHlKxzLxAozA=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -1464,7 +1476,7 @@
|
||||||
},
|
},
|
||||||
"babel-helper-explode-class": {
|
"babel-helper-explode-class": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-helper-explode-class/download/babel-helper-explode-class-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-helper-explode-class/-/babel-helper-explode-class-6.24.1.tgz",
|
||||||
"integrity": "sha1-fcKjkQ3uAHBW4eMdZAztPVTqqes=",
|
"integrity": "sha1-fcKjkQ3uAHBW4eMdZAztPVTqqes=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -1664,7 +1676,7 @@
|
||||||
},
|
},
|
||||||
"babel-plugin-syntax-async-generators": {
|
"babel-plugin-syntax-async-generators": {
|
||||||
"version": "6.13.0",
|
"version": "6.13.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-plugin-syntax-async-generators/download/babel-plugin-syntax-async-generators-6.13.0.tgz",
|
"resolved": "https://registry.npmjs.org/babel-plugin-syntax-async-generators/-/babel-plugin-syntax-async-generators-6.13.0.tgz",
|
||||||
"integrity": "sha1-a8lj67FuzLrmuStZbrfzXDQqi5o=",
|
"integrity": "sha1-a8lj67FuzLrmuStZbrfzXDQqi5o=",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
@ -1675,7 +1687,7 @@
|
||||||
},
|
},
|
||||||
"babel-plugin-syntax-decorators": {
|
"babel-plugin-syntax-decorators": {
|
||||||
"version": "6.13.0",
|
"version": "6.13.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-plugin-syntax-decorators/download/babel-plugin-syntax-decorators-6.13.0.tgz",
|
"resolved": "https://registry.npmjs.org/babel-plugin-syntax-decorators/-/babel-plugin-syntax-decorators-6.13.0.tgz",
|
||||||
"integrity": "sha1-MSVjtNvePMgGzuPkFszurd0RrAs=",
|
"integrity": "sha1-MSVjtNvePMgGzuPkFszurd0RrAs=",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
@ -1711,7 +1723,7 @@
|
||||||
},
|
},
|
||||||
"babel-plugin-transform-async-generator-functions": {
|
"babel-plugin-transform-async-generator-functions": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-plugin-transform-async-generator-functions/download/babel-plugin-transform-async-generator-functions-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-plugin-transform-async-generator-functions/-/babel-plugin-transform-async-generator-functions-6.24.1.tgz",
|
||||||
"integrity": "sha1-8FiQAUX9PpkHpt3yjaWfIVJYpds=",
|
"integrity": "sha1-8FiQAUX9PpkHpt3yjaWfIVJYpds=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -1743,7 +1755,7 @@
|
||||||
},
|
},
|
||||||
"babel-plugin-transform-decorators": {
|
"babel-plugin-transform-decorators": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-plugin-transform-decorators/download/babel-plugin-transform-decorators-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-plugin-transform-decorators/-/babel-plugin-transform-decorators-6.24.1.tgz",
|
||||||
"integrity": "sha1-eIAT2PjGtSIr33s0Q5Df13Vp4k0=",
|
"integrity": "sha1-eIAT2PjGtSIr33s0Q5Df13Vp4k0=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -2048,7 +2060,7 @@
|
||||||
},
|
},
|
||||||
"babel-plugin-transform-runtime": {
|
"babel-plugin-transform-runtime": {
|
||||||
"version": "6.23.0",
|
"version": "6.23.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-plugin-transform-runtime/download/babel-plugin-transform-runtime-6.23.0.tgz",
|
"resolved": "https://registry.npmjs.org/babel-plugin-transform-runtime/-/babel-plugin-transform-runtime-6.23.0.tgz",
|
||||||
"integrity": "sha1-iEkNRGUC6puOfvsP4J7E2ZR5se4=",
|
"integrity": "sha1-iEkNRGUC6puOfvsP4J7E2ZR5se4=",
|
||||||
"requires": {
|
"requires": {
|
||||||
"babel-runtime": "^6.22.0"
|
"babel-runtime": "^6.22.0"
|
||||||
|
@ -2065,9 +2077,8 @@
|
||||||
},
|
},
|
||||||
"babel-polyfill": {
|
"babel-polyfill": {
|
||||||
"version": "6.26.0",
|
"version": "6.26.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-polyfill/download/babel-polyfill-6.26.0.tgz",
|
"resolved": "https://registry.npmjs.org/babel-polyfill/-/babel-polyfill-6.26.0.tgz",
|
||||||
"integrity": "sha1-N5k3q8Z9eJWXCtxiHyhM2WbPIVM=",
|
"integrity": "sha1-N5k3q8Z9eJWXCtxiHyhM2WbPIVM=",
|
||||||
"dev": true,
|
|
||||||
"requires": {
|
"requires": {
|
||||||
"babel-runtime": "^6.26.0",
|
"babel-runtime": "^6.26.0",
|
||||||
"core-js": "^2.5.0",
|
"core-js": "^2.5.0",
|
||||||
|
@ -2076,9 +2087,8 @@
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"regenerator-runtime": {
|
"regenerator-runtime": {
|
||||||
"version": "0.10.5",
|
"version": "0.10.5",
|
||||||
"resolved": "https://registry.npm.taobao.org/regenerator-runtime/download/regenerator-runtime-0.10.5.tgz?cache=0&sync_timestamp=1595456367497&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fregenerator-runtime%2Fdownload%2Fregenerator-runtime-0.10.5.tgz",
|
"resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.10.5.tgz",
|
||||||
"integrity": "sha1-M2w+/BIgrc7dosn6tntaeVWjNlg=",
|
"integrity": "sha1-M2w+/BIgrc7dosn6tntaeVWjNlg="
|
||||||
"dev": true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -2121,7 +2131,7 @@
|
||||||
},
|
},
|
||||||
"babel-preset-es2015": {
|
"babel-preset-es2015": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-preset-es2015/download/babel-preset-es2015-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-preset-es2015/-/babel-preset-es2015-6.24.1.tgz",
|
||||||
"integrity": "sha1-1EBQ1rwsn+6nAqrzjXJ6AhBTiTk=",
|
"integrity": "sha1-1EBQ1rwsn+6nAqrzjXJ6AhBTiTk=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -2169,7 +2179,7 @@
|
||||||
},
|
},
|
||||||
"babel-preset-react": {
|
"babel-preset-react": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-preset-react/download/babel-preset-react-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-preset-react/-/babel-preset-react-6.24.1.tgz",
|
||||||
"integrity": "sha1-umnfrqRfw+xjm2pOzqbhdwLJE4A=",
|
"integrity": "sha1-umnfrqRfw+xjm2pOzqbhdwLJE4A=",
|
||||||
"requires": {
|
"requires": {
|
||||||
"babel-plugin-syntax-jsx": "^6.3.13",
|
"babel-plugin-syntax-jsx": "^6.3.13",
|
||||||
|
@ -2202,7 +2212,7 @@
|
||||||
},
|
},
|
||||||
"babel-preset-stage-2": {
|
"babel-preset-stage-2": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-preset-stage-2/download/babel-preset-stage-2-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-preset-stage-2/-/babel-preset-stage-2-6.24.1.tgz",
|
||||||
"integrity": "sha1-2eKWD7PXEYfw5k7sYrwHdnIZvcE=",
|
"integrity": "sha1-2eKWD7PXEYfw5k7sYrwHdnIZvcE=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -2214,7 +2224,7 @@
|
||||||
},
|
},
|
||||||
"babel-preset-stage-3": {
|
"babel-preset-stage-3": {
|
||||||
"version": "6.24.1",
|
"version": "6.24.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/babel-preset-stage-3/download/babel-preset-stage-3-6.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/babel-preset-stage-3/-/babel-preset-stage-3-6.24.1.tgz",
|
||||||
"integrity": "sha1-g2raCp56f6N8sTj7kyb4eTSkg5U=",
|
"integrity": "sha1-g2raCp56f6N8sTj7kyb4eTSkg5U=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -2367,6 +2377,12 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"base16": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/base16/download/base16-1.0.0.tgz",
|
||||||
|
"integrity": "sha1-4pf2DX7BAUp6lxo568ipjAtoHnA=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"base64-js": {
|
"base64-js": {
|
||||||
"version": "1.3.1",
|
"version": "1.3.1",
|
||||||
"resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.1.tgz",
|
"resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.1.tgz",
|
||||||
|
@ -3409,9 +3425,9 @@
|
||||||
"integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw=="
|
"integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw=="
|
||||||
},
|
},
|
||||||
"clipboard": {
|
"clipboard": {
|
||||||
"version": "2.0.6",
|
"version": "2.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/clipboard/-/clipboard-2.0.6.tgz",
|
"resolved": "https://registry.npmjs.org/clipboard/-/clipboard-2.0.8.tgz",
|
||||||
"integrity": "sha512-g5zbiixBRk/wyKakSwCKd7vQXDjFnAMGHoEyBogG/bw9kTD9GvdAvaoRR1ALcEzt3pVKxZR0pViekPMIS0QyGg==",
|
"integrity": "sha512-Y6WO0unAIQp5bLmk1zdThRhgJt/x3ks6f30s3oE3H1mgIEU33XyQjEf8gsf6DxC7NPX8Y1SsNWjUjL/ywLnnbQ==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"good-listener": "^1.2.2",
|
"good-listener": "^1.2.2",
|
||||||
"select": "^1.1.2",
|
"select": "^1.1.2",
|
||||||
|
@ -3491,7 +3507,7 @@
|
||||||
},
|
},
|
||||||
"code-prettify": {
|
"code-prettify": {
|
||||||
"version": "0.1.0",
|
"version": "0.1.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/code-prettify/download/code-prettify-0.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/code-prettify/-/code-prettify-0.1.0.tgz",
|
||||||
"integrity": "sha1-RocMyMGlDQm61TmzOpg9vUqjSx4="
|
"integrity": "sha1-RocMyMGlDQm61TmzOpg9vUqjSx4="
|
||||||
},
|
},
|
||||||
"codemirror": {
|
"codemirror": {
|
||||||
|
@ -3859,6 +3875,69 @@
|
||||||
"warning": "^4.0.3"
|
"warning": "^4.0.3"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"cross-env": {
|
||||||
|
"version": "7.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/cross-env/-/cross-env-7.0.3.tgz",
|
||||||
|
"integrity": "sha512-+/HKd6EgcQCJGh2PSjZuUitQBQynKor4wrFbRg4DtAgS1aWO+gU52xpH7M9ScGgXSYmAVS9bIJ8EzuaGw0oNAw==",
|
||||||
|
"requires": {
|
||||||
|
"cross-spawn": "^7.0.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"cross-spawn": {
|
||||||
|
"version": "7.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
|
||||||
|
"integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
|
||||||
|
"requires": {
|
||||||
|
"path-key": "^3.1.0",
|
||||||
|
"shebang-command": "^2.0.0",
|
||||||
|
"which": "^2.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"path-key": {
|
||||||
|
"version": "3.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
|
||||||
|
"integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q=="
|
||||||
|
},
|
||||||
|
"shebang-command": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
|
||||||
|
"requires": {
|
||||||
|
"shebang-regex": "^3.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"shebang-regex": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A=="
|
||||||
|
},
|
||||||
|
"which": {
|
||||||
|
"version": "2.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
|
||||||
|
"integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
|
||||||
|
"requires": {
|
||||||
|
"isexe": "^2.0.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"cross-fetch": {
|
||||||
|
"version": "3.1.4",
|
||||||
|
"resolved": "https://registry.nlark.com/cross-fetch/download/cross-fetch-3.1.4.tgz",
|
||||||
|
"integrity": "sha1-lyPzo6JHv4uJA586OAqSROj6Lzk=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"node-fetch": "2.6.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"node-fetch": {
|
||||||
|
"version": "2.6.1",
|
||||||
|
"resolved": "https://registry.nlark.com/node-fetch/download/node-fetch-2.6.1.tgz?cache=0&sync_timestamp=1626684741221&other_urls=https%3A%2F%2Fregistry.nlark.com%2Fnode-fetch%2Fdownload%2Fnode-fetch-2.6.1.tgz",
|
||||||
|
"integrity": "sha1-BFvTI2Mfdu0uK1VXM5RBa2OaAFI=",
|
||||||
|
"dev": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"cross-spawn": {
|
"cross-spawn": {
|
||||||
"version": "5.1.0",
|
"version": "5.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz",
|
||||||
|
@ -3893,11 +3972,6 @@
|
||||||
"randomfill": "^1.0.3"
|
"randomfill": "^1.0.3"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"crypto-js": {
|
|
||||||
"version": "4.0.0",
|
|
||||||
"resolved": "https://registry.npm.taobao.org/crypto-js/download/crypto-js-4.0.0.tgz",
|
|
||||||
"integrity": "sha1-KQSrJnep0EKFai6i74DekuSjbcw="
|
|
||||||
},
|
|
||||||
"crypto-random-string": {
|
"crypto-random-string": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz",
|
||||||
|
@ -5063,11 +5137,11 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"echarts": {
|
"echarts": {
|
||||||
"version": "4.7.0",
|
"version": "4.9.0",
|
||||||
"resolved": "https://registry.npmjs.org/echarts/-/echarts-4.7.0.tgz",
|
"resolved": "https://registry.nlark.com/echarts/download/echarts-4.9.0.tgz",
|
||||||
"integrity": "sha512-NlOTdUcAsIyCCG+N4uh0ZEvXtrPW2jvcuqf03RyqYeCKzyPbiOQ4I3MdKXMhxG3lBdqQNdNXVT71SB4KTQjN0A==",
|
"integrity": "sha1-qbm6oD8Doqcx5jQMVb77V6nhNH0=",
|
||||||
"requires": {
|
"requires": {
|
||||||
"zrender": "4.3.0"
|
"zrender": "4.3.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"echarts-wordcloud": {
|
"echarts-wordcloud": {
|
||||||
|
@ -6209,6 +6283,41 @@
|
||||||
"bser": "2.1.1"
|
"bser": "2.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"fbemitter": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/fbemitter/download/fbemitter-3.0.0.tgz",
|
||||||
|
"integrity": "sha1-ALKhr1QRJUqrQWzXX55iib7kv/M=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"fbjs": "^3.0.0"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"fbjs": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/fbjs/download/fbjs-3.0.0.tgz?cache=0&sync_timestamp=1602048313843&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Ffbjs%2Fdownload%2Ffbjs-3.0.0.tgz",
|
||||||
|
"integrity": "sha1-CQcGf7P1enj0XZXx6s/8rNYjwWU=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"cross-fetch": "^3.0.4",
|
||||||
|
"fbjs-css-vars": "^1.0.0",
|
||||||
|
"loose-envify": "^1.0.0",
|
||||||
|
"object-assign": "^4.1.0",
|
||||||
|
"promise": "^7.1.1",
|
||||||
|
"setimmediate": "^1.0.5",
|
||||||
|
"ua-parser-js": "^0.7.18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"promise": {
|
||||||
|
"version": "7.3.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/promise/download/promise-7.3.1.tgz",
|
||||||
|
"integrity": "sha1-BktyYCsY+Q8pGSuLG8QY/9Hr078=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"asap": "~2.0.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"fbjs": {
|
"fbjs": {
|
||||||
"version": "0.8.17",
|
"version": "0.8.17",
|
||||||
"resolved": "https://registry.npmjs.org/fbjs/-/fbjs-0.8.17.tgz",
|
"resolved": "https://registry.npmjs.org/fbjs/-/fbjs-0.8.17.tgz",
|
||||||
|
@ -6238,6 +6347,12 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"fbjs-css-vars": {
|
||||||
|
"version": "1.0.2",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/fbjs-css-vars/download/fbjs-css-vars-1.0.2.tgz",
|
||||||
|
"integrity": "sha1-IWVRE2rgL+JVkyw+yHdfGOLAeLg=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"fecha": {
|
"fecha": {
|
||||||
"version": "2.3.3",
|
"version": "2.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/fecha/-/fecha-2.3.3.tgz",
|
"resolved": "https://registry.npmjs.org/fecha/-/fecha-2.3.3.tgz",
|
||||||
|
@ -6682,6 +6797,42 @@
|
||||||
"readable-stream": "^2.3.6"
|
"readable-stream": "^2.3.6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"flux": {
|
||||||
|
"version": "4.0.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/flux/download/flux-4.0.1.tgz",
|
||||||
|
"integrity": "sha1-eENQKwKEHUqqU0rws3MDSh917lw=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"fbemitter": "^3.0.0",
|
||||||
|
"fbjs": "^3.0.0"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"fbjs": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/fbjs/download/fbjs-3.0.0.tgz?cache=0&sync_timestamp=1602048313843&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Ffbjs%2Fdownload%2Ffbjs-3.0.0.tgz",
|
||||||
|
"integrity": "sha1-CQcGf7P1enj0XZXx6s/8rNYjwWU=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"cross-fetch": "^3.0.4",
|
||||||
|
"fbjs-css-vars": "^1.0.0",
|
||||||
|
"loose-envify": "^1.0.0",
|
||||||
|
"object-assign": "^4.1.0",
|
||||||
|
"promise": "^7.1.1",
|
||||||
|
"setimmediate": "^1.0.5",
|
||||||
|
"ua-parser-js": "^0.7.18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"promise": {
|
||||||
|
"version": "7.3.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/promise/download/promise-7.3.1.tgz",
|
||||||
|
"integrity": "sha1-BktyYCsY+Q8pGSuLG8QY/9Hr078=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"asap": "~2.0.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"flv.js": {
|
"flv.js": {
|
||||||
"version": "1.5.0",
|
"version": "1.5.0",
|
||||||
"resolved": "https://registry.npmjs.org/flv.js/-/flv.js-1.5.0.tgz",
|
"resolved": "https://registry.npmjs.org/flv.js/-/flv.js-1.5.0.tgz",
|
||||||
|
@ -9898,6 +10049,11 @@
|
||||||
"esprima": "^4.0.0"
|
"esprima": "^4.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"js2wordcloud": {
|
||||||
|
"version": "1.1.12",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/js2wordcloud/download/js2wordcloud-1.1.12.tgz",
|
||||||
|
"integrity": "sha1-8BdC2k5qyzAAsoY1dmQWjvn8o64="
|
||||||
|
},
|
||||||
"jsbn": {
|
"jsbn": {
|
||||||
"version": "0.1.1",
|
"version": "0.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
|
||||||
|
@ -10320,6 +10476,12 @@
|
||||||
"resolved": "https://registry.npmjs.org/lodash.cond/-/lodash.cond-4.5.2.tgz",
|
"resolved": "https://registry.npmjs.org/lodash.cond/-/lodash.cond-4.5.2.tgz",
|
||||||
"integrity": "sha1-9HGh2khr5g9quVXRcRVSPdHSVdU="
|
"integrity": "sha1-9HGh2khr5g9quVXRcRVSPdHSVdU="
|
||||||
},
|
},
|
||||||
|
"lodash.curry": {
|
||||||
|
"version": "4.1.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/lodash.curry/download/lodash.curry-4.1.1.tgz",
|
||||||
|
"integrity": "sha1-JI42By7ekGUB11lmIAqG2riyMXA=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"lodash.debounce": {
|
"lodash.debounce": {
|
||||||
"version": "4.0.8",
|
"version": "4.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz",
|
"resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz",
|
||||||
|
@ -10330,6 +10492,12 @@
|
||||||
"resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz",
|
||||||
"integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw="
|
"integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw="
|
||||||
},
|
},
|
||||||
|
"lodash.flow": {
|
||||||
|
"version": "3.5.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/lodash.flow/download/lodash.flow-3.5.0.tgz",
|
||||||
|
"integrity": "sha1-h79AKSuM+D5OjOGjrkIJ4gBxZ1o=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"lodash.isequal": {
|
"lodash.isequal": {
|
||||||
"version": "4.5.0",
|
"version": "4.5.0",
|
||||||
"resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz",
|
"resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz",
|
||||||
|
@ -11551,7 +11719,7 @@
|
||||||
},
|
},
|
||||||
"output-file-sync": {
|
"output-file-sync": {
|
||||||
"version": "1.1.2",
|
"version": "1.1.2",
|
||||||
"resolved": "https://registry.npm.taobao.org/output-file-sync/download/output-file-sync-1.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/output-file-sync/-/output-file-sync-1.1.2.tgz",
|
||||||
"integrity": "sha1-0KM+7+YaIF+suQCS6CZZjVJFznY=",
|
"integrity": "sha1-0KM+7+YaIF+suQCS6CZZjVJFznY=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -13960,6 +14128,12 @@
|
||||||
"resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz",
|
||||||
"integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A=="
|
"integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A=="
|
||||||
},
|
},
|
||||||
|
"pure-color": {
|
||||||
|
"version": "1.3.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/pure-color/download/pure-color-1.3.0.tgz",
|
||||||
|
"integrity": "sha1-H+Bk+wrIUfDeYTIKi/eWg2Qi8z4=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"purgecss": {
|
"purgecss": {
|
||||||
"version": "2.1.2",
|
"version": "2.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/purgecss/-/purgecss-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/purgecss/-/purgecss-2.1.2.tgz",
|
||||||
|
@ -14034,9 +14208,9 @@
|
||||||
"integrity": "sha1-ys6GOG9ZoNuAUPqQ2baw6IoeNk8="
|
"integrity": "sha1-ys6GOG9ZoNuAUPqQ2baw6IoeNk8="
|
||||||
},
|
},
|
||||||
"qrcode.react": {
|
"qrcode.react": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/qrcode.react/-/qrcode.react-1.0.0.tgz",
|
"resolved": "https://registry.npm.taobao.org/qrcode.react/download/qrcode.react-1.0.1.tgz",
|
||||||
"integrity": "sha512-jBXleohRTwvGBe1ngV+62QvEZ/9IZqQivdwzo9pJM4LQMoCM2VnvNBnKdjvGnKyDZ/l0nCDgsPod19RzlPvm/Q==",
|
"integrity": "sha1-KDS7UOXidf/lr2kG7/FTkf6eOKU=",
|
||||||
"requires": {
|
"requires": {
|
||||||
"loose-envify": "^1.4.0",
|
"loose-envify": "^1.4.0",
|
||||||
"prop-types": "^15.6.0",
|
"prop-types": "^15.6.0",
|
||||||
|
@ -14731,6 +14905,18 @@
|
||||||
"prop-types": "^15.6.2"
|
"prop-types": "^15.6.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"react-base16-styling": {
|
||||||
|
"version": "0.6.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/react-base16-styling/download/react-base16-styling-0.6.0.tgz",
|
||||||
|
"integrity": "sha1-7yFW1mz0E5aVyKFniGy2nqZgeSw=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"base16": "^1.0.0",
|
||||||
|
"lodash.curry": "^4.0.1",
|
||||||
|
"lodash.flow": "^3.3.0",
|
||||||
|
"pure-color": "^1.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"react-beautiful-dnd": {
|
"react-beautiful-dnd": {
|
||||||
"version": "10.1.1",
|
"version": "10.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/react-beautiful-dnd/-/react-beautiful-dnd-10.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/react-beautiful-dnd/-/react-beautiful-dnd-10.1.1.tgz",
|
||||||
|
@ -15096,6 +15282,18 @@
|
||||||
"resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz",
|
"resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz",
|
||||||
"integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ=="
|
"integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ=="
|
||||||
},
|
},
|
||||||
|
"react-json-view": {
|
||||||
|
"version": "1.21.3",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/react-json-view/download/react-json-view-1.21.3.tgz",
|
||||||
|
"integrity": "sha1-8YQgnujxvzdPsMQbCBPP9UVJxHU=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"flux": "^4.0.1",
|
||||||
|
"react-base16-styling": "^0.6.0",
|
||||||
|
"react-lifecycles-compat": "^3.0.4",
|
||||||
|
"react-textarea-autosize": "^8.3.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
"react-jss": {
|
"react-jss": {
|
||||||
"version": "8.6.1",
|
"version": "8.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/react-jss/-/react-jss-8.6.1.tgz",
|
"resolved": "https://registry.npmjs.org/react-jss/-/react-jss-8.6.1.tgz",
|
||||||
|
@ -15255,9 +15453,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"react-slick": {
|
"react-slick": {
|
||||||
"version": "0.25.2",
|
"version": "0.28.1",
|
||||||
"resolved": "https://registry.nlark.com/react-slick/download/react-slick-0.25.2.tgz",
|
"resolved": "https://registry.nlark.com/react-slick/download/react-slick-0.28.1.tgz",
|
||||||
"integrity": "sha1-VjMbZ9R9i8/i3OtqyrHI/VvR9rw=",
|
"integrity": "sha1-EsGNmRtZQy35w3V7pUCiJ7P7hbk=",
|
||||||
"requires": {
|
"requires": {
|
||||||
"classnames": "^2.2.5",
|
"classnames": "^2.2.5",
|
||||||
"enquire.js": "^2.1.6",
|
"enquire.js": "^2.1.6",
|
||||||
|
@ -15284,6 +15482,34 @@
|
||||||
"prop-types": "^15.5.4"
|
"prop-types": "^15.5.4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"react-textarea-autosize": {
|
||||||
|
"version": "8.3.3",
|
||||||
|
"resolved": "https://registry.nlark.com/react-textarea-autosize/download/react-textarea-autosize-8.3.3.tgz?cache=0&sync_timestamp=1622628433420&other_urls=https%3A%2F%2Fregistry.nlark.com%2Freact-textarea-autosize%2Fdownload%2Freact-textarea-autosize-8.3.3.tgz",
|
||||||
|
"integrity": "sha1-9wkTlFNp2kU/1VTBaPa6rNH6BNg=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@babel/runtime": "^7.10.2",
|
||||||
|
"use-composed-ref": "^1.0.0",
|
||||||
|
"use-latest": "^1.0.0"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@babel/runtime": {
|
||||||
|
"version": "7.14.8",
|
||||||
|
"resolved": "https://registry.nlark.com/@babel/runtime/download/@babel/runtime-7.14.8.tgz",
|
||||||
|
"integrity": "sha1-cRmlb0IQGIUmlCkLn5FICXORtEY=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"regenerator-runtime": "^0.13.4"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"regenerator-runtime": {
|
||||||
|
"version": "0.13.9",
|
||||||
|
"resolved": "https://registry.nlark.com/regenerator-runtime/download/regenerator-runtime-0.13.9.tgz?cache=0&sync_timestamp=1626993001371&other_urls=https%3A%2F%2Fregistry.nlark.com%2Fregenerator-runtime%2Fdownload%2Fregenerator-runtime-0.13.9.tgz",
|
||||||
|
"integrity": "sha1-iSV0Kpj/2QgUmI11Zq0wyjsmO1I=",
|
||||||
|
"dev": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"react-transition-group": {
|
"react-transition-group": {
|
||||||
"version": "2.9.0",
|
"version": "2.9.0",
|
||||||
"resolved": "https://registry.npmjs.org/react-transition-group/-/react-transition-group-2.9.0.tgz",
|
"resolved": "https://registry.npmjs.org/react-transition-group/-/react-transition-group-2.9.0.tgz",
|
||||||
|
@ -16406,6 +16632,11 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"save-dev": {
|
||||||
|
"version": "0.0.1-security",
|
||||||
|
"resolved": "https://registry.npmjs.org/save-dev/-/save-dev-0.0.1-security.tgz",
|
||||||
|
"integrity": "sha512-k6knZTDNK8PKKbIqnvxiOveJinuw2LcQjqDoaorZWP9M5AR2EPsnpDeSbeoZZ0pHr5ze1uoaKdK8NBGQrJ34Uw=="
|
||||||
|
},
|
||||||
"sax": {
|
"sax": {
|
||||||
"version": "1.2.4",
|
"version": "1.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
|
"resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
|
||||||
|
@ -17040,6 +17271,11 @@
|
||||||
"is-fullwidth-code-point": "^2.0.0"
|
"is-fullwidth-code-point": "^2.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"slick-carousel": {
|
||||||
|
"version": "1.8.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/slick-carousel/download/slick-carousel-1.8.1.tgz?cache=0&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fslick-carousel%2Fdownload%2Fslick-carousel-1.8.1.tgz",
|
||||||
|
"integrity": "sha1-pL+ykBSIe7Zs5Si5C9DNomLMj40="
|
||||||
|
},
|
||||||
"snapdragon": {
|
"snapdragon": {
|
||||||
"version": "0.8.2",
|
"version": "0.8.2",
|
||||||
"resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz",
|
"resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz",
|
||||||
|
@ -18374,6 +18610,12 @@
|
||||||
"integrity": "sha512-c3zayb8/kWWpycWYg87P71E1S1ZL6b6IJxfb5fvsUgsf0S2MVGaDhDXXjDMpdCpfWXqptc+4mXwmiy1ypXqRAA==",
|
"integrity": "sha512-c3zayb8/kWWpycWYg87P71E1S1ZL6b6IJxfb5fvsUgsf0S2MVGaDhDXXjDMpdCpfWXqptc+4mXwmiy1ypXqRAA==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"ts-essentials": {
|
||||||
|
"version": "2.0.12",
|
||||||
|
"resolved": "https://registry.nlark.com/ts-essentials/download/ts-essentials-2.0.12.tgz",
|
||||||
|
"integrity": "sha1-yTA/PXT3X6dSjD1JuA4ImrCdh0U=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"tslib": {
|
"tslib": {
|
||||||
"version": "1.11.1",
|
"version": "1.11.1",
|
||||||
"resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz",
|
"resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz",
|
||||||
|
@ -18883,9 +19125,33 @@
|
||||||
"resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz",
|
||||||
"integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ=="
|
"integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ=="
|
||||||
},
|
},
|
||||||
|
"use-composed-ref": {
|
||||||
|
"version": "1.1.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/use-composed-ref/download/use-composed-ref-1.1.0.tgz",
|
||||||
|
"integrity": "sha1-kiDk6UqXt7AtfSfq6rCzcDRDi7w=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"ts-essentials": "^2.0.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"use-isomorphic-layout-effect": {
|
||||||
|
"version": "1.1.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/use-isomorphic-layout-effect/download/use-isomorphic-layout-effect-1.1.1.tgz",
|
||||||
|
"integrity": "sha1-e7ZYkXDNKYehUgQvkIT57/t1wiU=",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
|
"use-latest": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/use-latest/download/use-latest-1.2.0.tgz",
|
||||||
|
"integrity": "sha1-pE9lcrgojgly7EEb3QhAraNm8jI=",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"use-isomorphic-layout-effect": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"user-home": {
|
"user-home": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/user-home/download/user-home-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/user-home/-/user-home-1.1.1.tgz",
|
||||||
"integrity": "sha1-K1viOjK2Onyd640PKNSFcko98ZA=",
|
"integrity": "sha1-K1viOjK2Onyd640PKNSFcko98ZA=",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
@ -18942,7 +19208,7 @@
|
||||||
},
|
},
|
||||||
"v8flags": {
|
"v8flags": {
|
||||||
"version": "2.1.1",
|
"version": "2.1.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/v8flags/download/v8flags-2.1.1.tgz?cache=0&sync_timestamp=1590964281452&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fv8flags%2Fdownload%2Fv8flags-2.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/v8flags/-/v8flags-2.1.1.tgz",
|
||||||
"integrity": "sha1-qrGh+jDUX4jdMhFIh1rALAtV5bQ=",
|
"integrity": "sha1-qrGh+jDUX4jdMhFIh1rALAtV5bQ=",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
|
@ -20486,6 +20752,16 @@
|
||||||
"integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
|
"integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"xterm": {
|
||||||
|
"version": "4.8.1",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/xterm/download/xterm-4.8.1.tgz",
|
||||||
|
"integrity": "sha1-FVoXKaQ+Gom0BlJOIsVjQznjnKE="
|
||||||
|
},
|
||||||
|
"xterm-addon-fit": {
|
||||||
|
"version": "0.4.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/xterm-addon-fit/download/xterm-addon-fit-0.4.0.tgz",
|
||||||
|
"integrity": "sha1-BuDF0KaqrPsAnvVl76HIHpPZAZM="
|
||||||
|
},
|
||||||
"y18n": {
|
"y18n": {
|
||||||
"version": "3.2.1",
|
"version": "3.2.1",
|
||||||
"resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz",
|
"resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz",
|
||||||
|
@ -20523,9 +20799,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"zrender": {
|
"zrender": {
|
||||||
"version": "4.3.0",
|
"version": "4.3.2",
|
||||||
"resolved": "https://registry.npmjs.org/zrender/-/zrender-4.3.0.tgz",
|
"resolved": "https://registry.npm.taobao.org/zrender/download/zrender-4.3.2.tgz",
|
||||||
"integrity": "sha512-Dii6j2bDsPkxQayuVf2DXJeruIB/mKVxxcGRZQ9GExiBd4c3w7+oBuvo1O/JGHeFeA1nCmSDVDs/S7yKZG1nrA=="
|
"integrity": "sha1-7HQy+UFcgsc1hLa3uMR+GwFiCcY="
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
10
package.json
10
package.json
|
@ -22,10 +22,11 @@
|
||||||
"case-sensitive-paths-webpack-plugin": "2.1.1",
|
"case-sensitive-paths-webpack-plugin": "2.1.1",
|
||||||
"chalk": "1.1.3",
|
"chalk": "1.1.3",
|
||||||
"classnames": "^2.2.5",
|
"classnames": "^2.2.5",
|
||||||
"clipboard": "^2.0.6",
|
"clipboard": "^2.0.8",
|
||||||
"code-prettify": "^0.1.0",
|
"code-prettify": "^0.1.0",
|
||||||
"codemirror": "^5.53.0",
|
"codemirror": "^5.53.0",
|
||||||
"connected-react-router": "4.4.1",
|
"connected-react-router": "4.4.1",
|
||||||
|
"cross-env": "^7.0.3",
|
||||||
"css-loader": "^3.5.2",
|
"css-loader": "^3.5.2",
|
||||||
"dompurify": "^2.0.15",
|
"dompurify": "^2.0.15",
|
||||||
"dotenv": "4.0.0",
|
"dotenv": "4.0.0",
|
||||||
|
@ -103,6 +104,7 @@
|
||||||
"redux-thunk": "2.3.0",
|
"redux-thunk": "2.3.0",
|
||||||
"rsuite": "^4.3.4",
|
"rsuite": "^4.3.4",
|
||||||
"sass-loader": "7.3.1",
|
"sass-loader": "7.3.1",
|
||||||
|
"save-dev": "0.0.1-security",
|
||||||
"scroll-into-view": "^1.14.2",
|
"scroll-into-view": "^1.14.2",
|
||||||
"showdown": "^1.9.1",
|
"showdown": "^1.9.1",
|
||||||
"showdown-katex": "^0.8.0",
|
"showdown-katex": "^0.8.0",
|
||||||
|
@ -122,8 +124,8 @@
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node --max_old_space_size=15360 scripts/start.js",
|
"start": "node --max_old_space_size=15360 scripts/start.js",
|
||||||
"build": "NODE_ENV=production node --max_old_space_size=15360 scripts/build.js",
|
"build": "cross-env NODE_ENV=production node --max_old_space_size=15360 scripts/build.js",
|
||||||
"test-build": "NODE_ENV=testBuild node --max_old_space_size=15360 scripts/build.js",
|
"test-build": "cross-env NODE_ENV=testBuild node --max_old_space_size=15360 scripts/build.js",
|
||||||
"pre-build": "NODE_ENV=preBuild node --max_old_space_size=15360 scripts/build.js",
|
"pre-build": "NODE_ENV=preBuild node --max_old_space_size=15360 scripts/build.js",
|
||||||
"gen_stats": "NODE_ENV=production webpack --profile --config=./config/webpack.config.prod.js --json > stats.json",
|
"gen_stats": "NODE_ENV=production webpack --profile --config=./config/webpack.config.prod.js --json > stats.json",
|
||||||
"ana": "webpack-bundle-analyzer ./stats.json",
|
"ana": "webpack-bundle-analyzer ./stats.json",
|
||||||
|
@ -193,6 +195,7 @@
|
||||||
"babel-core": "^6.26.0",
|
"babel-core": "^6.26.0",
|
||||||
"babel-plugin-import": "^1.13.0",
|
"babel-plugin-import": "^1.13.0",
|
||||||
"babel-plugin-transform-runtime": "^6.23.0",
|
"babel-plugin-transform-runtime": "^6.23.0",
|
||||||
|
"babel-polyfill": "^6.26.0",
|
||||||
"babel-preset-es2015": "^6.24.1",
|
"babel-preset-es2015": "^6.24.1",
|
||||||
"babel-preset-react": "^6.24.1",
|
"babel-preset-react": "^6.24.1",
|
||||||
"babel-preset-stage-2": "^6.24.1",
|
"babel-preset-stage-2": "^6.24.1",
|
||||||
|
@ -203,6 +206,7 @@
|
||||||
"node-sass": "^4.12.0",
|
"node-sass": "^4.12.0",
|
||||||
"optimize-css-assets-webpack-plugin": "^5.0.3",
|
"optimize-css-assets-webpack-plugin": "^5.0.3",
|
||||||
"purgecss": "^2.1.2",
|
"purgecss": "^2.1.2",
|
||||||
|
"react-json-view": "^1.21.3",
|
||||||
"reqwest": "^2.0.5",
|
"reqwest": "^2.0.5",
|
||||||
"resize-observer-polyfill": "^1.5.1",
|
"resize-observer-polyfill": "^1.5.1",
|
||||||
"terser-webpack-plugin": "^2.3.5",
|
"terser-webpack-plugin": "^2.3.5",
|
||||||
|
|
|
@ -114,14 +114,6 @@ a:visited {
|
||||||
color: #898989;
|
color: #898989;
|
||||||
}
|
}
|
||||||
|
|
||||||
a:hover {
|
|
||||||
color: #FF7500;
|
|
||||||
}
|
|
||||||
|
|
||||||
a:hover.fa {
|
|
||||||
color: #FF7500;
|
|
||||||
}
|
|
||||||
|
|
||||||
input,
|
input,
|
||||||
textarea,
|
textarea,
|
||||||
select {
|
select {
|
||||||
|
|
|
@ -97,10 +97,6 @@ a:visited {
|
||||||
color: #05101a;
|
color: #05101a;
|
||||||
}
|
}
|
||||||
|
|
||||||
a:hover {
|
|
||||||
color: #459be5;
|
|
||||||
}
|
|
||||||
|
|
||||||
ol,
|
ol,
|
||||||
ul,
|
ul,
|
||||||
li {
|
li {
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
@charset "utf-8";
|
||||||
/* 头部 */
|
/* 头部 */
|
||||||
.header {
|
.header {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -1271,7 +1272,7 @@ html body {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 2.0;
|
line-height: 2.0;
|
||||||
background: #fafafa;
|
background: #fafafa;
|
||||||
font-family: "微软雅黑", "宋体";
|
font-family: "Microsoft YaHei", "SimSun";
|
||||||
color: #05101a;
|
color: #05101a;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
@ -1307,6 +1308,7 @@ td,
|
||||||
span {
|
span {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
margin-bottom: 0px!important;
|
||||||
}
|
}
|
||||||
|
|
||||||
table,
|
table,
|
||||||
|
@ -1363,10 +1365,6 @@ a:visited {
|
||||||
color: #05101a;
|
color: #05101a;
|
||||||
}
|
}
|
||||||
|
|
||||||
a:hover {
|
|
||||||
color: #459be5;
|
|
||||||
}
|
|
||||||
|
|
||||||
ol,
|
ol,
|
||||||
ul,
|
ul,
|
||||||
li {
|
li {
|
||||||
|
@ -1473,7 +1471,7 @@ a.edu-txt-w80,
|
||||||
|
|
||||||
/*隐藏*/
|
/*隐藏*/
|
||||||
.none {
|
.none {
|
||||||
display: none
|
display: none!important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.block {
|
.block {
|
||||||
|
@ -1522,7 +1520,15 @@ a.edu-txt-w80,
|
||||||
.font-16 {
|
.font-16 {
|
||||||
font-size: 16px !important;
|
font-size: 16px !important;
|
||||||
}
|
}
|
||||||
|
.weight400{
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
.weight500{
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
.weight{
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
.font-17 {
|
.font-17 {
|
||||||
font-size: 17px !important;
|
font-size: 17px !important;
|
||||||
}
|
}
|
||||||
|
@ -1542,6 +1548,9 @@ a.edu-txt-w80,
|
||||||
.font-25 {
|
.font-25 {
|
||||||
font-size: 25px !important;
|
font-size: 25px !important;
|
||||||
}
|
}
|
||||||
|
.font-26 {
|
||||||
|
font-size: 26px !important;
|
||||||
|
}
|
||||||
|
|
||||||
.font-24 {
|
.font-24 {
|
||||||
font-size: 24px !important;
|
font-size: 24px !important;
|
||||||
|
@ -1563,6 +1572,9 @@ a.edu-txt-w80,
|
||||||
font-size: 36px !important;
|
font-size: 36px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.font-40 {
|
||||||
|
font-size: 40px !important;
|
||||||
|
}
|
||||||
.font-50 {
|
.font-50 {
|
||||||
font-size: 50px !important;
|
font-size: 50px !important;
|
||||||
}
|
}
|
||||||
|
@ -1748,12 +1760,20 @@ a.decoration {
|
||||||
margin-bottom: 10px;
|
margin-bottom: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.mb12 {
|
||||||
|
margin-bottom: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mb13 {
|
||||||
|
margin-bottom: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
.mb14 {
|
.mb14 {
|
||||||
margin-bottom: 14px;
|
margin-bottom: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.mb15 {
|
.mb15 {
|
||||||
margin-bottom: 15px;
|
margin-bottom: 15px!important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.mb16 {
|
.mb16 {
|
||||||
|
@ -2424,13 +2444,23 @@ input::-ms-clear {
|
||||||
.color-grey-c {
|
.color-grey-c {
|
||||||
color: #ccc !important;
|
color: #ccc !important;
|
||||||
}
|
}
|
||||||
|
a.hoverLine:hover{
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
.color-grey-cd {
|
.color-grey-cd {
|
||||||
color: #cdcdcd !important;
|
color: #cdcdcd !important;
|
||||||
}
|
}
|
||||||
|
.color-grey-d {
|
||||||
|
color: #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
.color-grey-9 {
|
.color-grey-9 {
|
||||||
color: #999999 !important;
|
color: #999 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover{
|
||||||
|
color: #466AFF !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-grey-98 {
|
.color-grey-98 {
|
||||||
|
@ -2465,33 +2495,23 @@ input::-ms-clear {
|
||||||
a.color-grey-name:hover,
|
a.color-grey-name:hover,
|
||||||
a.color-dark:hover,
|
a.color-dark:hover,
|
||||||
a.color-grey-6:hover,
|
a.color-grey-6:hover,
|
||||||
a.color-grey-3:hover {
|
a.color-grey-3:hover,a.color-ooo:hover {
|
||||||
color: #4cacff !important;
|
color: #2A61FF !important;
|
||||||
}
|
|
||||||
|
|
||||||
a.color-grey-9:hover,
|
|
||||||
a.color-grey-8:hover,
|
|
||||||
a.color-grey-c:hover {
|
|
||||||
color: #111C24 !important;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*蓝色*/
|
/*蓝色*/
|
||||||
.color-blue {
|
.color-blue {
|
||||||
color: #4CACFF !important;
|
color: #2A61FF !important;
|
||||||
|
}
|
||||||
|
.color-blue-file {
|
||||||
|
color: #4598FA!important;
|
||||||
}
|
}
|
||||||
/* 绿色 */
|
/* 绿色 */
|
||||||
.color-green-file{
|
|
||||||
color: #28BD6C;
|
|
||||||
}
|
|
||||||
/*主*/
|
/*主*/
|
||||||
.color-blue_4C {
|
.color-blue_4C {
|
||||||
color: #4CACFF !important;
|
color: #4CACFF !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
a.color-blue:hover,
|
|
||||||
a.color-blue_4C:hover {
|
|
||||||
color: #459BE6 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*橙色*/
|
/*橙色*/
|
||||||
.color-orange {
|
.color-orange {
|
||||||
|
@ -3409,7 +3429,7 @@ a.user_bluebg_btn {
|
||||||
}
|
}
|
||||||
|
|
||||||
.cdefault {
|
.cdefault {
|
||||||
cursor: default
|
cursor: default!important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -3584,42 +3604,6 @@ a.user_bluebg_btn {
|
||||||
margin-right: 5px;
|
margin-right: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*-------------------个人主页:右侧提示区域--------------------------*/
|
|
||||||
.-task-sidebar {
|
|
||||||
position: fixed;
|
|
||||||
width: 40px;
|
|
||||||
height: 180px;
|
|
||||||
right: 0;
|
|
||||||
bottom: 80px;
|
|
||||||
z-index: 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
.-task-sidebar div {
|
|
||||||
height: 40px;
|
|
||||||
line-height: 40px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
width: 40px;
|
|
||||||
background: #4CACFF;
|
|
||||||
color: #fff;
|
|
||||||
font-size: 20px;
|
|
||||||
text-align: center;
|
|
||||||
margin-bottom: 5px;
|
|
||||||
border-radius: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.-task-sidebar div i {
|
|
||||||
color: #fff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.-task-sidebar div i:hover {
|
|
||||||
color: #fff !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.gotop {
|
|
||||||
background-color: rgba(208, 207, 207, 0.5) !important;
|
|
||||||
padding: 0px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/***** loading ******/
|
/***** loading ******/
|
||||||
/*****载入中******/
|
/*****载入中******/
|
||||||
|
@ -3944,11 +3928,21 @@ html>body #ajax-indicator {
|
||||||
max-height: 340px;
|
max-height: 340px;
|
||||||
}/*头部导航条样式---2018-03-19--by-cs*/
|
}/*头部导航条样式---2018-03-19--by-cs*/
|
||||||
|
|
||||||
|
.privateTag{
|
||||||
|
display: block;
|
||||||
|
padding:0px 6px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border:1px solid #2FC25B;
|
||||||
|
height: 18px;
|
||||||
|
line-height: 18px;
|
||||||
|
font-size: 12px;
|
||||||
|
margin-left: 10px;
|
||||||
|
color: #2FC25B;
|
||||||
|
}
|
||||||
.head-nav {
|
.head-nav {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
height: 70px;
|
height: 70px;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
min-width: 780px;
|
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
@ -6711,4 +6705,13 @@ p{
|
||||||
right: 0px;
|
right: 0px;
|
||||||
top:4px;
|
top:4px;
|
||||||
color: #999;
|
color: #999;
|
||||||
|
<<<<<<< HEAD
|
||||||
|
=======
|
||||||
|
}
|
||||||
|
.ant-input, .ant-input .ant-input-suffix{
|
||||||
|
background-color: #fff!important;
|
||||||
|
}
|
||||||
|
.has-error .ant-input{
|
||||||
|
background-color: #FEF1F0!important;
|
||||||
|
>>>>>>> pre_develop
|
||||||
}
|
}
|
|
@ -1,8 +1,8 @@
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "iconfont"; /* Project id 2340181 */
|
font-family: "iconfont"; /* Project id 2340181 */
|
||||||
src: url('iconfont.woff2?t=1622517296245') format('woff2'),
|
src: url('iconfont.woff2?t=1632964996877') format('woff2'),
|
||||||
url('iconfont.woff?t=1622517296245') format('woff'),
|
url('iconfont.woff?t=1632964996877') format('woff'),
|
||||||
url('iconfont.ttf?t=1622517296245') format('truetype');
|
url('iconfont.ttf?t=1632964996877') format('truetype');
|
||||||
}
|
}
|
||||||
|
|
||||||
.iconfont {
|
.iconfont {
|
||||||
|
@ -13,6 +13,462 @@
|
||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.icon-wenjian7:before {
|
||||||
|
content: "\e8e0";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiangyoujiantou:before {
|
||||||
|
content: "\e8de";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiangzuojiantou:before {
|
||||||
|
content: "\e8df";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-liulanicon2x:before {
|
||||||
|
content: "\e8dd";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wenjianicon:before {
|
||||||
|
content: "\e8dc";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-yuanquan2x:before {
|
||||||
|
content: "\e8db";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiangmubiaoqian:before {
|
||||||
|
content: "\e8da";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-icon:before {
|
||||||
|
content: "\e8ce";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-tar:before {
|
||||||
|
content: "\e8cf";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-fuzhi2:before {
|
||||||
|
content: "\e8d0";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-fujian1:before {
|
||||||
|
content: "\e8d1";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-bianji1:before {
|
||||||
|
content: "\e8d2";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-banbenicon:before {
|
||||||
|
content: "\e8d3";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-shanchuicon2:before {
|
||||||
|
content: "\e8d4";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-lajitong_icon3x:before {
|
||||||
|
content: "\e8d5";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xialaanniu2:before {
|
||||||
|
content: "\e8d6";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiazai-icon:before {
|
||||||
|
content: "\e8d7";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-master_icon1:before {
|
||||||
|
content: "\e8d8";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-shangchuanicon:before {
|
||||||
|
content: "\e8d9";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-gerenziliao1:before {
|
||||||
|
content: "\e8c7";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-lichengbeiicon:before {
|
||||||
|
content: "\e885";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-cangkushezhiicon:before {
|
||||||
|
content: "\e889";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-dongtaiicon:before {
|
||||||
|
content: "\e88a";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-gongzuoliuicon:before {
|
||||||
|
content: "\e88b";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-yixiuicon1:before {
|
||||||
|
content: "\e89b";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-wikiicon1:before {
|
||||||
|
content: "\e8c6";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-daimakuicon1:before {
|
||||||
|
content: "\e8c5";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wodetongzhi:before {
|
||||||
|
content: "\e8c8";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-tongzhiguanli:before {
|
||||||
|
content: "\e8c9";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xuanzhong3:before {
|
||||||
|
content: "\e8ca";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xitongtongzhiicon:before {
|
||||||
|
content: "\e8cb";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiaoxi2:before {
|
||||||
|
content: "\e8cc";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-sshmiyue:before {
|
||||||
|
content: "\e8cd";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-gerenziliao:before {
|
||||||
|
content: "\e8c4";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xinshouzhiyin:before {
|
||||||
|
content: "\e8e4";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xinjianxiangmu:before {
|
||||||
|
content: "\e8e6";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-jiaruketang1:before {
|
||||||
|
content: "\e8e9";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiangmugonggao:before {
|
||||||
|
content: "\e8c2";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-chengguo:before {
|
||||||
|
content: "\e8c3";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-chengjiaogonggao:before {
|
||||||
|
content: "\e8c0";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-jishuzichan:before {
|
||||||
|
content: "\e8c1";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-feibiaogonggao:before {
|
||||||
|
content: "\e8bc";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-zhongbiaogonggao:before {
|
||||||
|
content: "\e8bd";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-gengzhenggonggao:before {
|
||||||
|
content: "\e8be";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-zhaobiaogonggao:before {
|
||||||
|
content: "\e8bf";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wenjian6:before {
|
||||||
|
content: "\e8ba";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wenjianjia4:before {
|
||||||
|
content: "\e8bb";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-quxiaoguanzhu:before {
|
||||||
|
content: "\e89a";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-dianzan_icon:before {
|
||||||
|
content: "\e8a2";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wenjian5:before {
|
||||||
|
content: "\e896";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wenjianjia3:before {
|
||||||
|
content: "\e8a9";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-fuzhiicon:before {
|
||||||
|
content: "\e886";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-zhuye-fill:before {
|
||||||
|
content: "\e876";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-daimakuicon:before {
|
||||||
|
content: "\e884";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xinjian2:before {
|
||||||
|
content: "\e8b0";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xieyiicon:before {
|
||||||
|
content: "\e870";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-neicunicon:before {
|
||||||
|
content: "\e891";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-zishuwenjian_icon:before {
|
||||||
|
content: "\e8a6";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-biaoqianicon:before {
|
||||||
|
content: "\e882";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-bianji:before {
|
||||||
|
content: "\e883";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-lianjieicon:before {
|
||||||
|
content: "\e887";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-hebingqingqiuicon:before {
|
||||||
|
content: "\e888";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-morendianzan_icon:before {
|
||||||
|
content: "\e88e";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-muluicon:before {
|
||||||
|
content: "\e894";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-shezhi:before {
|
||||||
|
content: "\e899";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-tijiaoicon:before {
|
||||||
|
content: "\e89e";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-morenguanzhu_ICON:before {
|
||||||
|
content: "\e89f";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xialaanniu1:before {
|
||||||
|
content: "\e8a4";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-zuohuaicon:before {
|
||||||
|
content: "\e8b5";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-master_icon:before {
|
||||||
|
content: "\e8b6";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-morenfuke_icon:before {
|
||||||
|
content: "\e8b7";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-wikiicon:before {
|
||||||
|
content: "\e8b8";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-yixiuicon:before {
|
||||||
|
content: "\e8b9";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-suofang:before {
|
||||||
|
content: "\e87f";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-fangdaicon:before {
|
||||||
|
content: "\e881";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-sousuo_icon1:before {
|
||||||
|
content: "\e873";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-huanying_icon:before {
|
||||||
|
content: "\e878";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wenjianjia2:before {
|
||||||
|
content: "\e879";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-shanchuicon1:before {
|
||||||
|
content: "\e87a";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-qingqiuicon:before {
|
||||||
|
content: "\e871";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiangyingicon:before {
|
||||||
|
content: "\e87c";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-duoxuanxuanzhong:before {
|
||||||
|
content: "\e88f";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-cuowuicon:before {
|
||||||
|
content: "\e890";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-chenggongicon:before {
|
||||||
|
content: "\e892";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-weixuanzhongxiangyingicon:before {
|
||||||
|
content: "\e893";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-bitianicon:before {
|
||||||
|
content: "\e895";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-weixuanzhongxiangmubiaoqianicon:before {
|
||||||
|
content: "\e897";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-liebiaoicon:before {
|
||||||
|
content: "\e898";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xiezuozheguanliicon:before {
|
||||||
|
content: "\e8a1";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xuanzhongfenzhiicon:before {
|
||||||
|
content: "\e8a3";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xuanzhongjibenshezhiicon:before {
|
||||||
|
content: "\e8a5";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xuanzhongxiangmubiaoqianicon:before {
|
||||||
|
content: "\e8aa";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-a-xuanzhongwebhookicon:before {
|
||||||
|
content: "\e8af";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-shanchu_tc_icon:before {
|
||||||
|
content: "\e88c";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wiki_icon:before {
|
||||||
|
content: "\e88d";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-daorumoban_icon:before {
|
||||||
|
content: "\e86f";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-cuowu:before {
|
||||||
|
content: "\e872";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-gengduo_icon:before {
|
||||||
|
content: "\e874";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-fucengguanbi_icon:before {
|
||||||
|
content: "\e875";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-shanchuicon:before {
|
||||||
|
content: "\e877";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-sousuo_shanchuicon:before {
|
||||||
|
content: "\e87b";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-sousuo_icon:before {
|
||||||
|
content: "\e87d";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-wendangyulan_icon:before {
|
||||||
|
content: "\e87e";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xialaanniu:before {
|
||||||
|
content: "\e880";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-erciqueren_icon:before {
|
||||||
|
content: "\e867";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xuanzhongssh_icon:before {
|
||||||
|
content: "\e868";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-weixuanzhonganquanshezhi_icon:before {
|
||||||
|
content: "\e869";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-weixuanzhongssh_icon:before {
|
||||||
|
content: "\e86a";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-xuanzhonganquanshezhi_icon:before {
|
||||||
|
content: "\e86b";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-shanchu_icon:before {
|
||||||
|
content: "\e86c";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-liebiaossh_icon:before {
|
||||||
|
content: "\e86e";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-file-submodule:before {
|
||||||
|
content: "\e866";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-nv1:before {
|
||||||
|
content: "\e864";
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-nan1:before {
|
||||||
|
content: "\e865";
|
||||||
|
}
|
||||||
|
|
||||||
.icon-youxiang:before {
|
.icon-youxiang:before {
|
||||||
content: "\e8b2";
|
content: "\e8b2";
|
||||||
}
|
}
|
||||||
|
|
Binary file not shown.
File diff suppressed because one or more lines are too long
|
@ -5,6 +5,804 @@
|
||||||
"css_prefix_text": "icon-",
|
"css_prefix_text": "icon-",
|
||||||
"description": "",
|
"description": "",
|
||||||
"glyphs": [
|
"glyphs": [
|
||||||
|
{
|
||||||
|
"icon_id": "24656750",
|
||||||
|
"name": "文件",
|
||||||
|
"font_class": "wenjian7",
|
||||||
|
"unicode": "e8e0",
|
||||||
|
"unicode_decimal": 59616
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "630094",
|
||||||
|
"name": "向右箭头",
|
||||||
|
"font_class": "xiangyoujiantou",
|
||||||
|
"unicode": "e8de",
|
||||||
|
"unicode_decimal": 59614
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "630095",
|
||||||
|
"name": "向左箭头",
|
||||||
|
"font_class": "xiangzuojiantou",
|
||||||
|
"unicode": "e8df",
|
||||||
|
"unicode_decimal": 59615
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24600282",
|
||||||
|
"name": "浏览icon@2x",
|
||||||
|
"font_class": "a-liulanicon2x",
|
||||||
|
"unicode": "e8dd",
|
||||||
|
"unicode_decimal": 59613
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24567893",
|
||||||
|
"name": "文件icon",
|
||||||
|
"font_class": "wenjianicon",
|
||||||
|
"unicode": "e8dc",
|
||||||
|
"unicode_decimal": 59612
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24527422",
|
||||||
|
"name": "圆圈@2x",
|
||||||
|
"font_class": "a-yuanquan2x",
|
||||||
|
"unicode": "e8db",
|
||||||
|
"unicode_decimal": 59611
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24378423",
|
||||||
|
"name": "项目标签",
|
||||||
|
"font_class": "xiangmubiaoqian",
|
||||||
|
"unicode": "e8da",
|
||||||
|
"unicode_decimal": 59610
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24368060",
|
||||||
|
"name": "icon",
|
||||||
|
"font_class": "icon",
|
||||||
|
"unicode": "e8ce",
|
||||||
|
"unicode_decimal": 59598
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24368061",
|
||||||
|
"name": "tar",
|
||||||
|
"font_class": "tar",
|
||||||
|
"unicode": "e8cf",
|
||||||
|
"unicode_decimal": 59599
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289113",
|
||||||
|
"name": "复制 (2)",
|
||||||
|
"font_class": "a-fuzhi2",
|
||||||
|
"unicode": "e8d0",
|
||||||
|
"unicode_decimal": 59600
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289114",
|
||||||
|
"name": "附件",
|
||||||
|
"font_class": "fujian1",
|
||||||
|
"unicode": "e8d1",
|
||||||
|
"unicode_decimal": 59601
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289115",
|
||||||
|
"name": "编 辑",
|
||||||
|
"font_class": "a-bianji1",
|
||||||
|
"unicode": "e8d2",
|
||||||
|
"unicode_decimal": 59602
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289116",
|
||||||
|
"name": "版本icon",
|
||||||
|
"font_class": "banbenicon",
|
||||||
|
"unicode": "e8d3",
|
||||||
|
"unicode_decimal": 59603
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289117",
|
||||||
|
"name": "删除icon",
|
||||||
|
"font_class": "shanchuicon2",
|
||||||
|
"unicode": "e8d4",
|
||||||
|
"unicode_decimal": 59604
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289118",
|
||||||
|
"name": "垃圾桶_icon@3x",
|
||||||
|
"font_class": "a-lajitong_icon3x",
|
||||||
|
"unicode": "e8d5",
|
||||||
|
"unicode_decimal": 59605
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289119",
|
||||||
|
"name": "下拉按钮",
|
||||||
|
"font_class": "xialaanniu2",
|
||||||
|
"unicode": "e8d6",
|
||||||
|
"unicode_decimal": 59606
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289120",
|
||||||
|
"name": "下载-icon",
|
||||||
|
"font_class": "xiazai-icon",
|
||||||
|
"unicode": "e8d7",
|
||||||
|
"unicode_decimal": 59607
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289121",
|
||||||
|
"name": "master_icon",
|
||||||
|
"font_class": "master_icon1",
|
||||||
|
"unicode": "e8d8",
|
||||||
|
"unicode_decimal": 59608
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24289122",
|
||||||
|
"name": "上传icon",
|
||||||
|
"font_class": "shangchuanicon",
|
||||||
|
"unicode": "e8d9",
|
||||||
|
"unicode_decimal": 59609
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059956",
|
||||||
|
"name": "个人资料",
|
||||||
|
"font_class": "gerenziliao1",
|
||||||
|
"unicode": "e8c7",
|
||||||
|
"unicode_decimal": 59591
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059409",
|
||||||
|
"name": "里程碑icon",
|
||||||
|
"font_class": "lichengbeiicon",
|
||||||
|
"unicode": "e885",
|
||||||
|
"unicode_decimal": 59525
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059410",
|
||||||
|
"name": "仓库设置icon",
|
||||||
|
"font_class": "cangkushezhiicon",
|
||||||
|
"unicode": "e889",
|
||||||
|
"unicode_decimal": 59529
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059411",
|
||||||
|
"name": "动态icon",
|
||||||
|
"font_class": "dongtaiicon",
|
||||||
|
"unicode": "e88a",
|
||||||
|
"unicode_decimal": 59530
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059412",
|
||||||
|
"name": "工作流icon",
|
||||||
|
"font_class": "gongzuoliuicon",
|
||||||
|
"unicode": "e88b",
|
||||||
|
"unicode_decimal": 59531
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059413",
|
||||||
|
"name": "易修icon",
|
||||||
|
"font_class": "yixiuicon1",
|
||||||
|
"unicode": "e89b",
|
||||||
|
"unicode_decimal": 59547
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24059414",
|
||||||
|
"name": "wiki icon",
|
||||||
|
"font_class": "a-wikiicon1",
|
||||||
|
"unicode": "e8c6",
|
||||||
|
"unicode_decimal": 59590
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047186",
|
||||||
|
"name": "代码库icon",
|
||||||
|
"font_class": "daimakuicon1",
|
||||||
|
"unicode": "e8c5",
|
||||||
|
"unicode_decimal": 59589
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047189",
|
||||||
|
"name": "我的通知",
|
||||||
|
"font_class": "wodetongzhi",
|
||||||
|
"unicode": "e8c8",
|
||||||
|
"unicode_decimal": 59592
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047190",
|
||||||
|
"name": "通知管理",
|
||||||
|
"font_class": "tongzhiguanli",
|
||||||
|
"unicode": "e8c9",
|
||||||
|
"unicode_decimal": 59593
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047191",
|
||||||
|
"name": "选中",
|
||||||
|
"font_class": "xuanzhong3",
|
||||||
|
"unicode": "e8ca",
|
||||||
|
"unicode_decimal": 59594
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047192",
|
||||||
|
"name": "系统通知icon",
|
||||||
|
"font_class": "xitongtongzhiicon",
|
||||||
|
"unicode": "e8cb",
|
||||||
|
"unicode_decimal": 59595
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047193",
|
||||||
|
"name": "消息",
|
||||||
|
"font_class": "xiaoxi2",
|
||||||
|
"unicode": "e8cc",
|
||||||
|
"unicode_decimal": 59596
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24047194",
|
||||||
|
"name": "ssh密钥",
|
||||||
|
"font_class": "sshmiyue",
|
||||||
|
"unicode": "e8cd",
|
||||||
|
"unicode_decimal": 59597
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "24014152",
|
||||||
|
"name": "个人资料",
|
||||||
|
"font_class": "gerenziliao",
|
||||||
|
"unicode": "e8c4",
|
||||||
|
"unicode_decimal": 59588
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23655968",
|
||||||
|
"name": "新手指引",
|
||||||
|
"font_class": "xinshouzhiyin",
|
||||||
|
"unicode": "e8e4",
|
||||||
|
"unicode_decimal": 59620
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23655969",
|
||||||
|
"name": "新建项目",
|
||||||
|
"font_class": "xinjianxiangmu",
|
||||||
|
"unicode": "e8e6",
|
||||||
|
"unicode_decimal": 59622
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23658111",
|
||||||
|
"name": "加入课堂",
|
||||||
|
"font_class": "jiaruketang1",
|
||||||
|
"unicode": "e8e9",
|
||||||
|
"unicode_decimal": 59625
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23791639",
|
||||||
|
"name": "项目公告",
|
||||||
|
"font_class": "xiangmugonggao",
|
||||||
|
"unicode": "e8c2",
|
||||||
|
"unicode_decimal": 59586
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23791640",
|
||||||
|
"name": "成果",
|
||||||
|
"font_class": "chengguo",
|
||||||
|
"unicode": "e8c3",
|
||||||
|
"unicode_decimal": 59587
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23791410",
|
||||||
|
"name": "成交公告",
|
||||||
|
"font_class": "chengjiaogonggao",
|
||||||
|
"unicode": "e8c0",
|
||||||
|
"unicode_decimal": 59584
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23791411",
|
||||||
|
"name": "技术资产",
|
||||||
|
"font_class": "jishuzichan",
|
||||||
|
"unicode": "e8c1",
|
||||||
|
"unicode_decimal": 59585
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23790928",
|
||||||
|
"name": "废标公告",
|
||||||
|
"font_class": "feibiaogonggao",
|
||||||
|
"unicode": "e8bc",
|
||||||
|
"unicode_decimal": 59580
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23790929",
|
||||||
|
"name": "中标公告",
|
||||||
|
"font_class": "zhongbiaogonggao",
|
||||||
|
"unicode": "e8bd",
|
||||||
|
"unicode_decimal": 59581
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23790930",
|
||||||
|
"name": "更正公告",
|
||||||
|
"font_class": "gengzhenggonggao",
|
||||||
|
"unicode": "e8be",
|
||||||
|
"unicode_decimal": 59582
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23790931",
|
||||||
|
"name": "招标公告",
|
||||||
|
"font_class": "zhaobiaogonggao",
|
||||||
|
"unicode": "e8bf",
|
||||||
|
"unicode_decimal": 59583
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23732532",
|
||||||
|
"name": "文件",
|
||||||
|
"font_class": "wenjian6",
|
||||||
|
"unicode": "e8ba",
|
||||||
|
"unicode_decimal": 59578
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23732533",
|
||||||
|
"name": "文件夹",
|
||||||
|
"font_class": "wenjianjia4",
|
||||||
|
"unicode": "e8bb",
|
||||||
|
"unicode_decimal": 59579
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23642443",
|
||||||
|
"name": "取消关注",
|
||||||
|
"font_class": "quxiaoguanzhu",
|
||||||
|
"unicode": "e89a",
|
||||||
|
"unicode_decimal": 59546
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23642444",
|
||||||
|
"name": "点赞_icon",
|
||||||
|
"font_class": "dianzan_icon",
|
||||||
|
"unicode": "e8a2",
|
||||||
|
"unicode_decimal": 59554
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23639530",
|
||||||
|
"name": "文件",
|
||||||
|
"font_class": "wenjian5",
|
||||||
|
"unicode": "e896",
|
||||||
|
"unicode_decimal": 59542
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23639533",
|
||||||
|
"name": "文件夹",
|
||||||
|
"font_class": "wenjianjia3",
|
||||||
|
"unicode": "e8a9",
|
||||||
|
"unicode_decimal": 59561
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23639440",
|
||||||
|
"name": "复制icon",
|
||||||
|
"font_class": "fuzhiicon",
|
||||||
|
"unicode": "e886",
|
||||||
|
"unicode_decimal": 59526
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23639422",
|
||||||
|
"name": "主页-fill",
|
||||||
|
"font_class": "zhuye-fill",
|
||||||
|
"unicode": "e876",
|
||||||
|
"unicode_decimal": 59510
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23639423",
|
||||||
|
"name": "代码库icon",
|
||||||
|
"font_class": "daimakuicon",
|
||||||
|
"unicode": "e884",
|
||||||
|
"unicode_decimal": 59524
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23572260",
|
||||||
|
"name": "新建",
|
||||||
|
"font_class": "xinjian2",
|
||||||
|
"unicode": "e8b0",
|
||||||
|
"unicode_decimal": 59568
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23567674",
|
||||||
|
"name": "协议icon",
|
||||||
|
"font_class": "xieyiicon",
|
||||||
|
"unicode": "e870",
|
||||||
|
"unicode_decimal": 59504
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23567675",
|
||||||
|
"name": "内存icon",
|
||||||
|
"font_class": "neicunicon",
|
||||||
|
"unicode": "e891",
|
||||||
|
"unicode_decimal": 59537
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23567676",
|
||||||
|
"name": "自述文件_icon",
|
||||||
|
"font_class": "zishuwenjian_icon",
|
||||||
|
"unicode": "e8a6",
|
||||||
|
"unicode_decimal": 59558
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472253",
|
||||||
|
"name": "标签icon",
|
||||||
|
"font_class": "biaoqianicon",
|
||||||
|
"unicode": "e882",
|
||||||
|
"unicode_decimal": 59522
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472254",
|
||||||
|
"name": "编 辑",
|
||||||
|
"font_class": "a-bianji",
|
||||||
|
"unicode": "e883",
|
||||||
|
"unicode_decimal": 59523
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472258",
|
||||||
|
"name": "链接icon",
|
||||||
|
"font_class": "lianjieicon",
|
||||||
|
"unicode": "e887",
|
||||||
|
"unicode_decimal": 59527
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472259",
|
||||||
|
"name": "合并请求icon",
|
||||||
|
"font_class": "hebingqingqiuicon",
|
||||||
|
"unicode": "e888",
|
||||||
|
"unicode_decimal": 59528
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472263",
|
||||||
|
"name": "默认点赞_icon",
|
||||||
|
"font_class": "morendianzan_icon",
|
||||||
|
"unicode": "e88e",
|
||||||
|
"unicode_decimal": 59534
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472265",
|
||||||
|
"name": "目录icon",
|
||||||
|
"font_class": "muluicon",
|
||||||
|
"unicode": "e894",
|
||||||
|
"unicode_decimal": 59540
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472267",
|
||||||
|
"name": "设 置",
|
||||||
|
"font_class": "a-shezhi",
|
||||||
|
"unicode": "e899",
|
||||||
|
"unicode_decimal": 59545
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472269",
|
||||||
|
"name": "提交icon",
|
||||||
|
"font_class": "tijiaoicon",
|
||||||
|
"unicode": "e89e",
|
||||||
|
"unicode_decimal": 59550
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472270",
|
||||||
|
"name": "默认关注_ICON",
|
||||||
|
"font_class": "morenguanzhu_ICON",
|
||||||
|
"unicode": "e89f",
|
||||||
|
"unicode_decimal": 59551
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472272",
|
||||||
|
"name": "下拉按钮",
|
||||||
|
"font_class": "xialaanniu1",
|
||||||
|
"unicode": "e8a4",
|
||||||
|
"unicode_decimal": 59556
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472276",
|
||||||
|
"name": "左滑icon",
|
||||||
|
"font_class": "zuohuaicon",
|
||||||
|
"unicode": "e8b5",
|
||||||
|
"unicode_decimal": 59573
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472277",
|
||||||
|
"name": "master_icon",
|
||||||
|
"font_class": "master_icon",
|
||||||
|
"unicode": "e8b6",
|
||||||
|
"unicode_decimal": 59574
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472278",
|
||||||
|
"name": "默认复刻_icon",
|
||||||
|
"font_class": "morenfuke_icon",
|
||||||
|
"unicode": "e8b7",
|
||||||
|
"unicode_decimal": 59575
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472279",
|
||||||
|
"name": "wiki icon",
|
||||||
|
"font_class": "a-wikiicon",
|
||||||
|
"unicode": "e8b8",
|
||||||
|
"unicode_decimal": 59576
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23472280",
|
||||||
|
"name": "易修icon",
|
||||||
|
"font_class": "yixiuicon",
|
||||||
|
"unicode": "e8b9",
|
||||||
|
"unicode_decimal": 59577
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23436350",
|
||||||
|
"name": "缩放",
|
||||||
|
"font_class": "suofang",
|
||||||
|
"unicode": "e87f",
|
||||||
|
"unicode_decimal": 59519
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23436351",
|
||||||
|
"name": "放大icon",
|
||||||
|
"font_class": "fangdaicon",
|
||||||
|
"unicode": "e881",
|
||||||
|
"unicode_decimal": 59521
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23384231",
|
||||||
|
"name": "搜索_icon",
|
||||||
|
"font_class": "sousuo_icon1",
|
||||||
|
"unicode": "e873",
|
||||||
|
"unicode_decimal": 59507
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23384232",
|
||||||
|
"name": "欢迎_icon",
|
||||||
|
"font_class": "huanying_icon",
|
||||||
|
"unicode": "e878",
|
||||||
|
"unicode_decimal": 59512
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23384233",
|
||||||
|
"name": "文件夹",
|
||||||
|
"font_class": "wenjianjia2",
|
||||||
|
"unicode": "e879",
|
||||||
|
"unicode_decimal": 59513
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23384234",
|
||||||
|
"name": "删除icon",
|
||||||
|
"font_class": "shanchuicon1",
|
||||||
|
"unicode": "e87a",
|
||||||
|
"unicode_decimal": 59514
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23261798",
|
||||||
|
"name": "请求icon",
|
||||||
|
"font_class": "qingqiuicon",
|
||||||
|
"unicode": "e871",
|
||||||
|
"unicode_decimal": 59505
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23261799",
|
||||||
|
"name": "响应icon",
|
||||||
|
"font_class": "xiangyingicon",
|
||||||
|
"unicode": "e87c",
|
||||||
|
"unicode_decimal": 59516
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144143",
|
||||||
|
"name": "多选选中",
|
||||||
|
"font_class": "duoxuanxuanzhong",
|
||||||
|
"unicode": "e88f",
|
||||||
|
"unicode_decimal": 59535
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144144",
|
||||||
|
"name": "错误icon",
|
||||||
|
"font_class": "cuowuicon",
|
||||||
|
"unicode": "e890",
|
||||||
|
"unicode_decimal": 59536
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144146",
|
||||||
|
"name": "成功icon",
|
||||||
|
"font_class": "chenggongicon",
|
||||||
|
"unicode": "e892",
|
||||||
|
"unicode_decimal": 59538
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144147",
|
||||||
|
"name": "未选中响应icon",
|
||||||
|
"font_class": "weixuanzhongxiangyingicon",
|
||||||
|
"unicode": "e893",
|
||||||
|
"unicode_decimal": 59539
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144149",
|
||||||
|
"name": "必填icon",
|
||||||
|
"font_class": "bitianicon",
|
||||||
|
"unicode": "e895",
|
||||||
|
"unicode_decimal": 59541
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144151",
|
||||||
|
"name": "未选中项目标签icon",
|
||||||
|
"font_class": "weixuanzhongxiangmubiaoqianicon",
|
||||||
|
"unicode": "e897",
|
||||||
|
"unicode_decimal": 59543
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144152",
|
||||||
|
"name": "列表icon",
|
||||||
|
"font_class": "liebiaoicon",
|
||||||
|
"unicode": "e898",
|
||||||
|
"unicode_decimal": 59544
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144158",
|
||||||
|
"name": "协作者管理icon",
|
||||||
|
"font_class": "xiezuozheguanliicon",
|
||||||
|
"unicode": "e8a1",
|
||||||
|
"unicode_decimal": 59553
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144160",
|
||||||
|
"name": "选中分支icon",
|
||||||
|
"font_class": "xuanzhongfenzhiicon",
|
||||||
|
"unicode": "e8a3",
|
||||||
|
"unicode_decimal": 59555
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144162",
|
||||||
|
"name": "选中基本设置icon",
|
||||||
|
"font_class": "xuanzhongjibenshezhiicon",
|
||||||
|
"unicode": "e8a5",
|
||||||
|
"unicode_decimal": 59557
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144165",
|
||||||
|
"name": "选中项目标签icon",
|
||||||
|
"font_class": "xuanzhongxiangmubiaoqianicon",
|
||||||
|
"unicode": "e8aa",
|
||||||
|
"unicode_decimal": 59562
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23144167",
|
||||||
|
"name": "选中webhook icon",
|
||||||
|
"font_class": "a-xuanzhongwebhookicon",
|
||||||
|
"unicode": "e8af",
|
||||||
|
"unicode_decimal": 59567
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046290",
|
||||||
|
"name": "shanchu_tc_icon",
|
||||||
|
"font_class": "shanchu_tc_icon",
|
||||||
|
"unicode": "e88c",
|
||||||
|
"unicode_decimal": 59532
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046293",
|
||||||
|
"name": "wiki_icon",
|
||||||
|
"font_class": "wiki_icon",
|
||||||
|
"unicode": "e88d",
|
||||||
|
"unicode_decimal": 59533
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046244",
|
||||||
|
"name": "导入模版_icon",
|
||||||
|
"font_class": "daorumoban_icon",
|
||||||
|
"unicode": "e86f",
|
||||||
|
"unicode_decimal": 59503
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046252",
|
||||||
|
"name": "错误",
|
||||||
|
"font_class": "cuowu",
|
||||||
|
"unicode": "e872",
|
||||||
|
"unicode_decimal": 59506
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046255",
|
||||||
|
"name": "更多_icon",
|
||||||
|
"font_class": "gengduo_icon",
|
||||||
|
"unicode": "e874",
|
||||||
|
"unicode_decimal": 59508
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046258",
|
||||||
|
"name": "复层关闭_icon",
|
||||||
|
"font_class": "fucengguanbi_icon",
|
||||||
|
"unicode": "e875",
|
||||||
|
"unicode_decimal": 59509
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046268",
|
||||||
|
"name": "删除icon",
|
||||||
|
"font_class": "shanchuicon",
|
||||||
|
"unicode": "e877",
|
||||||
|
"unicode_decimal": 59511
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046273",
|
||||||
|
"name": "搜索_删除icon",
|
||||||
|
"font_class": "sousuo_shanchuicon",
|
||||||
|
"unicode": "e87b",
|
||||||
|
"unicode_decimal": 59515
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046275",
|
||||||
|
"name": "搜索_icon",
|
||||||
|
"font_class": "sousuo_icon",
|
||||||
|
"unicode": "e87d",
|
||||||
|
"unicode_decimal": 59517
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046276",
|
||||||
|
"name": "文档预览_icon",
|
||||||
|
"font_class": "wendangyulan_icon",
|
||||||
|
"unicode": "e87e",
|
||||||
|
"unicode_decimal": 59518
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "23046278",
|
||||||
|
"name": "下拉按钮",
|
||||||
|
"font_class": "xialaanniu",
|
||||||
|
"unicode": "e880",
|
||||||
|
"unicode_decimal": 59520
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906287",
|
||||||
|
"name": "二次确认_icon",
|
||||||
|
"font_class": "erciqueren_icon",
|
||||||
|
"unicode": "e867",
|
||||||
|
"unicode_decimal": 59495
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906288",
|
||||||
|
"name": "选中ssh_icon",
|
||||||
|
"font_class": "xuanzhongssh_icon",
|
||||||
|
"unicode": "e868",
|
||||||
|
"unicode_decimal": 59496
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906289",
|
||||||
|
"name": "未选中安全设置_icon",
|
||||||
|
"font_class": "weixuanzhonganquanshezhi_icon",
|
||||||
|
"unicode": "e869",
|
||||||
|
"unicode_decimal": 59497
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906290",
|
||||||
|
"name": "未选中ssh_icon",
|
||||||
|
"font_class": "weixuanzhongssh_icon",
|
||||||
|
"unicode": "e86a",
|
||||||
|
"unicode_decimal": 59498
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906291",
|
||||||
|
"name": "选中安全设置_icon",
|
||||||
|
"font_class": "xuanzhonganquanshezhi_icon",
|
||||||
|
"unicode": "e86b",
|
||||||
|
"unicode_decimal": 59499
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906292",
|
||||||
|
"name": "删除_icon",
|
||||||
|
"font_class": "shanchu_icon",
|
||||||
|
"unicode": "e86c",
|
||||||
|
"unicode_decimal": 59500
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "22906293",
|
||||||
|
"name": "列表ssh_icon",
|
||||||
|
"font_class": "liebiaossh_icon",
|
||||||
|
"unicode": "e86e",
|
||||||
|
"unicode_decimal": 59502
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "17575494",
|
||||||
|
"name": "file-submodule",
|
||||||
|
"font_class": "file-submodule",
|
||||||
|
"unicode": "e866",
|
||||||
|
"unicode_decimal": 59494
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "7539612",
|
||||||
|
"name": "nv",
|
||||||
|
"font_class": "nv1",
|
||||||
|
"unicode": "e864",
|
||||||
|
"unicode_decimal": 59492
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"icon_id": "7539613",
|
||||||
|
"name": "nan",
|
||||||
|
"font_class": "nan1",
|
||||||
|
"unicode": "e865",
|
||||||
|
"unicode_decimal": 59493
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"icon_id": "21936935",
|
"icon_id": "21936935",
|
||||||
"name": "邮箱",
|
"name": "邮箱",
|
||||||
|
|
File diff suppressed because one or more lines are too long
Before Width: | Height: | Size: 733 KiB |
|
@ -1,5 +1,5 @@
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="zh-CN" class="notranslate translated-ltr" translate="no">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta name=”Keywords” Content=”trustie,trustieforge,forge,确实让创建更美好,协同开发平台″>
|
<meta name=”Keywords” Content=”trustie,trustieforge,forge,确实让创建更美好,协同开发平台″>
|
||||||
|
|
|
@ -3319,9 +3319,9 @@
|
||||||
text = text.replace(emailReg, function ($1, $2, $3, $4) {
|
text = text.replace(emailReg, function ($1, $2, $3, $4) {
|
||||||
return $1.replace(/@/g, "_#_@_#_");
|
return $1.replace(/@/g, "_#_@_#_");
|
||||||
});
|
});
|
||||||
|
// " + editormd.urls.atLinkBase + "" + $2 + "
|
||||||
text = text.replace(atLinkReg, function ($1, $2) {
|
text = text.replace(atLinkReg, function ($1, $2) {
|
||||||
return "<a href=\"" + editormd.urls.atLinkBase + "" + $2 + "\" title=\"@" + $2 + "\" class=\"at-link\">" + $1 + "</a>";
|
return "<span title=\"@" + $2 + "\" class=\"at-link\"> " + $1 + " </span>";
|
||||||
}).replace(/_#_@_#_/g, "@");
|
}).replace(/_#_@_#_/g, "@");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,12 +60,13 @@ body {
|
||||||
.ant-progress-textno {
|
.ant-progress-textno {
|
||||||
color: #f5222d;
|
color: #f5222d;
|
||||||
}
|
}
|
||||||
|
.CodeMirror pre.CodeMirror-line{
|
||||||
|
font-size: 16px!important;
|
||||||
|
}
|
||||||
/* md多空格 */
|
/* md多空格 */
|
||||||
.markdown-body p {
|
.markdown-body p {
|
||||||
margin:10px 0px!important;
|
margin:10px 0px!important;
|
||||||
font-size: 16px !important;
|
font-size: 16px !important;
|
||||||
line-height: 2 !important;
|
|
||||||
white-space: pre-wrap;
|
white-space: pre-wrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -87,6 +88,10 @@ body {
|
||||||
border-left: 1px solid rgb(221, 221, 221);
|
border-left: 1px solid rgb(221, 221, 221);
|
||||||
/* 某些情况下,被cm盖住了 */
|
/* 某些情况下,被cm盖住了 */
|
||||||
z-index: 99;
|
z-index: 99;
|
||||||
|
padding:8px 8px 50px;
|
||||||
|
}
|
||||||
|
.editormd-preview .markdown-body{
|
||||||
|
padding:0px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 图片点击放大的场景,隐藏图片链接 */
|
/* 图片点击放大的场景,隐藏图片链接 */
|
||||||
|
|
298
src/App.js
298
src/App.js
|
@ -3,7 +3,7 @@ import './App.css';
|
||||||
import { ConfigProvider } from 'antd'
|
import { ConfigProvider } from 'antd'
|
||||||
import zhCN from 'antd/lib/locale-provider/zh_CN';
|
import zhCN from 'antd/lib/locale-provider/zh_CN';
|
||||||
import {
|
import {
|
||||||
BrowserRouter as Router,
|
// BrowserRouter as Router,
|
||||||
Route,
|
Route,
|
||||||
Switch
|
Switch
|
||||||
} from 'react-router-dom';
|
} from 'react-router-dom';
|
||||||
|
@ -17,10 +17,9 @@ import marked from './common/marked';
|
||||||
import moment from 'moment'
|
import moment from 'moment'
|
||||||
|
|
||||||
import { MuiThemeProvider, createMuiTheme } from 'material-ui/styles';
|
import { MuiThemeProvider, createMuiTheme } from 'material-ui/styles';
|
||||||
|
import SiderBar from './forge/Component/SiderBar'
|
||||||
|
|
||||||
import history from './history';
|
import { SnackbarHOC } from 'educoder';
|
||||||
|
|
||||||
import { SnackbarHOC } from 'educoder'
|
|
||||||
import { initAxiosInterceptors } from './AppConfig'
|
import { initAxiosInterceptors } from './AppConfig'
|
||||||
import { Provider } from 'react-redux';
|
import { Provider } from 'react-redux';
|
||||||
import configureStore from './redux/stores/configureStore';
|
import configureStore from './redux/stores/configureStore';
|
||||||
|
@ -40,6 +39,16 @@ const Projects = Loadable({
|
||||||
loader: () => import('./forge/Index'),
|
loader: () => import('./forge/Index'),
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
})
|
})
|
||||||
|
// forge项目详情
|
||||||
|
const ProjectDetail = Loadable({
|
||||||
|
loader: () => import("./forge/Main/DetailAdaptor"),
|
||||||
|
loading: Loading,
|
||||||
|
});
|
||||||
|
//forge安全设置
|
||||||
|
const Security = Loadable({
|
||||||
|
loader: () => import('./forge/SecuritySetting/Index'),
|
||||||
|
loading: Loading,
|
||||||
|
})
|
||||||
//forge项目-devOps详情
|
//forge项目-devOps详情
|
||||||
const OpsDetail = Loadable({
|
const OpsDetail = Loadable({
|
||||||
loader: () => import('./forge/DevOps/opsDetail'),
|
loader: () => import('./forge/DevOps/opsDetail'),
|
||||||
|
@ -74,6 +83,27 @@ const EducoderLogin = Loadable({
|
||||||
loader: () => import('./modules/login/EducoderLogin'),
|
loader: () => import('./modules/login/EducoderLogin'),
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
})
|
})
|
||||||
|
const Search = Loadable({
|
||||||
|
loader: () => import('./modules/search/'),
|
||||||
|
loading: Loading,
|
||||||
|
})
|
||||||
|
const WikiPreview = Loadable({
|
||||||
|
loader: () => import('./forge/Wiki/Preview'),
|
||||||
|
loading: Loading,
|
||||||
|
})
|
||||||
|
|
||||||
|
const ProjectIndex = Loadable({
|
||||||
|
loader: () => import("./forge/Index"),
|
||||||
|
loading: Loading,
|
||||||
|
});
|
||||||
|
|
||||||
|
// const CreateMerge = Loadable({
|
||||||
|
// loader: () => import('./forge/Merge/NewMerge'),
|
||||||
|
// loading: Loading,
|
||||||
|
// })
|
||||||
|
|
||||||
|
// 此处仅维护前端可能的一级路由,不用进行项目或者组织判断的字段。
|
||||||
|
const keyWord = ["explore", "settings", "setting", "mulan", "wiki", "issues", "setting", "trending", "code", "projects", "pulls", "mine", "login", "register", "email", "export", "nopage", "404", "403", "500", "501", "search", "organize"];
|
||||||
|
|
||||||
class App extends Component {
|
class App extends Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
|
@ -84,6 +114,51 @@ class App extends Component {
|
||||||
mydisplay: false,
|
mydisplay: false,
|
||||||
occupation: 0,
|
occupation: 0,
|
||||||
mygetHelmetapi: null,
|
mygetHelmetapi: null,
|
||||||
|
pathType: null,
|
||||||
|
pathName: null,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
UNSAFE_componentWillMount() {
|
||||||
|
initAxiosInterceptors(this.props);
|
||||||
|
let pathname = window.location.pathname ? window.location.pathname.split('/')[1] : '';
|
||||||
|
pathname && this.getPathnameType(pathname);
|
||||||
|
|
||||||
|
// 添加路由监听,决定组织还是个人
|
||||||
|
this.unlisten = this.props.history.listen((location) => {
|
||||||
|
let newPathname = location.pathname.split('/')[1];
|
||||||
|
if (this.state.pathName !== newPathname) {
|
||||||
|
// this.setState({ pathType: '' });
|
||||||
|
newPathname && this.getPathnameType(newPathname);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
shouldComponentUpdate(nextProps, nextState) {
|
||||||
|
// (!keyWord.includes(this.props.location.pathname.split('/')[1])) &&
|
||||||
|
if (nextProps.location.pathname.split('/')[1] !== this.props.location.pathname.split('/')[1] && nextState.pathType === this.state.pathType) {
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getPathnameType = (pathname) => {
|
||||||
|
if (!keyWord.includes(pathname)) {
|
||||||
|
let url = `/owners/${pathname}.json`;
|
||||||
|
axios.get(url).then((response) => {
|
||||||
|
if (response && response.status === 200) {
|
||||||
|
this.setState({
|
||||||
|
pathType: response.data.type || '404',
|
||||||
|
pathName: pathname,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}else{
|
||||||
|
this.setState({
|
||||||
|
pathType: pathname,
|
||||||
|
pathName: pathname,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,43 +175,22 @@ class App extends Component {
|
||||||
Addcoursestypes: false
|
Addcoursestypes: false
|
||||||
})
|
})
|
||||||
};
|
};
|
||||||
ModalCancelsy = () => {
|
|
||||||
this.setState({
|
|
||||||
mydisplay: false,
|
|
||||||
})
|
|
||||||
window.location.href = "/";
|
|
||||||
};
|
|
||||||
ModalshowCancelsy = () => {
|
|
||||||
this.setState({
|
|
||||||
mydisplay: true,
|
|
||||||
})
|
|
||||||
};
|
|
||||||
|
|
||||||
disableVideoContextMenu = () => {
|
|
||||||
window.$("body").on("mousedown", "video", function (event) {
|
|
||||||
if (event.which === 3) {
|
|
||||||
window.$('video').bind('contextmenu', function () { return false; });
|
|
||||||
} else {
|
|
||||||
window.$('video').unbind('contextmenu');
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
document.title = "loading...";
|
document.title = "loading...";
|
||||||
this.disableVideoContextMenu();
|
|
||||||
history.listen(() => {
|
|
||||||
this.forceUpdate()
|
|
||||||
const $ = window.$
|
|
||||||
$("html").animate({ scrollTop: $('html').scrollTop() - 0 })
|
|
||||||
});
|
|
||||||
|
|
||||||
initAxiosInterceptors(this.props);
|
|
||||||
this.getAppdata();
|
this.getAppdata();
|
||||||
|
|
||||||
window.addEventListener('error', (event) => {
|
window.addEventListener('error', (event) => {
|
||||||
const msg = `${event.type}: ${event.message}`;
|
const msg = `${event.type}: ${event.message}`;
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
this.unlisten && this.unlisten(); // 执行解绑
|
||||||
|
}
|
||||||
|
|
||||||
//修改登录方法
|
//修改登录方法
|
||||||
Modifyloginvalue = () => {
|
Modifyloginvalue = () => {
|
||||||
this.setState({
|
this.setState({
|
||||||
|
@ -207,69 +261,139 @@ class App extends Component {
|
||||||
};
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
const { mygetHelmetapi, pathType} = this.state;
|
||||||
|
let personal = mygetHelmetapi && mygetHelmetapi.personal;
|
||||||
return (
|
return (
|
||||||
<Provider store={store}>
|
<Provider store={store}>
|
||||||
<ConfigProvider locale={zhCN}>
|
<ConfigProvider locale={zhCN}>
|
||||||
<MuiThemeProvider theme={theme}>
|
<MuiThemeProvider theme={theme}>
|
||||||
<LoginDialog {...this.props} {...this.state} Modifyloginvalue={() => this.Modifyloginvalue()}></LoginDialog>
|
<LoginDialog {...this.props} {...this.state} Modifyloginvalue={() => this.Modifyloginvalue()}></LoginDialog>
|
||||||
<Router>
|
<SiderBar />
|
||||||
<Switch>
|
{/* <Router> */}
|
||||||
{/*项目*/}
|
<Switch>
|
||||||
<Route
|
{/* wiki预览 */}
|
||||||
path={"/projects/:owner/:projectId/devops/:opsId/detail"}
|
<Route path="/:owner/:projectsId/wiki/preview/:projectName/:projectId" render={
|
||||||
render={
|
(props) => {
|
||||||
(props) => {
|
return (<WikiPreview {...this.props} {...props} {...this.state} />)
|
||||||
return (<OpsDetail {...this.props} {...props} {...this.state} />)
|
}
|
||||||
}
|
} />
|
||||||
}>
|
|
||||||
</Route>
|
|
||||||
{/*项目*/}
|
|
||||||
<Route
|
|
||||||
path={"/projects"}
|
|
||||||
render={
|
|
||||||
(props) => {
|
|
||||||
return (<Projects {...this.props} {...props} {...this.state} />)
|
|
||||||
}
|
|
||||||
}>
|
|
||||||
</Route>
|
|
||||||
<Route
|
|
||||||
path="/register"
|
|
||||||
render={
|
|
||||||
(props) => {
|
|
||||||
return (<EducoderLogin {...this.props} {...props} {...this.state} />)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
{/*403*/}
|
|
||||||
<Route path="/403" component={Shixunauthority} />
|
|
||||||
|
|
||||||
<Route path="/500" component={http500} />
|
{/* 项目PR */}
|
||||||
<Route path={"/organize"}
|
<Route path="/:owner/:projectsId/compare"
|
||||||
render={
|
render={
|
||||||
(props) => {
|
(props) => (<ProjectDetail {...this.props} {...props} {...this.state} />)
|
||||||
return (<OrganizeIndex {...props} {...this.props} {...this.state} />)
|
}
|
||||||
}
|
></Route>
|
||||||
}>
|
|
||||||
</Route>
|
{/*项目*/}
|
||||||
{/*404*/}
|
<Route
|
||||||
<Route path="/nopage" component={Shixunnopage} />
|
path={"/:owner/:projectId/devops/:opsId/detail"}
|
||||||
{/* 个人主页 */}
|
render={
|
||||||
<Route path="/users/:username"
|
(props) => {
|
||||||
render={
|
return (<OpsDetail {...this.props} {...props} {...this.state} />)
|
||||||
(props) => {
|
|
||||||
return (<InfosIndex {...this.props} {...this.state} />)
|
|
||||||
}
|
|
||||||
}></Route>
|
|
||||||
<Route exact path="/"
|
|
||||||
render={
|
|
||||||
(props) => (
|
|
||||||
<Projects {...this.props} {...props} {...this.state}></Projects>
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
/>
|
}>
|
||||||
<Route component={Shixunnopage} />
|
</Route>
|
||||||
</Switch>
|
<Route
|
||||||
</Router>
|
path={"/settings"}
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<Security {...this.props} {...props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
</Route>
|
||||||
|
|
||||||
|
<Route
|
||||||
|
path="/register"
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<EducoderLogin {...this.props} {...props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
{/*403*/}
|
||||||
|
<Route path="/403" component={Shixunauthority} />
|
||||||
|
|
||||||
|
<Route path="/500" component={http500} />
|
||||||
|
|
||||||
|
{/*404*/}
|
||||||
|
<Route path="/nopage" component={Shixunnopage} />
|
||||||
|
|
||||||
|
{/* 查询 */}
|
||||||
|
<Route path="/search" component={Search} />
|
||||||
|
|
||||||
|
<Route exact path="/explore"
|
||||||
|
render={
|
||||||
|
(props) => (
|
||||||
|
<ProjectIndex {...this.props} {...props} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
|
||||||
|
|
||||||
|
{/* 组织 */}
|
||||||
|
<Route path={"/organize"}
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<OrganizeIndex {...props} {...this.props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
</Route>
|
||||||
|
|
||||||
|
{/*新建项目等*/}
|
||||||
|
<Route
|
||||||
|
path={"/projects"}
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<Projects {...this.props} {...props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
</Route>
|
||||||
|
|
||||||
|
{/* 判断为用户/组织,并进入对应页面 */}
|
||||||
|
{
|
||||||
|
pathType === 'User' ?
|
||||||
|
<Route exact path="/:username"
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<InfosIndex {...this.props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/> : pathType === 'Organization' ? <Route path={"/:OIdentifier"}
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<OrganizeIndex {...props} {...this.props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
</Route> : pathType === '404' ? <Route component={Shixunnopage} />:
|
||||||
|
<Route exact path="/"
|
||||||
|
render={
|
||||||
|
(props) => (
|
||||||
|
personal && personal.length > 0 ?
|
||||||
|
<InfosIndex {...this.props} {...props} />
|
||||||
|
:
|
||||||
|
<ProjectIndex {...this.props} {...props} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
// <Route path="/" component={Loading} />
|
||||||
|
// <Route path="/" component={Shixunnopage} />
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
{/* 个人主页 */}
|
||||||
|
<Route path="/:username"
|
||||||
|
render={
|
||||||
|
(props) => {
|
||||||
|
return (<InfosIndex {...this.props} {...this.state} />)
|
||||||
|
}
|
||||||
|
}></Route>
|
||||||
|
|
||||||
|
|
||||||
|
<Route component={Shixunnopage} />
|
||||||
|
</Switch>
|
||||||
|
{/* </Router> */}
|
||||||
</MuiThemeProvider>
|
</MuiThemeProvider>
|
||||||
</ConfigProvider>
|
</ConfigProvider>
|
||||||
</Provider>
|
</Provider>
|
||||||
|
|
|
@ -85,7 +85,10 @@ export function initAxiosInterceptors(props) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.data.status === 404) {
|
if (response.data.status === 404) {
|
||||||
locationurl('/nopage');
|
let responseURL = response.request ? response.request.responseURL:'';
|
||||||
|
if (responseURL.indexOf('/api/users/') === -1 && responseURL.indexOf('/api/organizations/') === -1 ) {
|
||||||
|
locationurl('/nopage');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.data.status === 500) {
|
if (response.data.status === 500) {
|
||||||
|
|
|
@ -9,20 +9,7 @@ class Loading extends Component {
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
return (
|
return ""
|
||||||
<div className="App" style={{ minHeight: '800px', width: "100%" }}>
|
|
||||||
<style>
|
|
||||||
{
|
|
||||||
`
|
|
||||||
.margintop{
|
|
||||||
margin-top:20%;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
<Spin size="large" className={"margintop"} />
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -195,7 +195,7 @@ class College extends Component {
|
||||||
align: 'center',
|
align: 'center',
|
||||||
className: "edu-txt-center font-14 maxnamewidth105",
|
className: "edu-txt-center font-14 maxnamewidth105",
|
||||||
render: (text, record) => (
|
render: (text, record) => (
|
||||||
<a href={`/users/${record.login}`} title={record.name} target="_blank" className="task-hide maxnamewidth105" style={{
|
<a href={`/${record.login}`} title={record.name} target="_blank" className="task-hide maxnamewidth105" style={{
|
||||||
color:'#007bff',
|
color:'#007bff',
|
||||||
|
|
||||||
}}> {
|
}}> {
|
||||||
|
|
|
@ -218,7 +218,7 @@ a:hover {
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-blue {
|
.color-blue {
|
||||||
color: #4CACFF;
|
color: #2A61FF;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-huang {
|
.color-huang {
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
|
import { number } from "prop-types";
|
||||||
|
|
||||||
// 处理整点 半点
|
// 处理整点 半点
|
||||||
// 取传入时间往后的第一个半点
|
// 取传入时间往后的第一个半点
|
||||||
|
@ -97,3 +98,41 @@ export function formatDuring(mss){
|
||||||
}
|
}
|
||||||
return days + "天" + hours + "小时" + minutes + "分";
|
return days + "天" + hours + "小时" + minutes + "分";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
返回:多久以前
|
||||||
|
backDate:以前的某个日期
|
||||||
|
*/
|
||||||
|
export function timeAgo(backDate) {
|
||||||
|
try {
|
||||||
|
moment(backDate);
|
||||||
|
} catch (e) {
|
||||||
|
return '刚刚';
|
||||||
|
}
|
||||||
|
if(typeof backDate ==='number'){
|
||||||
|
backDate=backDate*1000
|
||||||
|
}else{
|
||||||
|
backDate= moment(backDate);
|
||||||
|
}
|
||||||
|
let time = new Date() - backDate;
|
||||||
|
var days = Math.floor(time / (1000 * 60 * 60 * 24));
|
||||||
|
var hours = Math.floor((time % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
|
||||||
|
var minutes = Math.floor((time % (1000 * 60 * 60)) / (1000 * 60));
|
||||||
|
var seconds = Math.floor((time % (1000 * 60 * 60)) / 1000);
|
||||||
|
if (time <= 0) {
|
||||||
|
return "刚刚";
|
||||||
|
}
|
||||||
|
if (days) {
|
||||||
|
return days + "天前";
|
||||||
|
}
|
||||||
|
if (hours) {
|
||||||
|
return hours + "小时前";
|
||||||
|
}
|
||||||
|
if (minutes) {
|
||||||
|
return minutes + "分前";
|
||||||
|
}
|
||||||
|
if (seconds) {
|
||||||
|
return seconds + "秒前";
|
||||||
|
}
|
||||||
|
return "刚刚";
|
||||||
|
}
|
|
@ -69,7 +69,7 @@ export function appendFileSizeToUploadFile(item) {
|
||||||
}
|
}
|
||||||
export function appendFileSizeToUploadFileAll(fileList) {
|
export function appendFileSizeToUploadFileAll(fileList) {
|
||||||
return fileList.map(item => {
|
return fileList.map(item => {
|
||||||
if (item.name.indexOf(uploadNameSizeSeperator) == -1) {
|
if (item.name.indexOf(uploadNameSizeSeperator) === -1) {
|
||||||
return Object.assign({}, item, { name: `${item.name}${uploadNameSizeSeperator}${bytesToSize(item.size)}` })
|
return Object.assign({}, item, { name: `${item.name}${uploadNameSizeSeperator}${bytesToSize(item.size)}` })
|
||||||
}
|
}
|
||||||
return item
|
return item
|
||||||
|
|
|
@ -18,6 +18,23 @@ export function getImageUrl(path) {
|
||||||
return `${path}`;
|
return `${path}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function numFormat(num, digits){
|
||||||
|
let d = digits || 1;
|
||||||
|
var si = [
|
||||||
|
{ value: 1, symbol: "" },
|
||||||
|
{ value: 1E3, symbol: "k" },
|
||||||
|
{ value: 1E4, symbol: "W" }
|
||||||
|
];
|
||||||
|
var rx = /\.0+$|(\.[0-9]*[1-9])0+$/;
|
||||||
|
var i;
|
||||||
|
for (i = si.length - 1; i > 0; i--) {
|
||||||
|
if (num >= si[i].value) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (num / si[i].value).toFixed(d).replace(rx, "$1") + si[i].symbol;
|
||||||
|
}
|
||||||
|
|
||||||
export function getImage(path) {
|
export function getImage(path) {
|
||||||
// https://www.educoder.net
|
// https://www.educoder.net
|
||||||
// https://testbdweb.trustie.net
|
// https://testbdweb.trustie.net
|
||||||
|
@ -162,28 +179,28 @@ export function getmyUrl(geturl) {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUploadActionUrl(path, goTest) {
|
export function getUploadActionUrl(path, goTest) {
|
||||||
return `${getUrl()}/api/attachments.json?debug=${window._debugType || 'admin'}`;
|
return `${getUrl()}/api/attachments.json${isDev ?`${isDev ?`?debug=${window._debugType || 'admin'}` : ""}` : ""}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUploadLogoActionUrl() {
|
export function getUploadLogoActionUrl() {
|
||||||
return `${getUrl()}/api/resumes/logo.json?debug=${window._debugType || 'admin'}`;
|
return `${getUrl()}/api/resumes/logo.json${isDev ?`?debug=${window._debugType || 'admin'}` : ""}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUploadActionUrltwo(id) {
|
export function getUploadActionUrltwo(id) {
|
||||||
return `${getUrlmys()}/api/shixuns/${id}/upload_data_sets.json?debug=${window._debugType || 'admin'}`
|
return `${getUrlmys()}/api/shixuns/${id}/upload_data_sets.json${isDev ?`?debug=${window._debugType || 'admin'}` : ""}`
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUploadActionUrlthree() {
|
export function getUploadActionUrlthree() {
|
||||||
return `${getUrlmys()}/api/jupyters/import_with_tpm.json?debug=${window._debugType || 'admin'}`
|
return `${getUrlmys()}/api/jupyters/import_with_tpm.json${isDev ?`?debug=${window._debugType || 'admin'}` : ""}`
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getupload_git_file(id) {
|
export function getupload_git_file(id) {
|
||||||
return `${getUrlmys()}/api/shixuns/${id}/upload_git_file.json?debug=${window._debugType || 'admin'}`
|
return `${getUrlmys()}/api/shixuns/${id}/upload_git_file.json${isDev ?`?debug=${window._debugType || 'admin'}` : ""}`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export function getUploadActionUrlOfAuth(id) {
|
export function getUploadActionUrlOfAuth(id) {
|
||||||
return `${getUrl()}/api/users/accounts/${id}/auth_attachment.json?debug=${window._debugType || 'admin'}`
|
return `${getUrl()}/api/users/accounts/${id}/auth_attachment.json${isDev ?`?debug=${window._debugType || 'admin'}` : ""}`
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getRandomNumber(type) {
|
export function getRandomNumber(type) {
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
// export { default as OrderStateUtil } from '../routes/Order/components/OrderStateUtil';
|
// export { default as OrderStateUtil } from '../routes/Order/components/OrderStateUtil';
|
||||||
|
|
||||||
export {
|
export {
|
||||||
getUploadLogoActionUrl as getUploadLogoActionUrl,
|
getUploadLogoActionUrl as getUploadLogoActionUrl,numFormat as numFormat,
|
||||||
getImageUrl as getImageUrl,getImage as getImage, getmyUrl as getmyUrl, getRandomNumber as getRandomNumber, getUrl as getUrl, publicSearchs as publicSearchs, getRandomcode as getRandomcode, getUrlmys as getUrlmys, getUrl2 as getUrl2, setImagesUrl as setImagesUrl
|
getImageUrl as getImageUrl,getImage as getImage, getmyUrl as getmyUrl, getRandomNumber as getRandomNumber, getUrl as getUrl, publicSearchs as publicSearchs, getRandomcode as getRandomcode, getUrlmys as getUrlmys, getUrl2 as getUrl2, setImagesUrl as setImagesUrl
|
||||||
, getUploadActionUrl as getUploadActionUrl, getUploadActionUrltwo as getUploadActionUrltwo, getUploadActionUrlthree as getUploadActionUrlthree, getUploadActionUrlOfAuth as getUploadActionUrlOfAuth
|
, getUploadActionUrl as getUploadActionUrl, getUploadActionUrltwo as getUploadActionUrltwo, getUploadActionUrlthree as getUploadActionUrlthree, getUploadActionUrlOfAuth as getUploadActionUrlOfAuth
|
||||||
, getTaskUrlById as getTaskUrlById, TEST_HOST, htmlEncode as htmlEncode, getupload_git_file as getupload_git_file, getcdnImageUrl as getcdnImageUrl
|
, getTaskUrlById as getTaskUrlById, TEST_HOST, htmlEncode as htmlEncode, getupload_git_file as getupload_git_file, getcdnImageUrl as getcdnImageUrl
|
||||||
|
@ -27,7 +27,7 @@ export {
|
||||||
markdownToHTML, uploadNameSizeSeperator, appendFileSizeToUploadFile, appendFileSizeToUploadFileAll, isImageExtension,
|
markdownToHTML, uploadNameSizeSeperator, appendFileSizeToUploadFile, appendFileSizeToUploadFileAll, isImageExtension,
|
||||||
downloadFile, sortDirections, validateLength, mdJSONParse, exportMdtoHtml
|
downloadFile, sortDirections, validateLength, mdJSONParse, exportMdtoHtml
|
||||||
} from './TextUtil'
|
} from './TextUtil'
|
||||||
export { handleDateString, getNextHalfHourOfMoment, formatDuring, formatSeconds } from './DateUtil'
|
export { handleDateString, getNextHalfHourOfMoment, formatDuring, formatSeconds ,timeAgo} from './DateUtil'
|
||||||
|
|
||||||
export { configShareForIndex, configShareForPaths, configShareForShixuns, configShareForCourses, configShareForCustom } from './util/ShareUtil'
|
export { configShareForIndex, configShareForPaths, configShareForShixuns, configShareForCourses, configShareForCustom } from './util/ShareUtil'
|
||||||
|
|
||||||
|
|
|
@ -46,18 +46,18 @@ export default ({
|
||||||
let id = decodeURIComponent(u.split("#")[1]);
|
let id = decodeURIComponent(u.split("#")[1]);
|
||||||
let ele = document.getElementById(id);
|
let ele = document.getElementById(id);
|
||||||
if(ele){
|
if(ele){
|
||||||
window.scrollTo(0, ele.offsetTop + 220);
|
window.scrollTo(0, ele.offsetTop + 120);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},[url])
|
},[url,html])
|
||||||
|
|
||||||
const el = useRef();
|
const el = useRef();
|
||||||
function onAncherHandler(e) {
|
function onAncherHandler(e) {
|
||||||
let target = e.target
|
let target = e.target;
|
||||||
if (target.tagName.toUpperCase() === 'A') {
|
if (target.tagName.toUpperCase() === 'A') {
|
||||||
let ancher = target.getAttribute('href')
|
let ancher = target.getAttribute('href');
|
||||||
if (ancher.startsWith('#')) {
|
if (ancher && ancher.startsWith('#')) {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
let viewEl = document.getElementById(ancher.replace('#', ''))
|
let viewEl = document.getElementById(ancher.replace('#', ''))
|
||||||
if (viewEl) {
|
if (viewEl) {
|
||||||
|
|
|
@ -435,11 +435,11 @@ class TPIContextProvider extends Component {
|
||||||
image_url: "avatars/User/1"
|
image_url: "avatars/User/1"
|
||||||
login: "innov"
|
login: "innov"
|
||||||
name: "Coder"
|
name: "Coder"
|
||||||
user_url: "/users/innov"
|
user_url: "/innov"
|
||||||
*/
|
*/
|
||||||
let user = resData.user;
|
let user = resData.user;
|
||||||
user.username = resData.user.name;
|
user.username = resData.user.name;
|
||||||
user.user_url = `/users/${resData.user.login}`;
|
user.user_url = `/${resData.user.login}`;
|
||||||
// user.image_url = resData.image_url;
|
// user.image_url = resData.image_url;
|
||||||
user.is_teacher = resData.is_teacher;
|
user.is_teacher = resData.is_teacher;
|
||||||
resData.user = user;
|
resData.user = user;
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import React , { Component } from 'react';
|
import React , { Component } from 'react';
|
||||||
import { Dropdown , Menu , Icon , Pagination , Spin } from 'antd';
|
import { Dropdown , Menu , Icon , Pagination , Spin } from 'antd';
|
||||||
import '../css/index.scss';
|
import '../css/index.scss';
|
||||||
import '../Branch/branch.css';
|
import '../Branch/branch.scss';
|
||||||
import './activity.css';
|
import './activity.css';
|
||||||
import NoneData from '../Nodata';
|
import NoneData from '../Nodata';
|
||||||
|
|
||||||
|
@ -10,6 +10,10 @@ import ActivityItem from './ActivityItem';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
const LIMIT = 15;
|
const LIMIT = 15;
|
||||||
const ARRAY = [
|
const ARRAY = [
|
||||||
|
{
|
||||||
|
id:"",
|
||||||
|
name:'全部'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
id:1,
|
id:1,
|
||||||
name:'1天'
|
name:'1天'
|
||||||
|
@ -32,10 +36,15 @@ class Activity extends Component{
|
||||||
constructor(props){
|
constructor(props){
|
||||||
super(props);
|
super(props);
|
||||||
this.state={
|
this.state={
|
||||||
time:'30',
|
time:undefined,
|
||||||
type:undefined,
|
type:undefined,
|
||||||
state:undefined,
|
state:undefined,
|
||||||
page:1,
|
page:1,
|
||||||
|
pr_count:undefined,
|
||||||
|
new_pr_count:undefined,
|
||||||
|
close_issues_count:undefined,
|
||||||
|
open_issues_count:undefined,
|
||||||
|
pr_all_count:undefined,issues_count:undefined,
|
||||||
|
|
||||||
data:undefined,
|
data:undefined,
|
||||||
project_trends:undefined,
|
project_trends:undefined,
|
||||||
|
@ -63,8 +72,15 @@ class Activity extends Component{
|
||||||
this.setState({
|
this.setState({
|
||||||
data:result.data,
|
data:result.data,
|
||||||
project_trends:result.data.project_trends,
|
project_trends:result.data.project_trends,
|
||||||
isSpin:false
|
isSpin:false,
|
||||||
|
pr_count:result.data.pr_count,
|
||||||
|
new_pr_count:result.data.new_pr_count,
|
||||||
|
close_issues_count:result.data.close_issues_count,
|
||||||
|
open_issues_count:result.data.open_issues_count,
|
||||||
|
pr_all_count:result.data.pr_all_count,
|
||||||
|
issues_count:result.data.issues_count,
|
||||||
})
|
})
|
||||||
|
window.scrollTo(0,0);
|
||||||
}
|
}
|
||||||
}).catch(error=>{
|
}).catch(error=>{
|
||||||
console.log(error);
|
console.log(error);
|
||||||
|
@ -74,19 +90,19 @@ class Activity extends Component{
|
||||||
// 切换周期
|
// 切换周期
|
||||||
changeTime=(e)=>{
|
changeTime=(e)=>{
|
||||||
this.setState({
|
this.setState({
|
||||||
time:e.key,
|
time:e.key ==="item_0"?undefined:e.key,
|
||||||
isSpin:true
|
isSpin:true
|
||||||
})
|
})
|
||||||
const { type,status,page } = this.state;
|
const { type,status,page } = this.state;
|
||||||
this.getInfo(e.key,type,status,page);
|
this.getInfo(e.key ==="item_0"?undefined:e.key,type,status,page);
|
||||||
}
|
}
|
||||||
//筛选
|
//筛选
|
||||||
changeTrends=(type,status)=>{
|
changeTrends=(type,status)=>{
|
||||||
this.setState({
|
this.setState({
|
||||||
type,status
|
type,status,page:1
|
||||||
})
|
})
|
||||||
const {time,page}=this.state;
|
const {time}=this.state;
|
||||||
this.getInfo(time,type,status,page);
|
this.getInfo(time,type,status,1);
|
||||||
}
|
}
|
||||||
// 分页
|
// 分页
|
||||||
ChangePage=(page)=>{
|
ChangePage=(page)=>{
|
||||||
|
@ -108,12 +124,14 @@ class Activity extends Component{
|
||||||
</Menu>
|
</Menu>
|
||||||
)
|
)
|
||||||
render(){
|
render(){
|
||||||
const { time , data , page , project_trends , isSpin } = this.state;
|
const { time , data , page , project_trends , isSpin , pr_count , new_pr_count , close_issues_count , open_issues_count , pr_all_count ,issues_count } = this.state;
|
||||||
|
let name = time ? ARRAY.filter(item=>item.id === parseInt(time)) :[{name:"全部"}];
|
||||||
|
|
||||||
let name = time && ARRAY.filter(item=>item.id === parseInt(time)) ;
|
const first_per = pr_all_count > 0 ? `${parseFloat(pr_count/pr_all_count).toFixed(2)*100}%` :"50%";
|
||||||
const second_per = (parseInt(data && data.close_issues_count)/parseInt(data && data.issues_count)*100)+'%';
|
const second_per =pr_all_count > 0 ? `${parseFloat(new_pr_count/pr_all_count).toFixed(2)*100}%` :"50%";
|
||||||
const third_per = (parseInt(data && data.close_issues_count)/parseInt(data && data.issues_count)*100)+'%';
|
const third_per =issues_count > 0 ?`${parseFloat(close_issues_count/issues_count).toFixed(2)*100}%` :"50%";
|
||||||
const fourth_per = (parseInt(data && data.open_issues_count)/parseInt(data && data.issues_count)*100)+'%';
|
const fourth_per =issues_count > 0 ?`${parseFloat(open_issues_count/issues_count).toFixed(2)*100}%` :"50%";
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<div className="main">
|
<div className="main">
|
||||||
|
|
||||||
|
@ -122,7 +140,7 @@ class Activity extends Component{
|
||||||
<div className="orderInfo">
|
<div className="orderInfo">
|
||||||
<div>
|
<div>
|
||||||
<div className="percentLine prPercent">
|
<div className="percentLine prPercent">
|
||||||
<p className="percent_purple" style={{width:'100%'}}></p>
|
<p className="percent_purple" style={{width:first_per}}></p>
|
||||||
<p className="percent_green resetStyle" style={{width:`${second_per}`}}></p>
|
<p className="percent_green resetStyle" style={{width:`${second_per}`}}></p>
|
||||||
</div>
|
</div>
|
||||||
<span>{data && data.pr_all_count}合并请求</span>
|
<span>{data && data.pr_all_count}合并请求</span>
|
||||||
|
@ -132,25 +150,25 @@ class Activity extends Component{
|
||||||
<p className="percent_red" style={{width:`${third_per}`}}></p>
|
<p className="percent_red" style={{width:`${third_per}`}}></p>
|
||||||
<p className="percent_green" style={{width:`${fourth_per}`}}></p>
|
<p className="percent_green" style={{width:`${fourth_per}`}}></p>
|
||||||
</div>
|
</div>
|
||||||
<span>{data && data.issues_count}任务</span>
|
<span>{data && data.issues_count}易修</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ul className="percentBox">
|
<ul className="percentBox">
|
||||||
<li>
|
<li>
|
||||||
<span className="purple">{data && data.pr_count}</span>
|
<span className="purple">{data && data.pr_count}</span>
|
||||||
<span className="change" onClick={()=>this.changeTrends("PullRequest","close")}>已处理的合并请求</span>
|
<span className="change" onClick={()=>this.changeTrends("PullRequest","delay")}>已处理的合并请求</span>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<span className="green">{data && data.new_pr_count}</span>
|
<span className="green">{data && data.new_pr_count}</span>
|
||||||
<span className="change" onClick={()=>this.changeTrends("PullRequest","create")}>未处理的合并请求</span>
|
<span className="change" onClick={()=>this.changeTrends("PullRequest","not_delay")}>未处理的合并请求</span>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<span className="red">{data && data.close_issues_count}</span>
|
<span className="red">{data && data.close_issues_count}</span>
|
||||||
<span className="change" onClick={()=>this.changeTrends("Issue","close")}>已关闭的任务</span>
|
<span className="change" onClick={()=>this.changeTrends("Issue","delay")}>已关闭的易修</span>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<span className="green">{data && data.open_issues_count}</span>
|
<span className="green">{data && data.open_issues_count}</span>
|
||||||
<span className="change" onClick={()=>this.changeTrends("Issue","create")}>未处理的任务</span>
|
<span className="change" onClick={()=>this.changeTrends("Issue","not_delay")}>未处理的易修</span>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -14,25 +14,25 @@ class ActivityItem extends Component {
|
||||||
{/* 如果是版本发布 */}
|
{/* 如果是版本发布 */}
|
||||||
{item.trend_type === "VersionRelease" ?
|
{item.trend_type === "VersionRelease" ?
|
||||||
<p className="itemLine">
|
<p className="itemLine">
|
||||||
<Link to={`/projects/${owner}/${projectsId}/version`} className="color-blue font-16">{item.name}</Link>
|
<Link to={`/${owner}/${projectsId}/releases`} className="color-blue font-16">{item.name}</Link>
|
||||||
<span className="activity_type">{item.trend_type}</span>
|
<span className="activity_type">{item.trend_type}</span>
|
||||||
</p >
|
</p >
|
||||||
:
|
:
|
||||||
// 如果是任务
|
// 如果是任务
|
||||||
item.trend_type === "Issue" ?
|
item.trend_type === "Issue" ?
|
||||||
<p className="itemLine">
|
<p className="itemLine">
|
||||||
<Link to={`/projects/${owner}/${projectsId}/issues/${item.trend_id}/detail`} className="color-blue font-16">{item.name}</Link>
|
<Link to={`/${owner}/${projectsId}/issues/${item.trend_id}`} className="color-blue font-16">{item.name}</Link>
|
||||||
<span className="activity_type">{item.trend_type}</span>
|
<span className="activity_type">{item.trend_type}</span>
|
||||||
</p >
|
</p >
|
||||||
:
|
:
|
||||||
// 如果是合并请求
|
// 如果是合并请求
|
||||||
<p className="itemLine">
|
<p className="itemLine">
|
||||||
<Link to={`/projects/${owner}/${projectsId}/pulls/${item.trend_id}/Messagecount`} className="color-blue font-16">{item.name}</Link>
|
<Link to={`/${owner}/${projectsId}/pulls/${item.trend_id}`} className="color-blue font-16">{item.name}</Link>
|
||||||
<span className="activity_type">{item.trend_type}</span>
|
<span className="activity_type">{item.trend_type}</span>
|
||||||
</p >
|
</p >
|
||||||
}
|
}
|
||||||
<p className="itemLine mt10">
|
<p className="itemLine mt10">
|
||||||
<Link to={`/users/${item && item.user_login}`} className="show-user-link">
|
<Link to={`/${item && item.user_login}`} className="show-user-link">
|
||||||
<img alt="" src={getImageUrl(`/${item.user_avatar}`)} className="createImage" />
|
<img alt="" src={getImageUrl(`/${item.user_avatar}`)} className="createImage" />
|
||||||
<span className="mr20">{item.user_name}</span>
|
<span className="mr20">{item.user_name}</span>
|
||||||
</Link>
|
</Link>
|
||||||
|
|
|
@ -1,39 +1,27 @@
|
||||||
import React, { Component } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { Dropdown, Icon, Tooltip } from 'antd';
|
import { Menu } from 'antd';
|
||||||
import "./branch.css";
|
import "./branch.scss";
|
||||||
|
import CopyTool from '../Component/CopyTool';
|
||||||
|
|
||||||
class CloneAddress extends Component {
|
function CloneAddress({http_url , ssh_url , zip_url , tar_url}) {
|
||||||
// 点击按钮复制功能
|
const [ key , setKey ] = useState("HTTP");
|
||||||
jsCopy = () => {
|
return (
|
||||||
var e = document.getElementById("copy_rep_content");
|
<div className="downMenu">
|
||||||
e.select();
|
<div style={{borderBottom:"1px solid #eee"}}>
|
||||||
document.execCommand("Copy");
|
<Menu className="urlMenu" selectedKeys={[key]} mode={"horizontal"}>
|
||||||
}
|
<Menu.Item key="HTTP" onClick={(e)=>{setKey(e.key)}}>HTTP</Menu.Item>
|
||||||
|
<Menu.Item key="SSH" onClick={(e)=>{setKey(e.key)}}>SSH</Menu.Item>
|
||||||
render() {
|
</Menu>
|
||||||
const { http_url, downloadUrl } = this.props;
|
<div className="gitAddressClone">
|
||||||
return (
|
<input type="text" id="copy_rep_content" value={key==="HTTP" ? http_url:ssh_url} />
|
||||||
<div className="gitAddressClone">
|
<CopyTool inputId="copy_rep_content" className="copytool"/>
|
||||||
{/* <p className="addressTips"><span>版本库地址已变更,请基于新地址提交代码</span></p> */}
|
</div>
|
||||||
{
|
|
||||||
http_url && <span>HTTP</span>
|
|
||||||
}
|
|
||||||
<input type="text" id="copy_rep_content" value={http_url} />
|
|
||||||
<Tooltip title="复制链接">
|
|
||||||
<span className="color-blue" onClick={() => this.jsCopy()}><i className="iconfont icon-fuzhi"></i></span>
|
|
||||||
</Tooltip>
|
|
||||||
{
|
|
||||||
downloadUrl &&
|
|
||||||
<span>
|
|
||||||
<Dropdown overlay={downloadUrl} trigger={['click']} placement="bottomRight">
|
|
||||||
<a className="ant-dropdown-link">
|
|
||||||
<Icon type="cloud-download" className="font-18 fl color-blue" />
|
|
||||||
</a>
|
|
||||||
</Dropdown>
|
|
||||||
</span>
|
|
||||||
}
|
|
||||||
</div>
|
</div>
|
||||||
)
|
<Menu className="edu-txt-center">
|
||||||
}
|
<Menu.Item><a href={zip_url}>下载 ZIP</a></Menu.Item>
|
||||||
|
<Menu.Item><a href={tar_url}>下载 TAR.GZ</a></Menu.Item>
|
||||||
|
</Menu>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
export default CloneAddress;
|
export default CloneAddress;
|
|
@ -1,116 +1,65 @@
|
||||||
import React , { useState , useEffect } from 'react';
|
import React , { useState , useEffect , useRef } from 'react';
|
||||||
import { Popover , Input , Spin } from 'antd';
|
import { Dropdown} from 'antd';
|
||||||
import './branch.css';
|
import './branch.scss';
|
||||||
import { getBranch , getTag } from '../GetData/getData';
|
import SelectOverlay from './SelectOverlay';
|
||||||
|
import { findDOMNode } from 'react-dom';
|
||||||
|
|
||||||
export default (({ projectsId , branch , owner , changeBranch , branchList , tagflag = true })=>{
|
export default (({ projectsId , branch , owner , changeBranch , branchList , tagflag = true })=>{
|
||||||
const [ showValue , setShowValue ] = useState(branch);
|
const [ showValue , setShowValue ] = useState(branch);
|
||||||
const [ inputValue , setInputValue] = useState(undefined);
|
const [ visible , setVisible ] = useState(false);
|
||||||
const [ nav , setNav ] = useState(0);
|
|
||||||
const [ isSpin , setIsSpin ] = useState(true);
|
|
||||||
const [ flag , setFlag ] = useState(false);
|
|
||||||
|
|
||||||
const [ data , setData ] = useState(undefined);
|
const refFa = useRef(null);
|
||||||
const [ datas , setDatas ] = useState(undefined);
|
const refBox = useRef(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
document.addEventListener('click', clickMe , false);
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const clickMe = ({ target }) => {
|
||||||
|
// 查找父组件
|
||||||
|
const faComponent = findDOMNode(refFa.current);
|
||||||
|
const boxComponent = findDOMNode(refBox.current);
|
||||||
|
|
||||||
|
if (faComponent && boxComponent) {
|
||||||
|
const isChild = faComponent.contains(target);
|
||||||
|
const isBox = boxComponent.contains(target);
|
||||||
|
if(!isChild && !isBox){
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(()=>{
|
useEffect(()=>{
|
||||||
setShowValue(branch);
|
setShowValue(branch);
|
||||||
},[branch])
|
},[branch])
|
||||||
|
|
||||||
useEffect(()=>{
|
function ChangeB(params) {
|
||||||
document.body.addEventListener('click', e => {
|
setVisible(false);
|
||||||
let name = e.target.className;
|
changeBranch(params);
|
||||||
let turn = name === "ant-input OptionsInput" || name === "navli active"|| name === "navli" || name === "padding10 bor-bottom-greyE";
|
|
||||||
if(turn){
|
|
||||||
return;
|
|
||||||
}else{
|
|
||||||
setFlag(false);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
useEffect(()=>{
|
|
||||||
if(branchList){
|
|
||||||
setData(branchList);
|
|
||||||
setDatas(branchList);
|
|
||||||
setIsSpin(false);
|
|
||||||
}
|
|
||||||
},[branchList])
|
|
||||||
|
|
||||||
|
|
||||||
async function getBranchs(id,owner){
|
|
||||||
let result = await getBranch(id,owner);
|
|
||||||
setData(result);
|
|
||||||
setDatas(result);
|
|
||||||
setIsSpin(false);
|
|
||||||
}
|
}
|
||||||
async function getTags(id,owner){
|
|
||||||
let result = await getTag(id,owner);
|
|
||||||
setData(result);
|
|
||||||
setDatas(result);
|
|
||||||
setIsSpin(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
function changeInputValue(e){
|
|
||||||
setInputValue(e.target.value);
|
|
||||||
let filter = e.target.value ? data && data.length>0 && data.filter(item=>item.name.indexOf(e.target.value)>-1) : data;
|
|
||||||
setDatas(filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
function changeNav(nav){
|
|
||||||
setNav(nav);
|
|
||||||
setIsSpin(true);
|
|
||||||
if(nav === 0){
|
|
||||||
getBranchs(projectsId,owner);
|
|
||||||
}else{
|
|
||||||
getTags(projectsId,owner);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function chooseitem(value){
|
|
||||||
// setShowValue(value);
|
|
||||||
changeBranch(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
const menu = (
|
const menu = (
|
||||||
<div>
|
<div ref={refFa}>
|
||||||
<div className="padding10 bor-bottom-greyE">
|
<SelectOverlay
|
||||||
<Input
|
visible={visible}
|
||||||
placeholder="请输入分支或标签名称搜索"
|
changeBranch={ChangeB}
|
||||||
autocomplete="off" className="OptionsInput" value={inputValue}
|
tagflag={tagflag}
|
||||||
onChange={changeInputValue} style={{width:"220px"}}
|
projectsId={projectsId}
|
||||||
/>
|
owner={owner}
|
||||||
<ul className="navUl">
|
branchList={branchList}
|
||||||
<li className={nav === 0?"navli active":"navli"} onClick={()=>changeNav(0)}><i className="iconfont icon-fenzhi1 font-14 mr3"></i>分支列表</li>
|
/>
|
||||||
{ tagflag && <li className={nav === 1?"navli active":"navli"} onClick={()=>changeNav(1)}><i className="iconfont icon-biaoqian3 font-14 mr3"></i>标签列表</li> }
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<Spin spinning={isSpin}>
|
|
||||||
<ul className="OptionsUl" id="ul-btn">
|
|
||||||
{
|
|
||||||
datas && datas.length>0 ?
|
|
||||||
datas.map((item,key)=>{
|
|
||||||
return(
|
|
||||||
<li key={key} onClick={()=>chooseitem(item.name)}><a className="task-hide ulALink">{item.name}</a></li>
|
|
||||||
)
|
|
||||||
}):
|
|
||||||
<p className="listTips">暂无{inputValue}{nav === 0 ?"分支":"标签"}~</p>
|
|
||||||
}
|
|
||||||
</ul>
|
|
||||||
</Spin>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
return(
|
return(
|
||||||
<Popover placement="bottom" visible={flag} content={menu} onClick={()=>setFlag(!flag)} overlayClassName="branch-tagBox-list">
|
<Dropdown placement='bottomLeft' visible={visible} overlay={menu} overlayClassName="branch-tagBox-list" trigger={['click']} >
|
||||||
<div className="branch-tagBox">
|
<div className="branch-tagBox" ref={refBox} onClick={()=>setVisible(visible ? false : true)}>
|
||||||
{/* {nav === 0 ?"分支":"标签"} */}
|
{/* {nav === 0 ?"分支":"标签"} */}
|
||||||
<span className="color-grey-9 mr3 ml8"><i className="iconfont icon-fenzhi2 font-18"></i></span>
|
<span className="color-grey-9 mr3 ml8"><i className="iconfont icon-fenzhi2 font-18"></i></span>
|
||||||
<a className="ant-dropdown-link">
|
<span className="ant-dropdown-link task-hide" style={{fontWeight:"500",minWidth:"45px",maxWidth:"270px"}}>
|
||||||
{showValue}
|
{showValue}
|
||||||
</a>
|
</span>
|
||||||
<i className="showtag iconfont icon-xiajiantou font-14 color-grey-9 mr8" />
|
<i className="showtag iconfont icon-sanjiaoxing-down font-15 color-grey-9 mr5 ml5 mt1" />
|
||||||
</div>
|
</div>
|
||||||
</Popover>
|
</Dropdown>
|
||||||
)
|
)
|
||||||
})
|
})
|
|
@ -0,0 +1,90 @@
|
||||||
|
import React , { useState , useEffect } from 'react';
|
||||||
|
import { Input , Spin , Menu } from 'antd';
|
||||||
|
import { getBranch , getTag } from '../GetData/getData';
|
||||||
|
|
||||||
|
function SelectOverlay({ changeBranch , tagflag , projectsId , owner , visible }) {
|
||||||
|
const [ inputValue , setInputValue] = useState(undefined);
|
||||||
|
const [ nav , setNav ] = useState(0);
|
||||||
|
const [ isSpin , setIsSpin ] = useState(true);
|
||||||
|
|
||||||
|
const [ data , setData ] = useState(undefined);
|
||||||
|
const [ datas , setDatas ] = useState(undefined);
|
||||||
|
const [ keys ,setKeys] = useState("branch");
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
if(visible){
|
||||||
|
setKeys("branch");
|
||||||
|
getBranchs(projectsId,owner);
|
||||||
|
setIsSpin(true);
|
||||||
|
}
|
||||||
|
},[visible])
|
||||||
|
|
||||||
|
async function getBranchs(id,owner){
|
||||||
|
let result = await getBranch(id,owner);
|
||||||
|
setData(result);
|
||||||
|
setDatas(result);
|
||||||
|
setIsSpin(false);
|
||||||
|
}
|
||||||
|
async function getTags(id,owner){
|
||||||
|
let result = await getTag(id,owner);
|
||||||
|
setData(result);
|
||||||
|
setDatas(result);
|
||||||
|
setIsSpin(false);
|
||||||
|
}
|
||||||
|
function chooseitem(value){
|
||||||
|
changeBranch(value);
|
||||||
|
}
|
||||||
|
function changeInputValue(e){
|
||||||
|
setInputValue(e.target.value);
|
||||||
|
let filter = e.target.value ? data && data.length>0 && data.filter(item=>item.name.indexOf(e.target.value)>-1) : data;
|
||||||
|
setDatas(filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
function changeNav(e){
|
||||||
|
setKeys(e.key);
|
||||||
|
setIsSpin(true);
|
||||||
|
if(e.key === "branch"){
|
||||||
|
getBranchs(projectsId,owner);
|
||||||
|
setNav(0);
|
||||||
|
}else{
|
||||||
|
getTags(projectsId,owner);
|
||||||
|
setNav(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return(
|
||||||
|
<div className="overlayBranch">
|
||||||
|
<div className="padding15" style={{paddingBottom:"0px"}}>
|
||||||
|
<Input
|
||||||
|
prefix={<i className="iconfont icon-sousuo_icon1 font-14"></i>}
|
||||||
|
placeholder={`请输入分支${tagflag ? "或标签" :""}名称搜索`}
|
||||||
|
autocomplete="off" className="OptionsInput"
|
||||||
|
value={inputValue}
|
||||||
|
onChange={changeInputValue}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<Menu mode="horizontal" className="navUl" selectedKeys={[keys]} onClick={changeNav}>
|
||||||
|
<Menu.Item key={"branch"}>分支</Menu.Item>
|
||||||
|
{ tagflag && <Menu.Item key={"tag"}>标签</Menu.Item> }
|
||||||
|
</Menu>
|
||||||
|
<Spin spinning={isSpin}>
|
||||||
|
<ul className="OptionsUl" id="ul-btn">
|
||||||
|
{
|
||||||
|
datas && datas.length>0 &&
|
||||||
|
datas.map((item,key)=>{
|
||||||
|
return(
|
||||||
|
<li key={key} onClick={()=>chooseitem(item.name)}><a className="task-hide ulALink">{item.name}</a></li>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
{
|
||||||
|
datas && datas.length === 0 &&
|
||||||
|
<p className="listTips">暂无{inputValue}{nav === 0 ?"分支":"标签"}~</p>
|
||||||
|
}
|
||||||
|
</ul>
|
||||||
|
</Spin>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default SelectOverlay;
|
|
@ -1,84 +0,0 @@
|
||||||
.branchDropdown{
|
|
||||||
border:1px solid #eee;
|
|
||||||
border-radius: 4px;
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
height: 40px;
|
|
||||||
line-height: 40px;
|
|
||||||
min-width: 220px;
|
|
||||||
}
|
|
||||||
.branchDropdown .ant-dropdown-trigger{
|
|
||||||
width: 100%;
|
|
||||||
padding:0px 15px;
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: center;
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
.branchOptions{
|
|
||||||
width: 220px;
|
|
||||||
box-shadow: 0px 0px 3px 1px rgba(134, 134, 134, 0.4);
|
|
||||||
border-radius: 3px;
|
|
||||||
background: #fff;
|
|
||||||
max-height: 300px;
|
|
||||||
}
|
|
||||||
.OptionsUl{
|
|
||||||
max-height: 220px;
|
|
||||||
overflow-y: auto;
|
|
||||||
}
|
|
||||||
.OptionsUl li{
|
|
||||||
height: 35px;
|
|
||||||
line-height: 35px;
|
|
||||||
cursor: pointer;
|
|
||||||
padding:0px 10px;
|
|
||||||
}
|
|
||||||
.OptionsUl li:hover{
|
|
||||||
background-color: #F0F0F0;
|
|
||||||
}
|
|
||||||
.OptionsUl li a{
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
.OptionsInput{
|
|
||||||
height: 32px;
|
|
||||||
padding-left: 4px;
|
|
||||||
line-height: 32px;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
.branch-tagBox{
|
|
||||||
border:1px solid #eee;
|
|
||||||
border-radius: 3px;
|
|
||||||
height: 40px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
cursor: pointer;
|
|
||||||
min-width: 140px;
|
|
||||||
}
|
|
||||||
.branch-tagBox-list .ant-popover-arrow{
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
.branch-tagBox-list.ant-popover.ant-popover-placement-bottom{
|
|
||||||
padding-top:0px;
|
|
||||||
}
|
|
||||||
.branch-tagBox .ant-dropdown-link{
|
|
||||||
display: block;
|
|
||||||
flex:1;
|
|
||||||
}
|
|
||||||
.branch-tagBox-list .ant-popover-inner-content{
|
|
||||||
padding:0px;
|
|
||||||
}
|
|
||||||
.navUl{
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: center;
|
|
||||||
margin-top: 5px;
|
|
||||||
}
|
|
||||||
.navUl li{
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
.navUl li.active{
|
|
||||||
color:#5091FF;
|
|
||||||
}
|
|
||||||
.listTips{
|
|
||||||
padding:20px 0px;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
|
@ -0,0 +1,129 @@
|
||||||
|
.branchDropdown{
|
||||||
|
border:1px solid #eee;
|
||||||
|
border-radius: 4px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
min-width: 220px;
|
||||||
|
}
|
||||||
|
.branchDropdown .ant-dropdown-trigger{
|
||||||
|
width: 100%;
|
||||||
|
padding:0px 15px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.branchOptions{
|
||||||
|
width: 220px;
|
||||||
|
box-shadow: 0px 0px 3px 1px rgba(134, 134, 134, 0.4);
|
||||||
|
border-radius: 3px;
|
||||||
|
background: #fff;
|
||||||
|
max-height: 300px;
|
||||||
|
}
|
||||||
|
.OptionsUl{
|
||||||
|
min-height: 50px;
|
||||||
|
max-height: 220px;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
.OptionsUl li{
|
||||||
|
height: 30px;
|
||||||
|
line-height: 30px;
|
||||||
|
cursor: pointer;
|
||||||
|
padding:0px 20px;
|
||||||
|
margin:5px 0px;
|
||||||
|
}
|
||||||
|
.OptionsUl li:hover{
|
||||||
|
background-color: #F0F0F0;
|
||||||
|
}
|
||||||
|
.OptionsUl li a{
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.OptionsInput{
|
||||||
|
height: 32px;
|
||||||
|
padding-left: 4px;
|
||||||
|
line-height: 32px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
.branch-tagBox{
|
||||||
|
border:1px solid #D0D0D0;
|
||||||
|
border-radius: 3px;
|
||||||
|
height: 32px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
cursor: pointer;
|
||||||
|
min-width: 104px;
|
||||||
|
}
|
||||||
|
.branch-tagBox:hover{
|
||||||
|
background-color: #F3F4F6;
|
||||||
|
}
|
||||||
|
.branch-tagBox-list{
|
||||||
|
background: #FFFFFF;
|
||||||
|
box-shadow: 0px 4px 8px 2px rgba(212, 212, 212, 0.5);
|
||||||
|
border-radius: 4px;
|
||||||
|
.ant-popover-arrow{
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
&.ant-popover.ant-popover-placement-bottom{
|
||||||
|
padding-top:0px;
|
||||||
|
}
|
||||||
|
.branch-tagBox .ant-dropdown-link{
|
||||||
|
display: block;
|
||||||
|
flex:1;
|
||||||
|
max-width: 105px;
|
||||||
|
}
|
||||||
|
.ant-popover-inner-content{
|
||||||
|
padding:0px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.overlayBranch{
|
||||||
|
width: 325px;
|
||||||
|
.navUl{
|
||||||
|
margin-top: 8px;
|
||||||
|
height: 30px;
|
||||||
|
line-height: 30px;
|
||||||
|
li{
|
||||||
|
height: 30px;
|
||||||
|
line-height: 30px;
|
||||||
|
padding:0px 5px;
|
||||||
|
margin-left: 20px!important;
|
||||||
|
&.ant-menu-item-selected{
|
||||||
|
border-color:#466aff!important;
|
||||||
|
color:#466aff!important;
|
||||||
|
}
|
||||||
|
&.ant-menu-item-active{
|
||||||
|
border-color:transparent ;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.listTips{
|
||||||
|
padding:20px 0px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.urlMenu{
|
||||||
|
line-height: 30px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
padding:15px 20px 0px 20px;
|
||||||
|
border-bottom: none;
|
||||||
|
li.ant-menu-item{
|
||||||
|
height: 30px;
|
||||||
|
line-height: 30px;
|
||||||
|
padding:0px 5px;
|
||||||
|
margin-right: 20px!important;
|
||||||
|
&.ant-menu-item-selected,&.ant-menu-item-active{
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
&.ant-menu-item-selected{
|
||||||
|
border-color:#466aff!important;
|
||||||
|
}
|
||||||
|
&.ant-menu-item-active{
|
||||||
|
border-color:transparent ;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.copytool{
|
||||||
|
margin:0px 10px;
|
||||||
|
}
|
|
@ -55,6 +55,7 @@ function AddGroup({organizeId,getGroupID}){
|
||||||
|
|
||||||
function addCollaborator(){
|
function addCollaborator(){
|
||||||
getGroupID && getGroupID(id);
|
getGroupID && getGroupID(id);
|
||||||
|
setID(undefined);
|
||||||
}
|
}
|
||||||
|
|
||||||
return(
|
return(
|
||||||
|
|
|
@ -4,7 +4,7 @@ import axios from 'axios';
|
||||||
import { getImageUrl } from 'educoder';
|
import { getImageUrl } from 'educoder';
|
||||||
|
|
||||||
const { Option } = AutoComplete;
|
const { Option } = AutoComplete;
|
||||||
function AddMember({getID,login}){
|
function AddMember({getID,login,showNotification}){
|
||||||
const [ id , setID ] = useState(undefined);
|
const [ id , setID ] = useState(undefined);
|
||||||
const [ source , setSource ] = useState(undefined);
|
const [ source , setSource ] = useState(undefined);
|
||||||
const [ searchKey , setSearchKey ] = useState(undefined);
|
const [ searchKey , setSearchKey ] = useState(undefined);
|
||||||
|
@ -66,8 +66,12 @@ function AddMember({getID,login}){
|
||||||
};
|
};
|
||||||
|
|
||||||
function addCollaborator(){
|
function addCollaborator(){
|
||||||
getID && getID(id);
|
if(source && source.length>0){
|
||||||
setSearchKey(undefined);
|
getID && getID(id);
|
||||||
|
setSearchKey(undefined);
|
||||||
|
}else{
|
||||||
|
showNotification("请选择存在的用户!");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return(
|
return(
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { getImageUrl } from 'educoder';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
import './Component.scss';
|
import './Component.scss';
|
||||||
|
|
||||||
function Cards({img , title, desc , rightBtn , src}){
|
function Cards({img , title, desc , rightBtn , src , bottomInfos}){
|
||||||
return(
|
return(
|
||||||
<div className="cards">
|
<div className="cards">
|
||||||
{img &&<div className="img"><img src={getImageUrl(`/${img}`)} alt=""/></div>}
|
{img &&<div className="img"><img src={getImageUrl(`/${img}`)} alt=""/></div>}
|
||||||
|
@ -15,6 +15,7 @@ function Cards({img , title, desc , rightBtn , src}){
|
||||||
<div className="desc">
|
<div className="desc">
|
||||||
{desc}
|
{desc}
|
||||||
</div>
|
</div>
|
||||||
|
{bottomInfos}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|
|
@ -112,6 +112,16 @@ li.ant-menu-item{
|
||||||
right:240px;
|
right:240px;
|
||||||
z-index: 10000;
|
z-index: 10000;
|
||||||
}
|
}
|
||||||
|
.laterest{
|
||||||
|
background-color: #EF3131;
|
||||||
|
color: #fff;
|
||||||
|
font-size: 12px;
|
||||||
|
margin-left: 10px;
|
||||||
|
padding:0px 5px;
|
||||||
|
border-radius: 2px;
|
||||||
|
height: 18px;
|
||||||
|
line-height: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
@media screen and (max-width: 1800px){
|
@media screen and (max-width: 1800px){
|
||||||
.handleBox{
|
.handleBox{
|
||||||
|
@ -152,41 +162,280 @@ li.ant-menu-item{
|
||||||
margin:0px 20px!important;
|
margin:0px 20px!important;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.hoverA{
|
||||||
|
display:flex;
|
||||||
|
align-items: center;
|
||||||
|
max-width: 78px;
|
||||||
|
&:hover a{
|
||||||
|
color:#2A61FF !important ;
|
||||||
|
}
|
||||||
|
}
|
||||||
.menuPanels{
|
.menuPanels{
|
||||||
width: 240px;
|
width: 295px;
|
||||||
height: 180px;
|
.leftline{
|
||||||
|
position: relative;
|
||||||
|
color: #666;
|
||||||
|
height: 16px;
|
||||||
|
margin-left: 14px;
|
||||||
|
font-size: 12px;
|
||||||
|
&::before{
|
||||||
|
position: absolute;
|
||||||
|
left: -7px;
|
||||||
|
top:3px;
|
||||||
|
height: 12px;
|
||||||
|
width: 1px;
|
||||||
|
background-color: #999;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.ant-btn{
|
||||||
|
height: 36px;
|
||||||
|
line-height: 34px;
|
||||||
|
width: 83px;
|
||||||
|
text-align: center;
|
||||||
|
padding:0px ;
|
||||||
|
font-weight: 500;
|
||||||
|
font-size: 14px;
|
||||||
|
&.currentBtn{
|
||||||
|
cursor: default;
|
||||||
|
color: #333;
|
||||||
|
&:hover{
|
||||||
|
color: #333;
|
||||||
|
border-color: #d0d0d0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.ant-btn-default{
|
||||||
|
color: #333;
|
||||||
|
border-color: #d0d0d0;
|
||||||
|
&:hover{
|
||||||
|
background: #F3F4F6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.ant-btn{
|
||||||
|
width: 102px;
|
||||||
|
height: 32px;
|
||||||
|
line-height: 30px;
|
||||||
|
}
|
||||||
|
.ant-btn-primary{
|
||||||
|
color: #fff;
|
||||||
|
background-color: #466AFF;
|
||||||
|
border:none;
|
||||||
|
&:hover{
|
||||||
|
background-color: rgba(70,106,255,0.85);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.focusPanelHeadInfo{
|
||||||
|
padding:14px 16px;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
.ant-popover-content,.ant-popover-inner{
|
.ant-popover-content,.ant-popover-inner{
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
.ant-popover-inner-content{
|
||||||
|
padding:0px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.halfs{
|
.halfs{
|
||||||
margin-top: 24px;
|
margin-top: 24px;
|
||||||
padding:24px 0px 0px 0px;
|
padding:24px 0px 0px 0px;
|
||||||
border-top: 1px solid #e8e8e8;
|
border-top: 1px solid #e8e8e8;
|
||||||
.attrPerson{
|
}
|
||||||
padding-bottom: 24px;
|
.aboutSubTitle{
|
||||||
}
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.menuMaininfos{
|
||||||
|
padding:10px 16px 14px;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
}
|
}
|
||||||
.menuinfos{
|
.menuinfos{
|
||||||
padding:15px 0px;
|
padding:10px 20px 16px;
|
||||||
&>a{
|
&>a{
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
border-right: 1px solid #eee;
|
|
||||||
flex: 1;
|
flex: 1;
|
||||||
& >span:first-child{
|
& >span:first-child{
|
||||||
font-size: 18px;
|
font-size: 16px;
|
||||||
font-weight: 400;
|
font-weight: 500;
|
||||||
color: #333;
|
color: #333;
|
||||||
|
line-height: 22px;
|
||||||
}
|
}
|
||||||
& >span:last-child{
|
& >span:last-child{
|
||||||
color: #666;
|
color: #666;
|
||||||
}
|
font-weight: 400;
|
||||||
&:last-child{
|
line-height: 20px;
|
||||||
border-right: none;
|
margin-top: 6px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*-------------------个人主页:右侧提示区域--------------------------*/
|
||||||
|
.-task-sidebar {
|
||||||
|
position: fixed;
|
||||||
|
width: 40px;
|
||||||
|
right: 0;
|
||||||
|
bottom: 80px;
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 1920px){
|
||||||
|
.-task-sidebar{
|
||||||
|
right:220px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@media screen and (max-width: 1750px){
|
||||||
|
.-task-sidebar{
|
||||||
|
right:160px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@media screen and (max-width: 1650px){
|
||||||
|
.-task-sidebar{
|
||||||
|
right:115px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@media screen and (max-width: 1550px){
|
||||||
|
.-task-sidebar{
|
||||||
|
right:90px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@media screen and (max-width: 1450px){
|
||||||
|
.-task-sidebar{
|
||||||
|
right:45px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@media screen and (max-width: 1200px){
|
||||||
|
.-task-sidebar{
|
||||||
|
right:0px;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.-task-sidebar>div {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
width: 40px;
|
||||||
|
color: #999;
|
||||||
|
font-size: 20px;
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: #FFFFFF;
|
||||||
|
box-shadow: 0px 0px 10px 1px #F1F1F1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.-task-sidebar>div i {
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.-task-sidebar>div:hover i {
|
||||||
|
color: #fff !important;
|
||||||
|
}
|
||||||
|
.-task-sidebar>div:hover{
|
||||||
|
background: #1890FF;
|
||||||
|
box-shadow: 0px 0px 10px 2px #B6D0FC;
|
||||||
|
}
|
||||||
|
.helpBox{
|
||||||
|
width: 260px;
|
||||||
|
z-index: 103;
|
||||||
|
&.shareContent{
|
||||||
|
width: 200px;
|
||||||
|
}
|
||||||
|
.ant-popover-inner-content{
|
||||||
|
padding:0px;
|
||||||
|
}
|
||||||
|
p.titlecontent{
|
||||||
|
font-size: 18px;
|
||||||
|
color: #333;
|
||||||
|
line-height: 20px;
|
||||||
|
padding:15px 20px;
|
||||||
|
}
|
||||||
|
.faqUl{
|
||||||
|
padding:0px 20px 10px;
|
||||||
|
max-height: 230px;
|
||||||
|
overflow-y: auto;
|
||||||
|
li{
|
||||||
|
background: #F5F5F5;
|
||||||
|
border-radius: 20px;
|
||||||
|
padding:0px 20px;
|
||||||
|
color: #333;
|
||||||
|
height: 34px;
|
||||||
|
line-height: 34px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
a{
|
||||||
|
display: block;
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
&:hover{
|
||||||
|
background-color: #D1E9FF;
|
||||||
|
a{
|
||||||
|
color: #333!important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.shareUl{
|
||||||
|
padding:10px 0px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
.titlecontent{
|
||||||
|
margin-right: 20px;
|
||||||
|
}
|
||||||
|
li > i{
|
||||||
|
font-size: 32px!important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
.-task-desc {
|
||||||
|
background: #494949;
|
||||||
|
width: 90px;
|
||||||
|
line-height: 36px;
|
||||||
|
text-align: center;
|
||||||
|
position: absolute;
|
||||||
|
color: #fff;
|
||||||
|
font-size: 13px;
|
||||||
|
z-index: 999999;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.-task-desc div {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: -7px;
|
||||||
|
height: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.-task-desc div img {
|
||||||
|
float: left
|
||||||
|
}
|
||||||
|
|
||||||
|
.-task-sidebar .scan_ewm {
|
||||||
|
position: absolute !important;
|
||||||
|
right: 45px !important;
|
||||||
|
bottom: 0px !important;
|
||||||
|
background-color: #494949 !important;
|
||||||
|
-webkit-box-sizing: border-box !important;
|
||||||
|
box-sizing: border-box !important;
|
||||||
|
font-size: 14px !important;
|
||||||
|
line-height: 16px !important;
|
||||||
|
display: none;
|
||||||
|
height: 213px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.trangle_right {
|
||||||
|
position: absolute;
|
||||||
|
right: -5px;
|
||||||
|
bottom: 15px;
|
||||||
|
width: 0;
|
||||||
|
height: 0px;
|
||||||
|
border-top: 6px solid transparent;
|
||||||
|
border-left: 5px solid #494949;
|
||||||
|
border-bottom: 6px solid transparent
|
||||||
|
}
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { AlignCenter , FlexAJ } from '../Component/layout';
|
import { AlignCenter , FlexAJ } from '../Component/layout';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
import { Popover , Spin } from 'antd';
|
import { Popover , Spin , Button } from 'antd';
|
||||||
import { getImageUrl } from 'educoder';
|
import { getImageUrl } from 'educoder';
|
||||||
import './Component.scss';
|
import './Component.scss';
|
||||||
import { getUser } from '../GetData/getData';
|
import { getUser } from '../GetData/getData';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
|
|
||||||
function Contributors({contributors,owner,projectsId}){
|
function Contributors({contributors,owner,projectsId,currentLogin}){
|
||||||
const [ menuList ,setMenuList ]= useState([]);
|
const [ menuList ,setMenuList ]= useState([]);
|
||||||
const [ list , setList ]= useState(undefined);
|
const [ list , setList ]= useState(undefined);
|
||||||
const [ total , setTotal ]= useState(0);
|
const [ total , setTotal ]= useState(0);
|
||||||
|
@ -46,46 +46,60 @@ function Contributors({contributors,owner,projectsId}){
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function renderOrganize(list) {
|
||||||
|
let str = "";
|
||||||
|
list.map(i=>{
|
||||||
|
str = str+i.name + "、";
|
||||||
|
})
|
||||||
|
return str && str.substr(0,str.length - 1);
|
||||||
|
}
|
||||||
|
|
||||||
function setMenusFunc(data){
|
function setMenusFunc(data){
|
||||||
if(data){
|
if(data){
|
||||||
let ele = (
|
let ele = (
|
||||||
<Spin spinning={isSpin}>
|
<Spin spinning={isSpin}>
|
||||||
<FlexAJ>
|
<FlexAJ className="menuMaininfos">
|
||||||
<AlignCenter>
|
<AlignCenter>
|
||||||
<Link to={`/users/${data.login}`}><img src={getImageUrl(`/${data.image_url}`)} alt="" className="radius" width="38px" height="38px"/></Link>
|
<Link to={`/${data.login}`}><img src={getImageUrl(`/${data.image_url}`)} alt="" className="radius" width="38px" height="38px"/></Link>
|
||||||
<Link to={`/users/${data.login}`} className="ml10">{data.name}</Link>
|
|
||||||
</AlignCenter>
|
<div className="ml10">
|
||||||
{
|
<Link to={`/${data.login}`}>{data.name}</Link>
|
||||||
data.is_watch ? <a className="color-grey-9" onClick={()=>FocusFunc(false,data.login)}>取消关注</a>:<a className="color-blue" onClick={()=>FocusFunc(true,data.login)}>关注</a>
|
{ data.location && <span className="leftline">{data.location}</span> }
|
||||||
}
|
{
|
||||||
</FlexAJ>
|
data.organizations && data.organizations.length>0&&
|
||||||
<AlignCenter className="menuinfos">
|
<p className="task-hide" style={{maxWidth:"215px"}}>
|
||||||
<a href={data.projects_url}>
|
所属组织:{renderOrganize(data.organizations)}
|
||||||
<span>{data.projects_count}</span>
|
</p>
|
||||||
<span>项目数</span>
|
}
|
||||||
</a>
|
|
||||||
<a href={data.followers_url}>
|
|
||||||
<span>{data.followers_count}</span>
|
|
||||||
<span>粉丝数</span>
|
|
||||||
</a>
|
|
||||||
<a href={data.following_url}>
|
|
||||||
<span>{data.following_count}</span>
|
|
||||||
<span>关注数</span>
|
|
||||||
</a>
|
|
||||||
</AlignCenter>
|
|
||||||
{
|
|
||||||
data.organizations && data.organizations.length > 0 ?
|
|
||||||
<AlignCenter className="font-12 pt4 pb4">
|
|
||||||
<span>所属组织:</span>
|
|
||||||
<div className="task-hide flex1">
|
|
||||||
{renderArray(data.organizations)}
|
|
||||||
</div>
|
</div>
|
||||||
</AlignCenter>
|
</AlignCenter>
|
||||||
:""
|
</FlexAJ>
|
||||||
}
|
<AlignCenter className="menuinfos">
|
||||||
{
|
<Link to={`/${data.login}/projects`}>
|
||||||
data.location && <AlignCenter className="font-12 pt4 pb4"><span>所在地址:</span><span className="ml5">{data.location}</span></AlignCenter>
|
<span>{data.projects_count}</span>
|
||||||
}
|
<span>项目数</span>
|
||||||
|
</Link>
|
||||||
|
<Link to={`/${data.login}/followers`}>
|
||||||
|
<span>{data.followers_count}</span>
|
||||||
|
<span>粉丝数</span>
|
||||||
|
</Link>
|
||||||
|
<Link to={`/${data.login}/following`}>
|
||||||
|
<span>{data.following_count}</span>
|
||||||
|
<span>关注数</span>
|
||||||
|
</Link>
|
||||||
|
</AlignCenter>
|
||||||
|
<div className={"pb20"} style={{display:"flex",justifyContent:'center'}}>
|
||||||
|
{
|
||||||
|
currentLogin && (currentLogin === data.login)
|
||||||
|
?
|
||||||
|
<Button className="currentBtn">当前用户</Button>
|
||||||
|
:
|
||||||
|
data.is_watch ?
|
||||||
|
<Button type={"default"} onClick={()=>FocusFunc(false,data.login)}>已关注</Button>
|
||||||
|
:
|
||||||
|
<Button type={"primary"} onClick={()=>FocusFunc(true,data.login)}>关注TA</Button>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
</Spin>
|
</Spin>
|
||||||
)
|
)
|
||||||
setMenu(ele);
|
setMenu(ele);
|
||||||
|
@ -135,17 +149,17 @@ function Contributors({contributors,owner,projectsId}){
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<div className="halfs">
|
<div className="halfs">
|
||||||
<FlexAJ>
|
<Link to={`/${owner}/${projectsId}/contribute`} className="font-16 color-ooo hoverA">
|
||||||
<AlignCenter><span className="font-16 color-grey-6">贡献者</span>{ contributors && contributors.total_count > 0 && <span className="infoCount">{contributors.total_count}</span>}</AlignCenter>
|
<span>贡献者</span>
|
||||||
<Link className="font-12 color-grey-9" to={`/projects/${owner}/${projectsId}/contribute`}>全部</Link>
|
{ contributors && contributors.total_count > 0 && <span className="infoCount">{contributors.total_count}</span>}
|
||||||
</FlexAJ>
|
</Link>
|
||||||
<div className="attrPerson" onMouseLeave={()=>setVisibleFunc(false)}>
|
<div className="attrPerson" onMouseLeave={()=>setVisibleFunc(false)}>
|
||||||
{
|
{
|
||||||
total > 0 ?
|
total > 0 ?
|
||||||
list.map((item,key)=>{
|
list.map((item,key)=>{
|
||||||
return(
|
return(
|
||||||
<Popover content={menu} visible={item.visible} overlayClassName="menuPanels" placement="top">
|
<Popover content={menu} visible={item.visible} overlayClassName="menuPanels" placement="top">
|
||||||
<Link key={key} to={`/users/${item.login}`}>
|
<Link key={key} to={`/${item.login}`}>
|
||||||
<img src={getImageUrl(`/${item.image_url}`)} alt="" onMouseOver={()=>setVisibleFunc(true,item.login,key)}/>
|
<img src={getImageUrl(`/${item.image_url}`)} alt="" onMouseOver={()=>setVisibleFunc(true,item.login,key)}/>
|
||||||
</Link>
|
</Link>
|
||||||
</Popover>
|
</Popover>
|
||||||
|
|
|
@ -0,0 +1,51 @@
|
||||||
|
import React, { useState, useCallback, memo } from 'react';
|
||||||
|
import { Tooltip } from 'antd';
|
||||||
|
|
||||||
|
CopyTool.defaultProps = {
|
||||||
|
beforeText: '复制链接', //浮动过去显示的文字
|
||||||
|
afterText: '复制成功', //点击后显示的文字
|
||||||
|
className: '', //传给svg的class
|
||||||
|
inputId: 'copyText', //要复制的文本的ID
|
||||||
|
timeOut:true, //复制后将浮动的文字改为beforeText
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
function CopyTool({ beforeText, afterText, className , inputId , timeOut }) {
|
||||||
|
const [title, setTitle] = useState(() => {
|
||||||
|
return beforeText;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 复制链接
|
||||||
|
const copyUrl = useCallback(() => {
|
||||||
|
const copyEle = document.querySelector(`#${inputId}`); // 获取要复制的节点
|
||||||
|
if (!copyEle) {
|
||||||
|
console.error("您的CopyTool未设置正确的inputId");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
copyEle.select(); // 执行选中元素
|
||||||
|
if (document.execCommand('copy')) {
|
||||||
|
document.execCommand('copy');
|
||||||
|
}
|
||||||
|
document.getSelection().removeAllRanges();
|
||||||
|
|
||||||
|
setTitle(afterText);
|
||||||
|
if(timeOut){
|
||||||
|
setTimeout(function(){
|
||||||
|
setTitle(beforeText);
|
||||||
|
},1500)
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tooltip
|
||||||
|
placement="top"
|
||||||
|
title={title}
|
||||||
|
onVisibleChange={() => { setTitle(beforeText) }}
|
||||||
|
>
|
||||||
|
<i className={`iconfont icon-fuzhiicon ${className}`} style={{ color: '#466aff' }} onClick={copyUrl}></i>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export default memo(CopyTool);
|
|
@ -0,0 +1,33 @@
|
||||||
|
import React from 'react';
|
||||||
|
import { AlignCenter } from '../layout';
|
||||||
|
import { Button } from 'antd';
|
||||||
|
import Modals from '../PublicModal/Index';
|
||||||
|
|
||||||
|
function DeleteBox({
|
||||||
|
visible ,
|
||||||
|
onCancel ,
|
||||||
|
onSuccess ,
|
||||||
|
title ,
|
||||||
|
subTitle,
|
||||||
|
content
|
||||||
|
}) {
|
||||||
|
return(
|
||||||
|
<Modals
|
||||||
|
title={title}
|
||||||
|
btn={
|
||||||
|
<div>
|
||||||
|
<Button size={'large'} onClick={onCancel}>取消</Button>
|
||||||
|
<Button type={"danger"} size={"large"} onClick={onSuccess}>确认删除</Button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
onCancel={onCancel}
|
||||||
|
visible={visible}
|
||||||
|
>
|
||||||
|
<div className="desc">
|
||||||
|
<AlignCenter className="descMain"><i className="iconfont icon-shanchu_tc_icon mr10"></i>{content}</AlignCenter>
|
||||||
|
<p>{subTitle}</p>
|
||||||
|
</div>
|
||||||
|
</Modals>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default DeleteBox;
|
|
@ -4,6 +4,12 @@ import './Component.scss';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
const { TreeNode , DirectoryTree } = Tree;
|
const { TreeNode , DirectoryTree } = Tree;
|
||||||
|
|
||||||
|
function turnbar(str){
|
||||||
|
if(str && str.length>0 && str.indexOf("/")>-1){
|
||||||
|
return str.replaceAll('/','%2F');
|
||||||
|
}
|
||||||
|
return str;
|
||||||
|
}
|
||||||
function DrawerPanel({visible,onClose,branch,owner,projectsId,history, name , list}){
|
function DrawerPanel({visible,onClose,branch,owner,projectsId,history, name , list}){
|
||||||
const [ treeData , setTreeData ] = useState(undefined);
|
const [ treeData , setTreeData ] = useState(undefined);
|
||||||
const [ isSpin , setIsSpin ] = useState(true);
|
const [ isSpin , setIsSpin ] = useState(true);
|
||||||
|
@ -71,7 +77,8 @@ function DrawerPanel({visible,onClose,branch,owner,projectsId,history, name , li
|
||||||
let dataref = event.node.props.dataRef;
|
let dataref = event.node.props.dataRef;
|
||||||
if(dataref.type==="file"){
|
if(dataref.type==="file"){
|
||||||
onClose();
|
onClose();
|
||||||
history.push(`/projects/${owner}/${projectsId}/tree/${branch}/${dataref.path}`);
|
let value = turnbar(branch);
|
||||||
|
history.push(`/${owner}/${projectsId}/tree/${value}/${dataref.path}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
.ant-modal-mask{
|
||||||
|
z-index: 1031;
|
||||||
|
}
|
||||||
|
.ant-modal-wrap{
|
||||||
|
z-index: 1032;
|
||||||
|
}
|
|
@ -0,0 +1,60 @@
|
||||||
|
import React , {forwardRef, useEffect} from 'react';
|
||||||
|
import { Modal , Form , Input , Button } from 'antd';
|
||||||
|
import './EAccount.scss';
|
||||||
|
|
||||||
|
function EducoderAccount({form , visible , onOk , email}){
|
||||||
|
const { getFieldDecorator, validateFields , setFieldsValue } = form;
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
if(email){
|
||||||
|
setFieldsValue({email})
|
||||||
|
}
|
||||||
|
},[email])
|
||||||
|
|
||||||
|
function onSure(){
|
||||||
|
validateFields((error,values)=>{
|
||||||
|
if(!error){
|
||||||
|
onOk(values);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
const layout = {
|
||||||
|
labelCol: { span: 5 },
|
||||||
|
wrapperCol: { span: 18 },
|
||||||
|
};
|
||||||
|
return(
|
||||||
|
<Modal
|
||||||
|
visible={visible}
|
||||||
|
title="提示"
|
||||||
|
width="500px"
|
||||||
|
closable={false}
|
||||||
|
footer={
|
||||||
|
<Button type="primary" onClick={onSure}>确定</Button>
|
||||||
|
}
|
||||||
|
centered
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<p className="mb15 edu-txt-center" style={{maxWidth:"350px",margin:"0px auto"}}>
|
||||||
|
为确保您能正常使用平台功能,请确认以下信息:
|
||||||
|
</p>
|
||||||
|
<Form {...layout}>
|
||||||
|
<Form.Item label="邮箱">
|
||||||
|
{getFieldDecorator("email",{
|
||||||
|
rules:[{required:true,message:"请输入邮箱账号"}]
|
||||||
|
})(
|
||||||
|
<Input placeholder="请输入您的邮箱账号" width="220px"/>
|
||||||
|
)}
|
||||||
|
</Form.Item>
|
||||||
|
<Form.Item label="密码">
|
||||||
|
{getFieldDecorator("password",{
|
||||||
|
rules:[{required:true,message:"请输入邮箱密码"}]
|
||||||
|
})(
|
||||||
|
<Input.Password placeholder="请输入您的邮箱密码" width="220px"/>
|
||||||
|
)}
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default Form.create()(forwardRef(EducoderAccount));
|
|
@ -0,0 +1,41 @@
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { Input ,notification} from "antd";
|
||||||
|
|
||||||
|
const { Search } = Input;
|
||||||
|
export default ({history}) => {
|
||||||
|
const [openSearch, setOpenSearch] = useState(false);
|
||||||
|
|
||||||
|
function onGlobalSearch(value) {
|
||||||
|
history.push('/search?value=' + value);
|
||||||
|
// window.location.href = `search?value=` + value;
|
||||||
|
// history.push({
|
||||||
|
// pathname:'/search',
|
||||||
|
// state:value
|
||||||
|
// })
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<React.Fragment>
|
||||||
|
{
|
||||||
|
openSearch ?
|
||||||
|
<div
|
||||||
|
onBlur={() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
setOpenSearch(false)
|
||||||
|
}, 500)
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Search placeholder="请输入搜索关键字"
|
||||||
|
className={`search-input mr20`}
|
||||||
|
onSearch={onGlobalSearch}
|
||||||
|
autoFocus={true}
|
||||||
|
style={{width:'260px'}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
:
|
||||||
|
<i className="iconfont icon-sousuo font-18 color-grey-6 ml30" onClick={() => {
|
||||||
|
setOpenSearch(true)
|
||||||
|
}} />
|
||||||
|
}
|
||||||
|
</React.Fragment>
|
||||||
|
)
|
||||||
|
};
|
|
@ -25,7 +25,7 @@ function LanguagePower({languages}){
|
||||||
}
|
}
|
||||||
return(
|
return(
|
||||||
<div>
|
<div>
|
||||||
<p className="font-16 color-grey-6">开发语言</p>
|
<p className="font-16 color-ooo aboutSubTitle">开发语言</p>
|
||||||
<div className="progress">
|
<div className="progress">
|
||||||
{
|
{
|
||||||
array && array.map((item,key)=>{
|
array && array.map((item,key)=>{
|
||||||
|
|
|
@ -45,15 +45,15 @@ const Div = styled.div`{
|
||||||
export default (({ user , img, name, time, focusStatus, is_current_user, login , successFunc }) => {
|
export default (({ user , img, name, time, focusStatus, is_current_user, login , successFunc }) => {
|
||||||
return (
|
return (
|
||||||
<Div>
|
<Div>
|
||||||
<Link to={`/users/${user && user.login}`}><Img src={getImageUrl(`/${img}`)} /></Link>
|
<Link to={`/${user && user.login}`}><Img src={getImageUrl(`/${img}`)} /></Link>
|
||||||
<div className="m-infos">
|
<div className="m-infos">
|
||||||
<Link to={`/users/${user && user.login}`}><Name>{name}</Name></Link>
|
<Link to={`/${user && user.login}`}><Name>{name}</Name></Link>
|
||||||
<Time><I className="iconfont icon-shijian"></I>加入时间:{time}</Time>
|
<Time><I className="iconfont icon-shijian"></I>加入时间:{time}</Time>
|
||||||
{
|
{
|
||||||
is_current_user ?
|
is_current_user ?
|
||||||
<Button type="default">当前用户</Button>
|
<Button type="default">当前用户</Button>
|
||||||
:
|
:
|
||||||
<FocusButton is_watch={focusStatus} id={login} successFunc={successFunc}/>
|
<FocusButton is_watch={focusStatus} id={login} successFunc={successFunc} notReset={true}/>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</Div>
|
</Div>
|
||||||
|
|
|
@ -0,0 +1,125 @@
|
||||||
|
/* eslint-disable react/jsx-no-duplicate-props */
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
import * as ReactDOM from 'react-dom';
|
||||||
|
import { Modal, Button } from 'antd';
|
||||||
|
import './index.scss';
|
||||||
|
|
||||||
|
// 函数式调用删除、通知等模态框
|
||||||
|
|
||||||
|
InitModal.defaultProps = {
|
||||||
|
okText: '确认', //确定按钮的文字
|
||||||
|
cancelText: '取消', //取消按钮的文字
|
||||||
|
className: '', //传入的模态框类名
|
||||||
|
inputId: 'copyText', //要复制的文本的ID
|
||||||
|
onCancel:()=>{}, //取消的回调
|
||||||
|
onOk:()=>{}, //确认的回调
|
||||||
|
title:'提示', //模态框名字
|
||||||
|
contentTitle:'', //内容标题
|
||||||
|
content:'', //详细内容
|
||||||
|
afterClose:()=>{}, //关闭模态框以后的回调
|
||||||
|
};
|
||||||
|
|
||||||
|
// 使用函数调用删除组件
|
||||||
|
export default function DelModal(props) {
|
||||||
|
renderModal({ ...props, type: 'delete' })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 使用函数调用选择模态框组件
|
||||||
|
export function Confirm(props) {
|
||||||
|
renderModal({ ...props, type: 'confirm' })
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderModal(props) {
|
||||||
|
const { type, afterClose } = props;
|
||||||
|
const div = document.createElement('div');
|
||||||
|
document.body.appendChild(div);
|
||||||
|
|
||||||
|
function destroy() {
|
||||||
|
afterClose && afterClose();
|
||||||
|
const unmountResult = ReactDOM.unmountComponentAtNode(div);
|
||||||
|
if (unmountResult && div.parentNode) {
|
||||||
|
div.parentNode.removeChild(div);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function modalType(type) {
|
||||||
|
if (type === 'delete') {
|
||||||
|
return <InitModal
|
||||||
|
title="删除"
|
||||||
|
contentTitle="确定要删除吗?"
|
||||||
|
okText="确认删除"
|
||||||
|
{...props}
|
||||||
|
|
||||||
|
afterClose={destroy}
|
||||||
|
contentTitle={<React.Fragment>
|
||||||
|
<i className="red-circle iconfont icon-shanchu_tc_icon mr3"></i>
|
||||||
|
{props.contentTitle}
|
||||||
|
</React.Fragment>}
|
||||||
|
/>
|
||||||
|
} else if (type === 'confirm') {
|
||||||
|
return <InitModal title="选择" afterClose={destroy} {...props} />
|
||||||
|
} else {
|
||||||
|
return <InitModal title="选择" afterClose={destroy} {...props} />
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function render() {
|
||||||
|
setTimeout(() => {
|
||||||
|
ReactDOM.render(
|
||||||
|
modalType(type),
|
||||||
|
div,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
render();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 选择模态框组件
|
||||||
|
function InitModal({
|
||||||
|
onCancel,
|
||||||
|
onOk,
|
||||||
|
title,
|
||||||
|
contentTitle,
|
||||||
|
content,
|
||||||
|
okText,
|
||||||
|
cancelText,
|
||||||
|
afterClose,
|
||||||
|
className,
|
||||||
|
}) {
|
||||||
|
|
||||||
|
const [visible, setVisible] = useState(true);
|
||||||
|
|
||||||
|
function onCancelModal() {
|
||||||
|
setVisible(false);
|
||||||
|
onCancel && onCancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
function onSuccess() {
|
||||||
|
setVisible(false);
|
||||||
|
onOk && onOk();
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
visible={visible}
|
||||||
|
onCancel={onCancelModal}
|
||||||
|
afterClose={afterClose}
|
||||||
|
title={title}
|
||||||
|
className={`myself-modal ${className}`}
|
||||||
|
centered
|
||||||
|
footer={[
|
||||||
|
<Button type="default" key="back" onClick={onCancelModal}>
|
||||||
|
{cancelText}
|
||||||
|
</Button>,
|
||||||
|
<Button className="foot-submit" key="submit" onClick={onSuccess}>
|
||||||
|
{okText}
|
||||||
|
</Button>,
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
{contentTitle && <p className="content-title">{contentTitle}</p>}
|
||||||
|
<p className="content-descibe">{content}</p>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
)
|
||||||
|
}
|
|
@ -0,0 +1,63 @@
|
||||||
|
.myself-modal {
|
||||||
|
.ant-modal-header {
|
||||||
|
padding: 9px 24px;
|
||||||
|
background: #f8f8f8;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
.ant-modal-title {
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
.ant-modal-close {
|
||||||
|
top: 0px !important;
|
||||||
|
}
|
||||||
|
.ant-modal-close-x {
|
||||||
|
font-size: 24px;
|
||||||
|
}
|
||||||
|
.ant-modal-body {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.content-title {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
margin: 2rem 0 1rem !important;
|
||||||
|
font-size: 16px;
|
||||||
|
color: #333;
|
||||||
|
letter-spacing: 0;
|
||||||
|
line-height: 29px;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
.red-circle {
|
||||||
|
align-self: flex-start;
|
||||||
|
color: #ca0002;
|
||||||
|
font-size: 1.5rem !important;
|
||||||
|
}
|
||||||
|
.content-descibe {
|
||||||
|
font-size: 14px;
|
||||||
|
color: #666;
|
||||||
|
line-height: 33px;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
.ant-modal-footer {
|
||||||
|
padding: 2rem 0;
|
||||||
|
text-align: center;
|
||||||
|
border: 0;
|
||||||
|
.ant-btn {
|
||||||
|
width: 6rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.foot-submit {
|
||||||
|
margin-left: 3rem;
|
||||||
|
color: #df0002;
|
||||||
|
&:hover {
|
||||||
|
border-color: #df0002;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.ant-btn-default:hover,
|
||||||
|
.ant-btn-default:active,
|
||||||
|
.ant-btn-default:focus {
|
||||||
|
background: #f3f4f6;
|
||||||
|
color: #333;
|
||||||
|
border-color: #d0d0d0;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,62 @@
|
||||||
|
import React, {useEffect, useRef, useState} from 'react';
|
||||||
|
import * as monaco from 'monaco-editor/esm/vs/editor/editor.api.js';
|
||||||
|
import './Component.scss';
|
||||||
|
|
||||||
|
function Monaco(props) {
|
||||||
|
const {
|
||||||
|
style = { // dom节点样式
|
||||||
|
height: '400px',
|
||||||
|
},
|
||||||
|
value = '', // 代码文本
|
||||||
|
onChange = () => { // 改变的事件
|
||||||
|
},
|
||||||
|
fontSize = 14, // 代码字体大小
|
||||||
|
monacoOptions = {
|
||||||
|
scrollBeyondLastLine: false,
|
||||||
|
lineNumbers: "off",
|
||||||
|
wordWrap: true,
|
||||||
|
overviewRulerBorder: true,
|
||||||
|
lineHeight: 24,
|
||||||
|
readOnly:true
|
||||||
|
}, // monaco 自定义属性
|
||||||
|
language = 'html', // 语言 支持 js ts sql css json html等
|
||||||
|
} = props;
|
||||||
|
const editOrRef = useRef();
|
||||||
|
const ThisEditor = useRef();
|
||||||
|
useEffect(() => {
|
||||||
|
ThisEditor.current = monaco.editor.create(editOrRef.current, {
|
||||||
|
value: value || '',
|
||||||
|
language,
|
||||||
|
theme: "vs-grey",
|
||||||
|
fontSize: fontSize + 'px',
|
||||||
|
minimap: { // 关闭代码缩略图
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
...monacoOptions,
|
||||||
|
});
|
||||||
|
|
||||||
|
ThisEditor.current.onDidChangeModelContent((e) => {
|
||||||
|
let newValue = ThisEditor.current.getValue();
|
||||||
|
onChange(newValue);
|
||||||
|
});
|
||||||
|
return () => {
|
||||||
|
ThisEditor.current.dispose();
|
||||||
|
ThisEditor.current = undefined; // 清除编辑器对象
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
useEffect(() => {
|
||||||
|
if (ThisEditor.current) {
|
||||||
|
ThisEditor.current.updateOptions({
|
||||||
|
fontSize: fontSize + 'px',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}, [fontSize]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={style} ref={editOrRef}>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Monaco;
|
|
@ -0,0 +1,90 @@
|
||||||
|
.systemBox{
|
||||||
|
.ant-modal-body{
|
||||||
|
padding:1px 0px 0px 0px;
|
||||||
|
.sysBox{
|
||||||
|
background-image: url('./bg.png');
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: 100% 334px;
|
||||||
|
margin-top: -55px;
|
||||||
|
}
|
||||||
|
.sysnoticeBox{
|
||||||
|
width: 100%;
|
||||||
|
padding:80px 0px 34px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
width: 780px;
|
||||||
|
margin: 0px auto;
|
||||||
|
p.ntitle{
|
||||||
|
height: 33px;
|
||||||
|
font-size: 24px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #31FFF7;
|
||||||
|
line-height: 33px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
p.nSubtitle{
|
||||||
|
height: 25px;
|
||||||
|
line-height: 25px;
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #FFFFFF;
|
||||||
|
margin-top: 60px;
|
||||||
|
padding-left: 20px;
|
||||||
|
}
|
||||||
|
.markdown-body{
|
||||||
|
box-shadow: 0px 0px 17px rgba(0,0,0,0.2);
|
||||||
|
border-radius: 4px;
|
||||||
|
margin-top: 17px!important;
|
||||||
|
}
|
||||||
|
.nContent{
|
||||||
|
padding:20px 34px;
|
||||||
|
background-color: #fff;
|
||||||
|
line-height: 30px;
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 400;
|
||||||
|
color: #333;
|
||||||
|
.realmName{
|
||||||
|
margin-top: 20px;
|
||||||
|
display: flex;
|
||||||
|
ul{
|
||||||
|
width: 50%;
|
||||||
|
padding-left: 0px!important;
|
||||||
|
li{
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 500;
|
||||||
|
line-height: 32px;
|
||||||
|
text-align: left;
|
||||||
|
color: #000;
|
||||||
|
list-style-type: none!important;
|
||||||
|
&:first-child{
|
||||||
|
color: #E65714;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.nSubdesc{
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 400;
|
||||||
|
color: #000000;
|
||||||
|
line-height: 31px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
.nInfo{
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 400;
|
||||||
|
color: #333333;
|
||||||
|
text-align: right;
|
||||||
|
margin-top: 25px;
|
||||||
|
p{
|
||||||
|
height: 20px;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.nBtn{
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 33px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
import React , { useEffect , useState } from 'react';
|
||||||
|
import { Modal , Button } from 'antd';
|
||||||
|
import './Index.scss';
|
||||||
|
import '../../css/index.scss';
|
||||||
|
import RenderHtml from '../../../components/render-html';
|
||||||
|
import cookie from 'react-cookies';
|
||||||
|
|
||||||
|
function SystemNotice({system_notification,history}){
|
||||||
|
const [ visible , setVisible ] = useState(false);
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
if(system_notification && !cookie.load('notice_stage')){
|
||||||
|
setVisible(true);
|
||||||
|
}
|
||||||
|
},[system_notification,history.location])
|
||||||
|
|
||||||
|
function sureContinue() {
|
||||||
|
cookie.remove('notice_stage');
|
||||||
|
|
||||||
|
let inFifteenMinutes = new Date(new Date().getTime() + 24 * 3600 * 1000);//一天
|
||||||
|
// let inFifteenMinutes = new Date(new Date().getTime() + 60 * 1000);//一分钟
|
||||||
|
cookie.save('notice_stage', true,{ expires: inFifteenMinutes,path:"/" });
|
||||||
|
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
visible = {visible}
|
||||||
|
width="1000px"
|
||||||
|
footer={false}
|
||||||
|
title={false}
|
||||||
|
centered={true}
|
||||||
|
closable={false}
|
||||||
|
wrapClassName={'systemBox'}
|
||||||
|
>
|
||||||
|
<div className="sysBox">
|
||||||
|
<div className="sysnoticeBox">
|
||||||
|
<p className="ntitle">{system_notification && system_notification.subject}</p>
|
||||||
|
<p className="nSubtitle">{system_notification && system_notification.sub_subject}</p>
|
||||||
|
{/* <div className="nContent">
|
||||||
|
<div className="nMaindesc">
|
||||||
|
为了给用户提供更加稳定、优质的服务,我们即将对平台门户首页、平台名称、平台域名进行一次全面升级与变更。原平台名称:Trustie(中文名:确实)将于2021年10月xx日统一更改为Gitlink(中文名:确实开源)。届时平台域名将统一进行更换,更换规则如下
|
||||||
|
</div>
|
||||||
|
<div className="realmName">
|
||||||
|
<ul>
|
||||||
|
<li>原域名:</li>
|
||||||
|
<li>官网顶级域名https://www.trustie.net</li>
|
||||||
|
<li>版本库子域名https://forgeplus.trustie.net</li>
|
||||||
|
<li>论坛子域名https://forum.trustie.net/forums</li>
|
||||||
|
</ul>
|
||||||
|
<ul>
|
||||||
|
<li>更换后域名:</li>
|
||||||
|
<li>官网顶级域名https://www.gitlink.org.cn</li>
|
||||||
|
<li>版本库子域名https://www.git.gitlink.org.cn</li>
|
||||||
|
<li>论坛子域名https://forum.gitlink.org.cn</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div className="nSubdesc">
|
||||||
|
自2021年10月xx日起,旧域名将停止访问。因平台名称与域名变更给您带来的不便,我们深表歉意!非常感谢您一直以来对本平台的信任与支持,我们将一如既往地为您提供优质的服务。 特此通知!
|
||||||
|
</div>
|
||||||
|
<div className="nInfo">
|
||||||
|
<p>Gitlink运营团队</p>
|
||||||
|
<p>2021年10月xx日</p>
|
||||||
|
</div>
|
||||||
|
</div> */}
|
||||||
|
<RenderHtml className="break_word_comments imageLayerParent" value={system_notification && system_notification.content} url={history.location}/>
|
||||||
|
<div className="nBtn">
|
||||||
|
<Button type="primary" className="btnblue" onClick={sureContinue}>确认并继续</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default SystemNotice;
|
Binary file not shown.
After Width: | Height: | Size: 280 KiB |
|
@ -0,0 +1,72 @@
|
||||||
|
import React , { useEffect , useState } from 'react';
|
||||||
|
import Modals from '../PublicModal/Index';
|
||||||
|
import { Button } from 'antd';
|
||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
import ProfileImg from './images/profile.png';
|
||||||
|
import './Index.scss';
|
||||||
|
|
||||||
|
function ProfileModal({visible,onCancel,history}) {
|
||||||
|
const [ modalVis , setModalVis ] = useState(visible);
|
||||||
|
const [ addMemberCheck , setAddMemberCheck ] = useState(false);
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
axios.interceptors.response.use((response) => {
|
||||||
|
if (response && (response.data.status === 411 || response.data.status === 412)) {
|
||||||
|
setModalVis(true);
|
||||||
|
if(response.data.status === 412){
|
||||||
|
setAddMemberCheck(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return response;
|
||||||
|
}, (error) => {
|
||||||
|
});
|
||||||
|
},[])
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
setModalVis(visible);
|
||||||
|
},[visible])
|
||||||
|
|
||||||
|
function onOk(){
|
||||||
|
onCancel();
|
||||||
|
setModalVis(false);
|
||||||
|
setTimeout(function(){
|
||||||
|
window.open(`/settings/profile`,"_blank");
|
||||||
|
},200)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onNo() {
|
||||||
|
onCancel();
|
||||||
|
setModalVis(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return(
|
||||||
|
<Modals
|
||||||
|
title="完善资料"
|
||||||
|
onCancel={onNo}
|
||||||
|
visible={modalVis}
|
||||||
|
btn={
|
||||||
|
addMemberCheck?
|
||||||
|
<div>
|
||||||
|
<Button type={'primary'} size={"large"} onClick={onNo}>好的</Button>
|
||||||
|
</div>
|
||||||
|
:
|
||||||
|
<div>
|
||||||
|
<Button size={"large"} onClick={onNo}>暂不补充</Button>
|
||||||
|
<Button type={'primary'} size={"large"} onClick={onOk}>好的</Button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div className="contents">
|
||||||
|
<img src={ProfileImg} alt=""/>
|
||||||
|
{
|
||||||
|
addMemberCheck ?
|
||||||
|
<p>目标用户个人资料不完整,需提醒目标用户补充资料后以进行后续操作</p>
|
||||||
|
:
|
||||||
|
<p>您目前的个人资料不完整,需要补充资料以进行后续操作。是否前往补充个人信息?</p>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</Modals>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default ProfileModal;
|
|
@ -0,0 +1,18 @@
|
||||||
|
.contents{
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
margin:10px auto 0px;
|
||||||
|
img{
|
||||||
|
margin-right: 13px;
|
||||||
|
width: 44px;
|
||||||
|
}
|
||||||
|
p{
|
||||||
|
line-height: 29px;
|
||||||
|
max-width: 327px;
|
||||||
|
font-size: 16px!important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.font-44{
|
||||||
|
font-size: 44px!important;
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
function Profile({children,sureFunc,showCompeleteDialog , completeProfile, className}) {
|
||||||
|
|
||||||
|
function checkProfile() {
|
||||||
|
if(!completeProfile){
|
||||||
|
showCompeleteDialog && showCompeleteDialog();
|
||||||
|
}else{
|
||||||
|
sureFunc();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return(
|
||||||
|
<a className={className} onClick={checkProfile}>{children}</a>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default Profile;
|
Binary file not shown.
After Width: | Height: | Size: 7.2 KiB |
|
@ -0,0 +1,20 @@
|
||||||
|
import React from 'react';
|
||||||
|
import { Modal } from 'antd';
|
||||||
|
import './Index.scss';
|
||||||
|
|
||||||
|
function Modals({title,children,btn,onCancel,visible}) {
|
||||||
|
return(
|
||||||
|
<Modal
|
||||||
|
visible={visible}
|
||||||
|
onCancel={onCancel}
|
||||||
|
title={title}
|
||||||
|
width={"520px"}
|
||||||
|
footer={btn}
|
||||||
|
centered={true}
|
||||||
|
wrapClassName={"deleteBox"}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Modal>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default Modals;
|
|
@ -0,0 +1,75 @@
|
||||||
|
.deleteBox{
|
||||||
|
z-index: 1033;
|
||||||
|
.ant-modal-close-x{
|
||||||
|
font-size: 17px!important;
|
||||||
|
}
|
||||||
|
.ant-modal-header{
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
padding:10px 30px;
|
||||||
|
.ant-modal-title{
|
||||||
|
text-align: left;
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.ant-modal-close{
|
||||||
|
top:0px !important;
|
||||||
|
font-size: 24px !important;
|
||||||
|
}
|
||||||
|
.ant-modal-body{
|
||||||
|
padding:30px 50px;
|
||||||
|
p{
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 26px;
|
||||||
|
color:#666;
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
.desc{
|
||||||
|
.descMain{
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 20px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
i.red{
|
||||||
|
color:#DF0002;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.ant-modal-footer{
|
||||||
|
border-top: none;
|
||||||
|
text-align: center;
|
||||||
|
padding-bottom: 40px;
|
||||||
|
button,a{
|
||||||
|
width: 96px;
|
||||||
|
height: 32px;
|
||||||
|
margin:0px 20px;
|
||||||
|
font-weight: 400;
|
||||||
|
font-size: 14px;
|
||||||
|
&.ant-btn{
|
||||||
|
border-color: #D0D0D0;
|
||||||
|
color: #666;
|
||||||
|
&:hover,&:active,&:focus{
|
||||||
|
background: #f3f4f6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.ant-btn-danger{
|
||||||
|
background-color: #fff;
|
||||||
|
color: #DF0002;
|
||||||
|
border-color: #D0D0D0;
|
||||||
|
&:hover,&:active,&:focus{
|
||||||
|
border-color: #DF0002;
|
||||||
|
background-color: #fff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.ant-btn.ant-btn-primary{
|
||||||
|
background-color: #466AFF;
|
||||||
|
color: #fff;
|
||||||
|
border-color: #466AFF;
|
||||||
|
&:hover,&:focus,&:active{
|
||||||
|
background-color: rgba(70,106,255,0.85);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,38 +1,35 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { AlignCenter , AlignTop , FlexAJ } from '../Component/layout';
|
import { AlignTop } from '../Component/layout';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
|
||||||
function Releases({owner,projectsId,releaseVersions}){
|
function Releases({ owner, projectsId, releaseVersions, distribution }) {
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<div>
|
<div>
|
||||||
<FlexAJ>
|
<Link to={`/${owner}/${projectsId}/releases`} className="font-16 color-ooo hoverA">
|
||||||
<AlignCenter><span className="font-16 color-grey-6">发行版</span>
|
<span>发行版</span>
|
||||||
{ releaseVersions && releaseVersions.total_count > 0 && <span className="infoCount">{releaseVersions.total_count}</span>}
|
{ releaseVersions && releaseVersions.total_count > 0 && <span className="infoCount">{releaseVersions.total_count}</span>}
|
||||||
</AlignCenter>
|
</Link>
|
||||||
{ releaseVersions && releaseVersions.total_count > 0 ?
|
|
||||||
<Link className="font-12 color-grey-9" to={`/projects/${owner}/${projectsId}/releases`}>全部</Link>
|
|
||||||
:
|
|
||||||
<Link className="font-12 color-blue" to={`/projects/${owner}/${projectsId}/releases/new`}>新建</Link>
|
|
||||||
}
|
|
||||||
</FlexAJ>
|
|
||||||
{
|
{
|
||||||
releaseVersions && releaseVersions.total_count>0 ?
|
releaseVersions && releaseVersions.total_count>0 ?
|
||||||
releaseVersions.list.map((item,key)=>{
|
releaseVersions.list.map((item,key)=>{
|
||||||
return(
|
return(
|
||||||
key === 0 &&<AlignTop className="mt10">
|
key === 0 &&<AlignTop className="mt10">
|
||||||
<i className="iconfont icon-biaoqian3 color-grey-6 font-18 mr10"></i>
|
|
||||||
<div>
|
<div>
|
||||||
<p className="font-16 color-grey-6">
|
<p className="font-16 color-grey-6" style={{display:'flex',alignItems:'center'}}>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/releases`}>{item.name}</Link>
|
{/* 如果是点击最新则发行版列表页只展示最新的一个 */}
|
||||||
|
<Link to={{pathname:`/${owner}/${projectsId}/releases`,query:{turnFromNew:true}}} style={{maxWidth:'200px',overflow: 'hidden',whiteSpace: 'nowrap',textOverflow:'ellipsis'}}>{item.name}</Link>
|
||||||
<span className="font-12 laterest ml5">最新</span>
|
<span className="font-12 laterest ml5">最新</span>
|
||||||
</p>
|
</p>
|
||||||
<p className="color-grey-9 font-13">{item.created_at}</p>
|
<p className="color-grey-3 font-12">{item.created_at}</p>
|
||||||
</div>
|
</div>
|
||||||
</AlignTop>
|
</AlignTop>
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
:""
|
:
|
||||||
|
<div className="mt8">
|
||||||
|
您暂未发布任何版本
|
||||||
|
{distribution && <Link className="color-blue ml20" to={{pathname:`/${owner}/${projectsId}/releases/new`,state:{stable:true}}}>创建新版本</Link>}
|
||||||
|
</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,45 +1,49 @@
|
||||||
import React , { useState } from 'react';
|
import React , { useState , useEffect } from 'react';
|
||||||
import { AutoComplete } from 'antd';
|
import { AutoComplete } from 'antd';
|
||||||
import { getImageUrl } from "educoder";
|
import { getImageUrl } from "educoder";
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
|
|
||||||
const Option = AutoComplete.Option;
|
const Option = AutoComplete.Option;
|
||||||
|
|
||||||
export default ({ getUser })=>{
|
export default ({ getUser , placeholder, width ,value })=>{
|
||||||
|
const [ source , setSource ] = useState(undefined);
|
||||||
const [ searchKey , setSearchKey ] = useState(undefined);
|
const [ searchKey , setSearchKey ] = useState(undefined);
|
||||||
const [ userDataSource , setUserDataSource ] = useState(undefined);
|
|
||||||
|
useEffect(()=>{
|
||||||
|
if(!value){
|
||||||
|
setSearchKey(undefined);
|
||||||
|
}
|
||||||
|
},[value])
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
getUserList();
|
||||||
|
},[searchKey])
|
||||||
|
|
||||||
function getUserList(e){
|
function getUserList(e){
|
||||||
const url = `/users/list.json`;
|
const url = `/users/list.json`;
|
||||||
axios.get(url, {
|
axios.get(url, {
|
||||||
params: {
|
params: {
|
||||||
search: e,
|
search: searchKey,
|
||||||
},
|
},
|
||||||
})
|
}).then((result) => {
|
||||||
.then((result) => {
|
|
||||||
if (result) {
|
if (result) {
|
||||||
setUserDataSource(result.data.users);
|
sourceOptions(result.data.users);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.log(error);
|
console.log(error);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
function changeInputUser(value){
|
|
||||||
setSearchKey(value);
|
|
||||||
getUserList(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
function selectInputUser(id, option){
|
function sourceOptions(userDataSource){
|
||||||
setSearchKey(option.props.value);
|
const s = userDataSource && userDataSource.map((item, key) => {
|
||||||
getUserList(option.props.value);
|
|
||||||
getUser && getUser(id);
|
|
||||||
}
|
|
||||||
const source =
|
|
||||||
userDataSource && userDataSource.map((item, key) => {
|
|
||||||
return (
|
return (
|
||||||
<Option key={key} value={`${item.login}`}>
|
<Option
|
||||||
|
key={key}
|
||||||
|
value={`${item.user_id}`}
|
||||||
|
login={`${item.login}`}
|
||||||
|
name={item.username}
|
||||||
|
>
|
||||||
<img
|
<img
|
||||||
className="user_img radius"
|
className="user_img radius"
|
||||||
width="28"
|
width="28"
|
||||||
|
@ -54,14 +58,31 @@ export default ({ getUser })=>{
|
||||||
</Option>
|
</Option>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
setSource(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
function changeInputUser(e){
|
||||||
|
setSearchKey(e);
|
||||||
|
};
|
||||||
|
|
||||||
|
// 选择用户
|
||||||
|
function selectInputUser(e, option){
|
||||||
|
setSearchKey(option.props.name);
|
||||||
|
getUser(option.props.login);
|
||||||
|
};
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<AutoComplete
|
<div className="addPanel">
|
||||||
dataSource={source}
|
<AutoComplete
|
||||||
value={searchKey}
|
getPopupContainer={trigger => trigger.parentNode}
|
||||||
style={{ width: 300 }}
|
dataSource={source}
|
||||||
onChange={changeInputUser}
|
value={searchKey}
|
||||||
onSelect={selectInputUser}
|
style={{ width: width || 300 }}
|
||||||
placeholder="搜索需要添加的用户..."
|
onChange={changeInputUser}
|
||||||
/>
|
onSelect={selectInputUser}
|
||||||
|
placeholder={placeholder || "搜索需要添加的用户..."}
|
||||||
|
allowClear
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
|
@ -0,0 +1,91 @@
|
||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
import { Popover , Tooltip } from 'antd';
|
||||||
|
import './Component.scss';
|
||||||
|
import axios from 'axios';
|
||||||
|
import ShareModal from './SiderBarShareModal';
|
||||||
|
|
||||||
|
const $ = window.$;
|
||||||
|
|
||||||
|
$(window).scroll(function () {
|
||||||
|
if ($(".gotop").length > 0) {
|
||||||
|
if ($(document).scrollTop() > 0) {
|
||||||
|
$(".-task-sidebar .gotop").show();
|
||||||
|
$(".gotop").click(function () {
|
||||||
|
$("html,body").scrollTop(0);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if ($(document).scrollTop() === 0) {
|
||||||
|
$(".-task-sidebar .gotop").hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function SiderBar() {
|
||||||
|
const [ data , setData ] = useState([]);
|
||||||
|
const [ visible , setVisible ] = useState(false);
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
getFAQ();
|
||||||
|
},[])
|
||||||
|
|
||||||
|
function getFAQ(){
|
||||||
|
const url = `/faqs.json`;
|
||||||
|
axios.get(url).then(result=>{
|
||||||
|
if(result && result.data){
|
||||||
|
setData(result.data);
|
||||||
|
}
|
||||||
|
}).catch(error=>{})
|
||||||
|
}
|
||||||
|
function content(list){
|
||||||
|
return <div>
|
||||||
|
<p className="titlecontent">帮助</p>
|
||||||
|
<ul className="faqUl">
|
||||||
|
{
|
||||||
|
list && list.map((i,k)=>{
|
||||||
|
return(
|
||||||
|
<li key={i.question+k}><a href={`${i.url}`} title={i.question} target="_blank">{i.question}</a></li>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
function shareContent(){
|
||||||
|
return <div>
|
||||||
|
<ul className="shareUl">
|
||||||
|
<p className="titlecontent">分享到</p>
|
||||||
|
<li onClick={()=>setVisible(true)}><i className="iconfont icon-weixin2" style={{color:"#62b900"}}></i></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div className={"-task-sidebar"} >
|
||||||
|
<ShareModal visible={visible} urlValue={window.location.href} onCancel={()=>setVisible(false)}/>
|
||||||
|
{
|
||||||
|
data && data.length > 0 && (data[0] && data[0].question) ?
|
||||||
|
<Popover content={content(data)} overlayClassName="helpBox" placement={"left"}>
|
||||||
|
<div className="feedback">
|
||||||
|
<i className="iconfont icon-bangzhu font-22"></i>
|
||||||
|
</div>
|
||||||
|
</Popover>
|
||||||
|
:""
|
||||||
|
}
|
||||||
|
{/* <div className="scan pr" title="微信扫一扫">
|
||||||
|
<span className="inline erweima"><i className="iconfont icon-erweima color-white font-22 fl"></i></span>
|
||||||
|
</div>*/}
|
||||||
|
<Popover content={shareContent()} overlayClassName="helpBox shareContent" placement={"left"}>
|
||||||
|
<div className="consult">
|
||||||
|
<i className="iconfont icon-fenxiang1"></i>
|
||||||
|
</div>
|
||||||
|
</Popover>
|
||||||
|
<div className="gotop">
|
||||||
|
<Tooltip title="返回顶部" placement={"right"}>
|
||||||
|
<a><i className="iconfont icon-huidaodingbu1"></i></a>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default SiderBar;
|
|
@ -0,0 +1,27 @@
|
||||||
|
import React from 'react';
|
||||||
|
import { Modal } from 'antd';
|
||||||
|
import QRCode from 'qrcode.react';
|
||||||
|
|
||||||
|
function SiderBarShareModal({visible,urlValue,onCancel}) {
|
||||||
|
return(
|
||||||
|
<Modal
|
||||||
|
title={"分享到微信"}
|
||||||
|
visible={visible}
|
||||||
|
width="500px"
|
||||||
|
closable={true}
|
||||||
|
footer={false}
|
||||||
|
onCancel={onCancel}
|
||||||
|
>
|
||||||
|
<div style={{textAlign:"center"}}>
|
||||||
|
{urlValue &&<QRCode
|
||||||
|
value={urlValue}
|
||||||
|
size={200}
|
||||||
|
fgColor="#000000"
|
||||||
|
style={{margin:"20px"}}
|
||||||
|
/>}
|
||||||
|
<p>打开微信“扫一扫”,点击右上角菜单,即可将网页分享至朋友圈</p>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default SiderBarShareModal;
|
|
@ -2,9 +2,9 @@ import React from 'react';
|
||||||
import {Popover} from 'antd';
|
import {Popover} from 'antd';
|
||||||
import './Component.scss';
|
import './Component.scss';
|
||||||
|
|
||||||
export default (({menu , children})=>{
|
export default (({menu , children, overlayClassName})=>{
|
||||||
return(
|
return(
|
||||||
<Popover content={menu} trigger={['click']} placement='bottom'>
|
<Popover content={menu} trigger={['click']} placement='bottom' overlayClassName={overlayClassName}>
|
||||||
{children}
|
{children}
|
||||||
</Popover>
|
</Popover>
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,10 +1,11 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import styled from 'styled-components';
|
import styled from 'styled-components';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
|
||||||
|
export default ({ url , name , column , id , login })=>{
|
||||||
export default ({ url , name , column })=>{
|
|
||||||
const Img = styled.span`
|
const Img = styled.span`
|
||||||
display:flex;
|
display:flex;
|
||||||
|
font-weight: bold;
|
||||||
${column && "flex-direction: column;text-align:center;"}
|
${column && "flex-direction: column;text-align:center;"}
|
||||||
align-items: center;
|
align-items: center;
|
||||||
& img{
|
& img{
|
||||||
|
@ -19,6 +20,14 @@ export default ({ url , name , column })=>{
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
return(
|
return(
|
||||||
|
id?
|
||||||
|
<Link to={`/${login}`}>
|
||||||
|
<Img>
|
||||||
|
{ url && <img src={url} alt=""/> }
|
||||||
|
<span>{name}</span>
|
||||||
|
</Img>
|
||||||
|
</Link>
|
||||||
|
:
|
||||||
<Img>
|
<Img>
|
||||||
{ url && <img src={url} alt=""/> }
|
{ url && <img src={url} alt=""/> }
|
||||||
<span>{name}</span>
|
<span>{name}</span>
|
||||||
|
|
|
@ -36,8 +36,7 @@ function About(props, ref) {
|
||||||
const [ disabled, setDisabled ] = useState(false);
|
const [ disabled, setDisabled ] = useState(false);
|
||||||
const [ typeFlag, setTypeFlag] = useState(false);
|
const [ typeFlag, setTypeFlag] = useState(false);
|
||||||
|
|
||||||
|
const AuthorLogin = props.projectDetail && props.projectDetail.author && props.projectDetail.author.login;
|
||||||
const AuthorLogin = props.author && props.author.login;
|
|
||||||
const CurrentLogin = props.current_user && props.current_user.login;
|
const CurrentLogin = props.current_user && props.current_user.login;
|
||||||
useEffect(()=>{
|
useEffect(()=>{
|
||||||
if(CurrentLogin === AuthorLogin){
|
if(CurrentLogin === AuthorLogin){
|
||||||
|
@ -56,11 +55,6 @@ function About(props, ref) {
|
||||||
setIsSpining(false);
|
setIsSpining(false);
|
||||||
if(result && result.data ){
|
if(result && result.data ){
|
||||||
setStep(result.data.step);
|
setStep(result.data.step);
|
||||||
// setStep(0);
|
|
||||||
// setFieldsValue({...result.data.cloud_account});
|
|
||||||
// if(result.data.cloud_account){
|
|
||||||
// setDisabled(true);
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
}).catch(error=>{
|
}).catch(error=>{
|
||||||
setIsSpining(false);
|
setIsSpining(false);
|
||||||
|
@ -155,7 +149,7 @@ function About(props, ref) {
|
||||||
axios.post(url).then(result=>{
|
axios.post(url).then(result=>{
|
||||||
setIsSpining(false);
|
setIsSpining(false);
|
||||||
if(result && result.data.status === 0){
|
if(result && result.data.status === 0){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/devops/dispose`);
|
props.history.push(`/${owner}/${projectsId}/devops`);
|
||||||
// 需要将顶部的open_devops修改
|
// 需要将顶部的open_devops修改
|
||||||
let { changeOpenDevops } = props;
|
let { changeOpenDevops } = props;
|
||||||
changeOpenDevops && changeOpenDevops(true);
|
changeOpenDevops && changeOpenDevops(true);
|
||||||
|
|
|
@ -96,7 +96,7 @@ function Dispose(props){
|
||||||
setVisible(false);
|
setVisible(false);
|
||||||
if(result && result.data){
|
if(result && result.data){
|
||||||
props.showNotification("流水线新增成功,请进行工作流配置!");
|
props.showNotification("流水线新增成功,请进行工作流配置!");
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/devops/dispose/${result.data.id}`);
|
props.history.push(`/${owner}/${projectsId}/devops/${result.data.id}`);
|
||||||
}else{
|
}else{
|
||||||
props.showNotification("流水线新增失败,请稍后再试!");
|
props.showNotification("流水线新增失败,请稍后再试!");
|
||||||
}
|
}
|
||||||
|
@ -134,12 +134,12 @@ function Dispose(props){
|
||||||
|
|
||||||
// 模板管理
|
// 模板管理
|
||||||
function toModalManage(){
|
function toModalManage(){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/devops/mould`);
|
props.history.push(`/${owner}/${projectsId}/devops/mould`);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 参数管理
|
// 参数管理
|
||||||
function toparameter(){
|
function toparameter(){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/devops/params`);
|
props.history.push(`/${owner}/${projectsId}/devops/params`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const operate = current_user && (permission && permission !== "Reporter");
|
const operate = current_user && (permission && permission !== "Reporter");
|
||||||
|
|
|
@ -10,6 +10,12 @@ import { Link } from 'react-router-dom';
|
||||||
// killed:"已撤销",
|
// killed:"已撤销",
|
||||||
// pending:"准备中"
|
// pending:"准备中"
|
||||||
// }
|
// }
|
||||||
|
function turnbar(str){
|
||||||
|
if(str && str.length>0 && str.indexOf("/")>-1){
|
||||||
|
return str.replaceAll('/','%2F');
|
||||||
|
}
|
||||||
|
return str;
|
||||||
|
}
|
||||||
function renderTableStatus(status) {
|
function renderTableStatus(status) {
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case "running":
|
case "running":
|
||||||
|
@ -65,8 +71,9 @@ function List({ list, operate , projectsId , owner , showModal , deleteFunc }){
|
||||||
width:"15%",
|
width:"15%",
|
||||||
ellipsis:true,
|
ellipsis:true,
|
||||||
render:(value,item)=>{
|
render:(value,item)=>{
|
||||||
|
let v = turnbar(item.branch);
|
||||||
return(
|
return(
|
||||||
<Link to={`/projects/${owner}/${projectsId}/tree/${item.branch}/${value}`} className="color-blue">{value}</Link>
|
<Link to={`/${owner}/${projectsId}/tree/${v}/${value}`} className="color-blue">{value}</Link>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -110,7 +117,7 @@ function List({ list, operate , projectsId , owner , showModal , deleteFunc }){
|
||||||
return(
|
return(
|
||||||
<span>
|
<span>
|
||||||
{ operate ?
|
{ operate ?
|
||||||
<Link to={`/projects/${owner}/${projectsId}/devops/dispose/${item.id}`} className="mr10 color-grey-6">
|
<Link to={`/${owner}/${projectsId}/devops/${item.id}`} className="mr10 color-grey-6">
|
||||||
<i className="iconfont icon-zaibianji font-13 mr3"></i>编辑</Link> :""
|
<i className="iconfont icon-zaibianji font-13 mr3"></i>编辑</Link> :""
|
||||||
}
|
}
|
||||||
{ operate ?
|
{ operate ?
|
||||||
|
@ -118,7 +125,7 @@ function List({ list, operate , projectsId , owner , showModal , deleteFunc }){
|
||||||
<a className="mr10 color-grey-6"><i className="iconfont icon-lajitong font-13 mr3"></i>删除</a>
|
<a className="mr10 color-grey-6"><i className="iconfont icon-lajitong font-13 mr3"></i>删除</a>
|
||||||
</Popconfirm>:""
|
</Popconfirm>:""
|
||||||
}
|
}
|
||||||
<Link to={`/projects/${owner}/${projectsId}/devops/list/${item.branch}`} className="color-grey-6"><i className="iconfont icon-yunhang font-13 mr3"></i>查看运行记录</Link>
|
<Link to={`/${owner}/${projectsId}/devops/list/${item.branch}`} className="color-grey-6"><i className="iconfont icon-yunhang font-13 mr3"></i>查看运行记录</Link>
|
||||||
</span>
|
</span>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,7 +42,7 @@ function PipelineName({visible,onCancel,onOk,value ,branchList}){
|
||||||
</div>
|
</div>
|
||||||
<div className="choosenList mt20">
|
<div className="choosenList mt20">
|
||||||
<span>触发条件:</span>
|
<span>触发条件:</span>
|
||||||
<Select value={branchValue} style={{width:"150px"}} onChange={(e)=>setBranchValue(e)}>
|
<Select value={branchValue} style={{width:"150px"}} dropdownClassName="chooseCon" onChange={(e)=>setBranchValue(e)}>
|
||||||
{
|
{
|
||||||
branchList && branchList.length>0 && branchList.map((item,key)=>{
|
branchList && branchList.length>0 && branchList.map((item,key)=>{
|
||||||
return(
|
return(
|
||||||
|
@ -51,7 +51,7 @@ function PipelineName({visible,onCancel,onOk,value ,branchList}){
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
</Select>
|
</Select>
|
||||||
<Select mode="multiple" allowClear value={eventValue} style={{width:"180px",marginLeft:"10px"}} onChange={(e)=>{console.log(e);setEventValue(e)}}>
|
<Select mode="multiple" allowClear value={eventValue} dropdownClassName="chooseCon" style={{width:"180px",marginLeft:"10px"}} onChange={(e)=>{setEventValue(e)}}>
|
||||||
{
|
{
|
||||||
EVENT.map((item,key)=>{
|
EVENT.map((item,key)=>{
|
||||||
return(
|
return(
|
||||||
|
|
|
@ -36,39 +36,37 @@ export default ((props)=>{
|
||||||
return(
|
return(
|
||||||
<WhiteBack className="opsPanel">
|
<WhiteBack className="opsPanel">
|
||||||
<Switch {...props}>
|
<Switch {...props}>
|
||||||
<Route path="/projects/:owner/:projectsId/devops/dispose/:disposeId"
|
|
||||||
render={
|
<Route path="/:owner/:projectsId/devops/params"
|
||||||
(p) => (<New {...props} {...p}/>)
|
|
||||||
}
|
|
||||||
></Route>
|
|
||||||
<Route path="/projects/:owner/:projectsId/devops/params"
|
|
||||||
render={
|
render={
|
||||||
(p) => (<Params {...props} {...p}/>)
|
(p) => (<Params {...props} {...p}/>)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/devops/mould"
|
<Route path="/:owner/:projectsId/devops/mould"
|
||||||
render={
|
render={
|
||||||
(p) => (<Mould {...props} {...p}/>)
|
(p) => (<Mould {...props} {...p}/>)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/devops/dispose/new"
|
<Route path="/:owner/:projectsId/devops/new"
|
||||||
render={
|
render={
|
||||||
(p) => (<New {...props} {...p}/>)
|
(p) => (<New {...props} {...p}/>)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/devops/dispose"
|
|
||||||
render={
|
<Route path="/:owner/:projectsId/devops/list/:branch"
|
||||||
(p) => (<Dispose {...props} {...p}/>)
|
|
||||||
}
|
|
||||||
></Route>
|
|
||||||
<Route path="/projects/:owner/:projectsId/devops/list/:branch"
|
|
||||||
render={
|
render={
|
||||||
(p) => (<Stucture {...props} {...p}/>)
|
(p) => (<Stucture {...props} {...p}/>)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/devops"
|
<Route path="/:owner/:projectsId/devops/:disposeId"
|
||||||
render={
|
render={
|
||||||
(p) => (<About {...props} {...p}/>)
|
(p) => (<New {...props} {...p}/>)
|
||||||
|
}
|
||||||
|
></Route>
|
||||||
|
{/* 原本的两种合为一个 */}
|
||||||
|
<Route path="/:owner/:projectsId/devops"
|
||||||
|
render={
|
||||||
|
(p) =>{return( p.location.state.open_devops?<Dispose {...props} {...p}/>:<About {...props} {...p}/>)}
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
</Switch>
|
</Switch>
|
||||||
|
|
|
@ -26,7 +26,7 @@ export default ((props)=>{
|
||||||
return(
|
return(
|
||||||
<div className="disposePanel">
|
<div className="disposePanel">
|
||||||
<Banner>
|
<Banner>
|
||||||
{ permission !=="Reporter" && <Link to={`/projects/${owner}/${props.match.params.projectsId}/devops/dispose`}>工作流配置</Link>}
|
{ permission !=="Reporter" && <Link to={`/${owner}/${props.match.params.projectsId}/devops`}>工作流配置</Link>}
|
||||||
</Banner>
|
</Banner>
|
||||||
<Div>
|
<Div>
|
||||||
<Dispost {...props}/>
|
<Dispost {...props}/>
|
||||||
|
|
|
@ -104,7 +104,7 @@ function Params(props){
|
||||||
<Banner>
|
<Banner>
|
||||||
<FlexAJ>
|
<FlexAJ>
|
||||||
<span className="font-18">工作流 - 参数管理</span>
|
<span className="font-18">工作流 - 参数管理</span>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/devops/dispose`} className="font-14 color-grey-9 ml20">返回</Link>
|
<Link to={`/${owner}/${projectsId}/devops`} className="font-14 color-grey-9 ml20">返回</Link>
|
||||||
</FlexAJ>
|
</FlexAJ>
|
||||||
</Banner>
|
</Banner>
|
||||||
<Div className="disposeList">
|
<Div className="disposeList">
|
||||||
|
|
|
@ -126,7 +126,7 @@ function Mould(props){
|
||||||
<div>
|
<div>
|
||||||
<New wrappedComponentRef={(f) => childRef.current = f} ref={childRef} visible={visible} onCancel={()=>setVisible(false)} onOk={onOk}></New>
|
<New wrappedComponentRef={(f) => childRef.current = f} ref={childRef} visible={visible} onCancel={()=>setVisible(false)} onOk={onOk}></New>
|
||||||
<Banner>
|
<Banner>
|
||||||
<FlexAJ><span>工作流 - 模板管理</span><Link to={`/projects/${owner}/${projectsId}/devops/dispose`} className="font-14 color-grey-9">返回</Link></FlexAJ>
|
<FlexAJ><span>工作流 - 模板管理</span><Link to={`/${owner}/${projectsId}/devops`} className="font-14 color-grey-9">返回</Link></FlexAJ>
|
||||||
</Banner>
|
</Banner>
|
||||||
<Div className="disposeList">
|
<Div className="disposeList">
|
||||||
<FlexAJ>
|
<FlexAJ>
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
import React, { useState, useEffect } from "react";
|
import React, { useState, useEffect } from "react";
|
||||||
import { Spin } from "antd";
|
import { Spin , Menu } from "antd";
|
||||||
import { FlexAJ, AlignCenter } from "../Component/layout";
|
import { FlexAJ, AlignCenter } from "../Component/layout";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
|
import CodeSSH from './ssh/Index';
|
||||||
|
|
||||||
export default ({
|
export default ({
|
||||||
data,
|
data,
|
||||||
|
@ -17,6 +18,8 @@ export default ({
|
||||||
const [spining, setSpining] = useState(true);
|
const [spining, setSpining] = useState(true);
|
||||||
const [stage, setStage] = useState(undefined);
|
const [stage, setStage] = useState(undefined);
|
||||||
const [step, setStep] = useState(undefined);
|
const [step, setStep] = useState(undefined);
|
||||||
|
const [nav, setNav] = useState("0");
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setSpining(rightSpin);
|
setSpining(rightSpin);
|
||||||
}, [rightSpin]);
|
}, [rightSpin]);
|
||||||
|
@ -35,6 +38,7 @@ export default ({
|
||||||
: p[0];
|
: p[0];
|
||||||
|
|
||||||
setStep(sub);
|
setStep(sub);
|
||||||
|
setNav("0");
|
||||||
if (sub && sub.status !== "skipped") {
|
if (sub && sub.status !== "skipped") {
|
||||||
getStep(pre.number, sub.number);
|
getStep(pre.number, sub.number);
|
||||||
}
|
}
|
||||||
|
@ -65,43 +69,55 @@ export default ({
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<Spin spinning={spining}>
|
<React.Fragment>
|
||||||
<div className="rightMainContent">
|
{/* <Menu className="devopsNav" onClick={(e)=>{setNav(e.key)}} selectedKeys={[nav]} mode="horizontal">
|
||||||
{data && data.status !== "error" ? (
|
<Menu.Item key={'0'} value="0">开发流水线</Menu.Item>
|
||||||
<div>
|
<Menu.Item key={'1'} value="1">命令行</Menu.Item>
|
||||||
<FlexAJ className="items">
|
</Menu> */}
|
||||||
<span>{step && step.name}</span>
|
{
|
||||||
<AlignCenter>
|
nav === "0" &&
|
||||||
{step && step.duration_time}
|
<Spin spinning={spining}>
|
||||||
<i className="iconfont icon-sanjiaoxing-down"></i>
|
<div className="rightMainContent">
|
||||||
</AlignCenter>
|
{data && data.status !== "error" ? (
|
||||||
</FlexAJ>
|
<div>
|
||||||
<div>
|
<FlexAJ className="items">
|
||||||
{coders && coders.length > 0 ? (
|
<span>{step && step.name}</span>
|
||||||
coders.map((item, key) => {
|
<AlignCenter>
|
||||||
return (
|
{step && step.duration_time}
|
||||||
|
<i className="iconfont icon-sanjiaoxing-down"></i>
|
||||||
|
</AlignCenter>
|
||||||
|
</FlexAJ>
|
||||||
|
<div>
|
||||||
|
{coders && coders.length > 0 ? (
|
||||||
|
coders.map((item, key) => {
|
||||||
|
return (
|
||||||
|
<div className="opsDetailOut">
|
||||||
|
<span>{key + 1}</span>
|
||||||
|
<p>{item.out}</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
) : empty ? (
|
||||||
<div className="opsDetailOut">
|
<div className="opsDetailOut">
|
||||||
<span>{key + 1}</span>
|
<span>1</span>
|
||||||
<p>{item.out}</p>
|
<p>
|
||||||
|
{stage && stage.name} – {step && step.name}: Skipped
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
);
|
) : (
|
||||||
})
|
""
|
||||||
) : empty ? (
|
)}
|
||||||
<div className="opsDetailOut">
|
|
||||||
<span>1</span>
|
|
||||||
<p>
|
|
||||||
{stage && stage.name} – {step && step.name}: Skipped
|
|
||||||
</p>
|
|
||||||
</div>
|
</div>
|
||||||
) : (
|
</div>
|
||||||
""
|
) : (
|
||||||
)}
|
<div style={{ color: "red" }}>error:{data && data.error}</div>
|
||||||
</div>
|
)}
|
||||||
</div>
|
</div>
|
||||||
) : (
|
</Spin>
|
||||||
<div style={{ color: "red" }}>error:{data && data.error}</div>
|
}
|
||||||
)}
|
{
|
||||||
</div>
|
nav === "1" && <CodeSSH />
|
||||||
</Spin>
|
}
|
||||||
|
</React.Fragment>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -2,7 +2,7 @@ import React, { useState, useEffect , useImperativeHandle ,forwardRef } from "re
|
||||||
import { FlexAJ, AlignCenter , Banner } from "../Component/layout";
|
import { FlexAJ, AlignCenter , Banner } from "../Component/layout";
|
||||||
import { Table, Pagination, Popconfirm } from "antd";
|
import { Table, Pagination, Popconfirm } from "antd";
|
||||||
import { truncateCommitId } from "../common/util";
|
import { truncateCommitId } from "../common/util";
|
||||||
import {getUrl} from 'educoder';
|
import { getImageUrl } from 'educoder';
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
|
||||||
|
@ -58,6 +58,7 @@ function Structure(props,ref){
|
||||||
return {
|
return {
|
||||||
...item,
|
...item,
|
||||||
author:item.author && item.author.name,
|
author:item.author && item.author.name,
|
||||||
|
image_url:item.author && item.author.image_url,
|
||||||
message: {
|
message: {
|
||||||
branch: item.branch_target,
|
branch: item.branch_target,
|
||||||
message: item.message,
|
message: item.message,
|
||||||
|
@ -197,7 +198,7 @@ function Structure(props,ref){
|
||||||
}
|
}
|
||||||
|
|
||||||
function clickRows(event,e){
|
function clickRows(event,e){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/devops/${e.number}/detail`);
|
props.history.push(`/${owner}/${projectsId}/devops/${e.number}/detail`);
|
||||||
}
|
}
|
||||||
const column = [
|
const column = [
|
||||||
{
|
{
|
||||||
|
@ -244,7 +245,7 @@ function Structure(props,ref){
|
||||||
{meg.sha && <span className="color-orange">{meg.sha}</span>}
|
{meg.sha && <span className="color-orange">{meg.sha}</span>}
|
||||||
</div>
|
</div>
|
||||||
<AlignCenter>
|
<AlignCenter>
|
||||||
<img style={{borderRadius:"50%",marginRight:"10px",width:"25px",height:"25px"}} src={`${current_user && getUrl(`/images/${current_user.image_url}`)}`} />
|
<img style={{borderRadius:"50%",marginRight:"10px",width:"25px",height:"25px"}} alt="" src={`${item.image_url && getImageUrl(`/${item.image_url}`)}`} />
|
||||||
<div className="task-hide ml5" style={{ maxWidth: "300px" }}>
|
<div className="task-hide ml5" style={{ maxWidth: "300px" }}>
|
||||||
{meg.message}
|
{meg.message}
|
||||||
</div>
|
</div>
|
||||||
|
@ -289,7 +290,7 @@ function Structure(props,ref){
|
||||||
<Banner>
|
<Banner>
|
||||||
<FlexAJ>
|
<FlexAJ>
|
||||||
<span>构建列表</span>
|
<span>构建列表</span>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/devops/dispose`} className="font-15 color-grey-9">返回</Link>
|
<Link to={`/${owner}/${projectsId}/devops`} className="font-15 color-grey-9">返回</Link>
|
||||||
</FlexAJ>
|
</FlexAJ>
|
||||||
</Banner>
|
</Banner>
|
||||||
<Div>
|
<Div>
|
||||||
|
|
|
@ -275,7 +275,7 @@ function disposePipeline(props){
|
||||||
...params
|
...params
|
||||||
}).then(result=>{
|
}).then(result=>{
|
||||||
if(result){
|
if(result){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/devops/dispose`);
|
props.history.push(`/${owner}/${projectsId}/devops`);
|
||||||
}
|
}
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}).catch(error=>{
|
}).catch(error=>{
|
||||||
|
|
|
@ -194,6 +194,18 @@
|
||||||
&.rightSection{
|
&.rightSection{
|
||||||
width:100%;
|
width:100%;
|
||||||
background-color: #081930;
|
background-color: #081930;
|
||||||
|
.devopsNav{
|
||||||
|
background-color: #111c24;
|
||||||
|
border-bottom: none;
|
||||||
|
.ant-menu-item{
|
||||||
|
color: #ccc;
|
||||||
|
padding:0px;
|
||||||
|
margin:0px 20px!important;
|
||||||
|
}
|
||||||
|
.ant-menu-item.ant-menu-item-selected{
|
||||||
|
color: #1890ff;
|
||||||
|
}
|
||||||
|
}
|
||||||
.rightMainContent{
|
.rightMainContent{
|
||||||
padding:24px 30px;
|
padding:24px 30px;
|
||||||
height:100vh;
|
height:100vh;
|
||||||
|
@ -379,6 +391,9 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.chooseCon.ant-select-dropdown{
|
||||||
|
z-index: 100001;
|
||||||
|
}
|
||||||
|
|
||||||
.choosenList{
|
.choosenList{
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|
|
@ -48,7 +48,7 @@ export default (props) => {
|
||||||
axios.post(url).then((result) => {
|
axios.post(url).then((result) => {
|
||||||
if (result && result.data) {
|
if (result && result.data) {
|
||||||
props.showNotification("工作流正在重新构建!");
|
props.showNotification("工作流正在重新构建!");
|
||||||
props.history.push(`/projects/${owner}/${projectId}/devops/${result.data.number}/detail`);
|
props.history.push(`/${owner}/${projectId}/devops/${result.data.number}/detail`);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
|
@ -87,7 +87,7 @@ export default (props) => {
|
||||||
</AlignCenter>
|
</AlignCenter>
|
||||||
<Link
|
<Link
|
||||||
style={{ color: "#ddd" }}
|
style={{ color: "#ddd" }}
|
||||||
to={`/projects/${owner}/${projectId}/devops/dispose`}
|
to={`/${owner}/${projectId}/devops`}
|
||||||
>
|
>
|
||||||
<i className="iconfont icon-yiguanbi font-15 mr5"></i>退出
|
<i className="iconfont icon-yiguanbi font-15 mr5"></i>退出
|
||||||
</Link>
|
</Link>
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
import React, { useState, useEffect } from "react";
|
||||||
|
import XmlPanel from "./XmlPanel";
|
||||||
|
import mediator from "./mediator";
|
||||||
|
import axios from "axios";
|
||||||
|
|
||||||
|
// const defaulturl = `http://47.111.130.18:48088`;
|
||||||
|
const defaultValue = {
|
||||||
|
host: "106.75.231.63",
|
||||||
|
port: "2021",
|
||||||
|
ws_url: "wss://pre-webssh.educoder.net/ws",
|
||||||
|
username: "root",
|
||||||
|
secret: "Dron_123123",
|
||||||
|
};
|
||||||
|
function Index() {
|
||||||
|
const [sshConfigData, setSshConfigData] = useState(undefined);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!sshConfigData) {
|
||||||
|
init();
|
||||||
|
}
|
||||||
|
setTimeout(() => {
|
||||||
|
mediator.publish("create-socket", 1);
|
||||||
|
}, 300);
|
||||||
|
}, [sshConfigData]);
|
||||||
|
|
||||||
|
// 获取服务器连接信息
|
||||||
|
function init() {
|
||||||
|
const url = `/api/ci/pipelines/ssh_server.json`;
|
||||||
|
axios.get(url).then(result=>{
|
||||||
|
if(result && result.data){
|
||||||
|
setSshConfigData({...result.data})
|
||||||
|
}
|
||||||
|
}).catch(error=>{})
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<XmlPanel
|
||||||
|
sshConfigData={sshConfigData||{}}
|
||||||
|
sid={1}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export default Index;
|
|
@ -0,0 +1,216 @@
|
||||||
|
import React, { useRef, useEffect, useState } from 'react';
|
||||||
|
import { Base64 } from 'js-base64';
|
||||||
|
|
||||||
|
import { Terminal } from 'xterm';
|
||||||
|
import 'xterm/css/xterm.css';
|
||||||
|
import mediator from './mediator';
|
||||||
|
import ResizeObserver from 'resize-observer-polyfill';
|
||||||
|
|
||||||
|
function getColsAndRows(width, height, term) {
|
||||||
|
let w = term._core._renderService.dimensions.actualCellWidth || 9.5;
|
||||||
|
let h = term._core._renderService.dimensions.actualCellHeight || 18;
|
||||||
|
const rows = Math.floor(height / h);
|
||||||
|
const cols = Math.floor(width / w);
|
||||||
|
return [cols, rows];
|
||||||
|
}
|
||||||
|
|
||||||
|
function onLayout(term, el) {
|
||||||
|
const ro = new ResizeObserver(entries => {
|
||||||
|
console.log(entries);
|
||||||
|
for (let entry of entries) {
|
||||||
|
if (entry.target.offsetHeight > 0 || entry.target.offsetWidth > 0) {
|
||||||
|
const [cols, rows] = getColsAndRows(
|
||||||
|
entry.target.offsetWidth,
|
||||||
|
entry.target.offsetHeight,
|
||||||
|
term,
|
||||||
|
);
|
||||||
|
term.resize(cols, rows);
|
||||||
|
mediator.publish('ssh-xterm-resize', {
|
||||||
|
columns: cols,
|
||||||
|
rows: rows,
|
||||||
|
width: entry.target.offsetWidth,
|
||||||
|
height: entry.target.offsetHeight,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
ro.observe(el);
|
||||||
|
return ro;
|
||||||
|
}
|
||||||
|
|
||||||
|
const TimeTicket = 30000;
|
||||||
|
|
||||||
|
//建立 websockt 来交互
|
||||||
|
//根据容器大小计算行数和列数并做到自适应
|
||||||
|
//socket 与 term 需要分开初始化 因为socket 可能重置连接
|
||||||
|
//mediator 监听消息,如果和id匹配,则建立连接,重置,或关闭连接
|
||||||
|
|
||||||
|
export default ({ sshConfigData, sid }) => {
|
||||||
|
const [term, setTerm] = useState(null);
|
||||||
|
|
||||||
|
const { ws_url, password, port, secret } = sshConfigData;
|
||||||
|
const el = useRef();
|
||||||
|
const socket = useRef();
|
||||||
|
const isFirstConnected = useRef(false);
|
||||||
|
|
||||||
|
//term init
|
||||||
|
useEffect(() => {
|
||||||
|
if (el.current && ws_url) {
|
||||||
|
const term = new Terminal({ fontSize: 16, rendererType: 'dom' });
|
||||||
|
term.open(el.current);
|
||||||
|
|
||||||
|
term.onData(data => {
|
||||||
|
if (socket.current) {
|
||||||
|
if (socket.current.readyState === 1) {
|
||||||
|
socket.current.send(JSON.stringify({ tp: 'client', data: data }));
|
||||||
|
mediator.publish('on-operating-ssh'); //有操作则自动延时
|
||||||
|
} else {
|
||||||
|
//断开连接后重连
|
||||||
|
// socket.current = null
|
||||||
|
// mediator.publish('create-socket', sid)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
term.write('Connecting...');
|
||||||
|
setTerm(term);
|
||||||
|
const ro = onLayout(term, el.current);
|
||||||
|
return () => {
|
||||||
|
term.dispose();
|
||||||
|
ro.unobserve(el.current);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, [ws_url, el.current]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (term && ws_url) {
|
||||||
|
function createSocket() {
|
||||||
|
const socketInstance = new WebSocket(ws_url);
|
||||||
|
socket.current = socketInstance;
|
||||||
|
|
||||||
|
socketInstance.onopen = () => {
|
||||||
|
let container = term.element.parentElement;
|
||||||
|
if (container) {
|
||||||
|
let width = container.offsetWidth;
|
||||||
|
let height = container.offsetHeight;
|
||||||
|
console.log('init', {
|
||||||
|
tp: 'init',
|
||||||
|
data: {
|
||||||
|
...sshConfigData,
|
||||||
|
secret: secret,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
rows: term.rows,
|
||||||
|
columns: term.cols,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
socketInstance.send(
|
||||||
|
JSON.stringify({
|
||||||
|
tp: 'init',
|
||||||
|
data: {
|
||||||
|
...sshConfigData,
|
||||||
|
secret: secret,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
rows: term.rows,
|
||||||
|
columns: term.cols,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
term.focus();
|
||||||
|
};
|
||||||
|
socketInstance.onerror = error => {
|
||||||
|
console.log(
|
||||||
|
'------in socket error----',
|
||||||
|
error,
|
||||||
|
socketInstance,
|
||||||
|
ws_url,
|
||||||
|
);
|
||||||
|
//连接报错后,重新请求资源
|
||||||
|
// mediator.publish('on-recreate-socket')
|
||||||
|
};
|
||||||
|
socketInstance.onmessage = event => {
|
||||||
|
if (!isFirstConnected.current) {
|
||||||
|
term.write('\r');
|
||||||
|
// term.focus()
|
||||||
|
setTimeout(() => {
|
||||||
|
// term.clear();
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
|
isFirstConnected.current = true;
|
||||||
|
|
||||||
|
const data = Base64.decode(event.data.toString());
|
||||||
|
let w = term._core._renderService.dimensions.actualCellWidth || 9.5;
|
||||||
|
|
||||||
|
term.write(data);
|
||||||
|
};
|
||||||
|
|
||||||
|
socketInstance.onclose = evt => {
|
||||||
|
if (tid) {
|
||||||
|
clearInterval(tid);
|
||||||
|
}
|
||||||
|
term.write('\r\nconnection closed');
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const tid = setInterval(() => {
|
||||||
|
if (socket.current) {
|
||||||
|
socket.current.send(JSON.stringify({ tp: 'h' }));
|
||||||
|
}
|
||||||
|
}, TimeTicket);
|
||||||
|
|
||||||
|
const unSubCreate = mediator.subscribe('create-socket', id => {
|
||||||
|
if (sid === id) {
|
||||||
|
if (socket.current && socket.current.readyState === 1) {
|
||||||
|
term.focus();
|
||||||
|
} else {
|
||||||
|
createSocket();
|
||||||
|
}
|
||||||
|
term.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const unSubClose = mediator.subscribe('close-socket', id => {
|
||||||
|
if (sid === id) {
|
||||||
|
if (socket.current) {
|
||||||
|
socket.current.close();
|
||||||
|
isFirstConnected.current = false;
|
||||||
|
term.clear();
|
||||||
|
}
|
||||||
|
socket.current = null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const unSubResize = mediator.subscribe('ssh-xterm-resize', option => {
|
||||||
|
if (socket.current && socket.current.readyState === 1) {
|
||||||
|
socket.current.send(
|
||||||
|
JSON.stringify({ tp: 'resize', data: { ...option } }),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const unSubAddTime = mediator.subscribe('ssh-add-connect-time', () => {
|
||||||
|
if (socket.current && socket.current.readyState === 1) {
|
||||||
|
socket.current.send(JSON.stringify({ tp: 'overtime' }));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
unSubClose();
|
||||||
|
unSubCreate();
|
||||||
|
unSubResize();
|
||||||
|
unSubAddTime();
|
||||||
|
if (socket.current) {
|
||||||
|
socket.current.close();
|
||||||
|
isFirstConnected.current = false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, [term, ws_url, port]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div ref={el} className="xterm-panel" style={{height:"100%"}}>
|
||||||
|
{!ws_url ? <p style={{ color: '#fff' }}>正在连接命令行服务...</p> : null}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
|
@ -0,0 +1,46 @@
|
||||||
|
function Mediator(obj) {
|
||||||
|
const channels = {};
|
||||||
|
|
||||||
|
const mediator = {
|
||||||
|
subscribe: function(channel, cb) {
|
||||||
|
if (!channels[channel]) {
|
||||||
|
channels[channel] = [];
|
||||||
|
}
|
||||||
|
channels[channel].push(cb);
|
||||||
|
return this.unsubscribe.bind(null, channel, cb);
|
||||||
|
},
|
||||||
|
|
||||||
|
unsubscribe: function(channel, cb) {
|
||||||
|
let rs = channels[channel];
|
||||||
|
let index = -1;
|
||||||
|
if (rs) {
|
||||||
|
for (let i = 0; i < rs.length; i++) {
|
||||||
|
if (rs[i].name === cb.name) {
|
||||||
|
index = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (index >= 0) {
|
||||||
|
channels[channel].splice(index, 1);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
publish: function(channel) {
|
||||||
|
if (!channels[channel]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const args = Array.prototype.slice.call(arguments, 1);
|
||||||
|
channels[channel].forEach(subscription => {
|
||||||
|
subscription.apply(null, args);
|
||||||
|
});
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return obj ? Object.assign(obj, mediator) : mediator;
|
||||||
|
}
|
||||||
|
const mediator = new Mediator();
|
||||||
|
export default mediator;
|
|
@ -41,14 +41,15 @@ function DivertModal({form , visible , onSuccess , onCancel,owner,repo}){
|
||||||
// 确认转移
|
// 确认转移
|
||||||
function onOk(){
|
function onOk(){
|
||||||
validateFields((error,values)=>{
|
validateFields((error,values)=>{
|
||||||
console.log(...values);
|
|
||||||
if(!error){
|
if(!error){
|
||||||
const url = `/${owner}/${repo}/applied_transfer_projects.json`;
|
const url = `/${owner}/${repo}/applied_transfer_projects.json`;
|
||||||
Axios.post(url,{
|
Axios.post(url,{
|
||||||
...values
|
...values
|
||||||
}).then(result=>{
|
}).then(result=>{
|
||||||
if(result){
|
if(result && result.data.id){
|
||||||
onSuccess(result.data && result.data.owner);
|
onSuccess(result.data && result.data.owner);
|
||||||
|
}else{
|
||||||
|
onSuccess();
|
||||||
}
|
}
|
||||||
}).catch(error=>{})
|
}).catch(error=>{})
|
||||||
}
|
}
|
||||||
|
@ -104,7 +105,7 @@ function DivertModal({form , visible , onSuccess , onCancel,owner,repo}){
|
||||||
</ul>
|
</ul>
|
||||||
:
|
:
|
||||||
<ul className="descUl">
|
<ul className="descUl">
|
||||||
<li>仓库仅可以转移到您已经加入的组织中,不可以转移到未加入的组织中</li>
|
<li>仓库仅可以转移到您具有管理权限的组织中</li>
|
||||||
<li>涉及到仓库改名操作,请提前做好仓库备份并且在转移后对本地仓库的remote进行修改</li>
|
<li>涉及到仓库改名操作,请提前做好仓库备份并且在转移后对本地仓库的remote进行修改</li>
|
||||||
<li>转移仓库到组织后,你和组织创建者/管理员同时拥有对该仓库的管理操作</li>
|
<li>转移仓库到组织后,你和组织创建者/管理员同时拥有对该仓库的管理操作</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
@ -152,20 +153,21 @@ function DivertModal({form , visible , onSuccess , onCancel,owner,repo}){
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
}
|
}
|
||||||
|
|
||||||
<Form.Item label="仓库名称:">
|
<Form.Item label="仓库标识:" style={{marginBottom:"0px"}}>
|
||||||
{getFieldDecorator("identifier",
|
{getFieldDecorator("identifier",
|
||||||
{
|
{
|
||||||
rules:[
|
rules:[
|
||||||
{required:true,message:"请输入仓库名称"},
|
{required:true,message:"请输入仓库标识!"},
|
||||||
{
|
{
|
||||||
validator:checkIdentifier
|
validator:checkIdentifier
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
)(
|
)(
|
||||||
<Input placeholder="请输入仓库名称" autoComplete={"off"}/>
|
<Input placeholder="请输入仓库标识" autoComplete={"off"}/>
|
||||||
)}
|
)}
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
|
<span className="color-grey-9" style={{marginLeft:"120px"}}>请输入当前项目的标识:<span className="ml5 mr5 color-grey-3">{repo}</span>进行确认!</span>
|
||||||
</Form>
|
</Form>
|
||||||
</div>
|
</div>
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
.diverModal{
|
||||||
|
.descUl{
|
||||||
|
background-color: #fffae6;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding:10px 15px;
|
||||||
|
color: #efc16b;
|
||||||
|
border:1px solid #efc16b;
|
||||||
|
}
|
||||||
|
.ant-form-item-required::before{
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,85 @@
|
||||||
|
import React, { useState , forwardRef, useEffect } from 'react';
|
||||||
|
import { Form , Modal , Input , Radio } from 'antd';
|
||||||
|
import Axios from 'axios';
|
||||||
|
import CheckProfile from '../Component/ProfileModal/Profile';
|
||||||
|
|
||||||
|
export default Form.create()(
|
||||||
|
forwardRef((props)=>{
|
||||||
|
const { getFieldDecorator, validateFields , setFieldsValue } = props && props.form;
|
||||||
|
const [ visible , setVisible ] = useState(false);
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
if(!visible){
|
||||||
|
setFieldsValue({
|
||||||
|
code:undefined,
|
||||||
|
role:"developer"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},[visible])
|
||||||
|
|
||||||
|
function onOk() {
|
||||||
|
validateFields((error,values)=>{
|
||||||
|
if(!error){
|
||||||
|
const url = `/applied_projects.json`;
|
||||||
|
Axios.post(url,{
|
||||||
|
applied_project:{
|
||||||
|
...values
|
||||||
|
}
|
||||||
|
}).then(result=>{
|
||||||
|
if(result && result.data){
|
||||||
|
setVisible(false);
|
||||||
|
props.showNotification("申请加入项目成功,等待审核!");
|
||||||
|
}
|
||||||
|
}).catch(error=>{})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
function checkValue(rule, value, callback){
|
||||||
|
if(!value){
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
if(value.length < 6 || value.length > 6){
|
||||||
|
callback("请输入6位数的邀请码");
|
||||||
|
}
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
|
||||||
|
return(
|
||||||
|
<React.Fragment>
|
||||||
|
<Modal
|
||||||
|
title="加入项目"
|
||||||
|
width="480px"
|
||||||
|
visible={visible}
|
||||||
|
centered={true}
|
||||||
|
onOk={onOk}
|
||||||
|
onCancel={()=>setVisible(false)}
|
||||||
|
>
|
||||||
|
<Form layout={'inline'} className="inviteForm">
|
||||||
|
<Form.Item label="项目邀请码">
|
||||||
|
{getFieldDecorator("code",{
|
||||||
|
rules:[
|
||||||
|
{required:true,message:"请输入6位项目邀请码"},
|
||||||
|
{validator:checkValue}
|
||||||
|
]
|
||||||
|
})(
|
||||||
|
<Input placeholder="请输入6位项目邀请码" autoComplete={"off"} maxLength="6" style={{width:"300px"}}/>
|
||||||
|
)}
|
||||||
|
</Form.Item>
|
||||||
|
<Form.Item label="选择角色">
|
||||||
|
{getFieldDecorator("role",{
|
||||||
|
rules:[{required:true,message:"请选择角色"}]
|
||||||
|
})(
|
||||||
|
<Radio.Group defaultValue={"developer"}>
|
||||||
|
<Radio value="manager">管理员</Radio>
|
||||||
|
<Radio value="developer">开发者</Radio>
|
||||||
|
<Radio value="reporter">报告者</Radio>
|
||||||
|
</Radio.Group>
|
||||||
|
)}
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
</Modal>
|
||||||
|
<CheckProfile {...props} sureFunc={()=>setVisible(true)}>加入项目</CheckProfile>
|
||||||
|
</React.Fragment>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
)
|
|
@ -0,0 +1,108 @@
|
||||||
|
import React, { Component } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { Icon } from 'antd';
|
||||||
|
import _ from 'lodash';
|
||||||
|
import Nodata from '../Nodata';
|
||||||
|
|
||||||
|
|
||||||
|
class PullRefresh extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
}
|
||||||
|
this.pullRef = {};
|
||||||
|
// 节流
|
||||||
|
this.onScrollList = _.throttle(this.handleScroll, 200, {
|
||||||
|
leading: false,
|
||||||
|
trailing: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
let dom = document.querySelector('.pull-refresh-wrap');
|
||||||
|
dom && dom.addEventListener('scroll', this.onScrollList);
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
let dom = document.querySelector('.pull-refresh-wrap');
|
||||||
|
dom && dom.removeEventListener('scroll', this.onScrollList)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
handleScroll = () => {
|
||||||
|
if (this.props.count < this.props.pageSize) return;
|
||||||
|
if (this.props.type === 1 || this.props.type === 2) return;
|
||||||
|
const wrap = this.pullRef;
|
||||||
|
const currentScroll = wrap.scrollTop + wrap.clientHeight
|
||||||
|
|
||||||
|
// 触底
|
||||||
|
if (currentScroll >= (wrap.scrollHeight - 200)) {
|
||||||
|
this.loadData()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
handleLoadClick = () => {
|
||||||
|
this.loadData();
|
||||||
|
}
|
||||||
|
|
||||||
|
loadData = () => {
|
||||||
|
this.props.onPullRefresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
renderLoading() {
|
||||||
|
switch (this.props.type) {
|
||||||
|
case 0: // 加载更多
|
||||||
|
return <div className='text-center' onClick={this.handleLoadClick}>显示更多</div>
|
||||||
|
case 1: // 加载中
|
||||||
|
return (
|
||||||
|
<div className='text-center'>
|
||||||
|
<Icon type="loading" />
|
||||||
|
<span className='text-center'>加载中...</span>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
case 2: // 无样式
|
||||||
|
return <div className='text-center'>没有更多了</div>
|
||||||
|
default:
|
||||||
|
return <div className='text-center'>没有更多了</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { className, count, children } = this.props;
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={`pull-refresh-wrap ${className}`}
|
||||||
|
ref={dom => { this.pullRef = dom }}
|
||||||
|
>
|
||||||
|
|
||||||
|
{children}
|
||||||
|
|
||||||
|
{
|
||||||
|
count < 1 && <Nodata _html="暂无未读消息"/>
|
||||||
|
}
|
||||||
|
|
||||||
|
{/* 大于分页数据才显示loading */}
|
||||||
|
{/* {this.props.count >= this.props.pageSize ? this.renderLoading() : null} */}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PullRefresh.propTypes = {
|
||||||
|
className: PropTypes.string,
|
||||||
|
children: PropTypes.any,
|
||||||
|
onPullRefresh: PropTypes.func.isRequired,
|
||||||
|
type: PropTypes.oneOf([0, 1, 2]),
|
||||||
|
count: PropTypes.number.isRequired,
|
||||||
|
pageSize: PropTypes.number.isRequired,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export default PullRefresh
|
|
@ -19,7 +19,7 @@ function Footer(){
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<div>
|
<div>
|
||||||
<div style={{height:"483px"}}></div>
|
<div style={{height:"543px"}}></div>
|
||||||
<div className="newFooter edu-txt-center">
|
<div className="newFooter edu-txt-center">
|
||||||
{value && showhtml(value)}
|
{value && showhtml(value)}
|
||||||
{/* <div className="footerInfos">
|
{/* <div className="footerInfos">
|
||||||
|
|
|
@ -2,14 +2,18 @@ import React, { Component } from 'react';
|
||||||
import AccountProfile from "../../modules/user/AccountProfile";
|
import AccountProfile from "../../modules/user/AccountProfile";
|
||||||
import { getImageUrl } from 'educoder'
|
import { getImageUrl } from 'educoder'
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { Input , notification , Dropdown , Menu } from 'antd';
|
import { Input , notification , Dropdown ,Popover, Menu,Badge, Button } from 'antd';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
|
||||||
import LoginDialog from '../../modules/login/LoginDialog';
|
import LoginDialog from '../../modules/login/LoginDialog';
|
||||||
import GotoQQgroup from '../../modal/GotoQQgroup'
|
import HeadSearch from '../Component/HeadSearch';
|
||||||
|
|
||||||
|
import AddProjectModal from './AddProjectModal';
|
||||||
import '../../modules/tpm/TPMIndex.css';
|
import '../../modules/tpm/TPMIndex.css';
|
||||||
|
import CheckProfile from '../Component/ProfileModal/Profile';
|
||||||
|
|
||||||
import './header.scss';
|
import './header.scss';
|
||||||
|
import NoticeContent from './NoticeContent';
|
||||||
const $ = window.$
|
const $ = window.$
|
||||||
// TODO 这部分脚本从公共脚本中直接调用
|
// TODO 这部分脚本从公共脚本中直接调用
|
||||||
const { Search } = Input;
|
const { Search } = Input;
|
||||||
|
@ -44,6 +48,7 @@ class NewHeader extends Component {
|
||||||
settings: null,
|
settings: null,
|
||||||
visiblemyss: false,
|
visiblemyss: false,
|
||||||
openSearch:false,
|
openSearch:false,
|
||||||
|
visible:false, //浮动消息框展示控制
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
|
@ -77,36 +82,6 @@ class NewHeader extends Component {
|
||||||
} catch (e) {}
|
} catch (e) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
SearchInput = (open,item)=>{
|
|
||||||
if(open){
|
|
||||||
return(
|
|
||||||
<div
|
|
||||||
onBlur={() => {
|
|
||||||
setTimeout(() => {
|
|
||||||
this.setState({
|
|
||||||
openSearch:false
|
|
||||||
})
|
|
||||||
}, 300)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Search placeholder="实践课程/教学课堂/实践项目/交流问答"
|
|
||||||
className={`search-input mr20`}
|
|
||||||
onSearch={(value)=>this.onGlobalSearch(value,item)}
|
|
||||||
autoFocus={true}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}else{
|
|
||||||
return <i className="iconfont icon-sousuo font-18 color-grey-6 ml30" onClick={() => {
|
|
||||||
this.setState({openSearch:true})
|
|
||||||
}} />
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
onGlobalSearch=(value,item)=>{
|
|
||||||
window.location.href=`${item}?value=` + value;
|
|
||||||
}
|
|
||||||
|
|
||||||
openNotification = (messge) => {
|
openNotification = (messge) => {
|
||||||
notification.open({
|
notification.open({
|
||||||
message: "提示",
|
message: "提示",
|
||||||
|
@ -115,18 +90,12 @@ class NewHeader extends Component {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
componentWillReceiveProps(newProps, oldProps) {
|
componentWillReceiveProps(newProps, oldProps) {
|
||||||
this.setState({
|
this.setState({
|
||||||
user: newProps.user
|
user: newProps.user
|
||||||
})
|
})
|
||||||
if (newProps.Headertop !== undefined) {
|
|
||||||
old_url = newProps.Headertop.old_url
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
educoderlogin = () => {
|
educoderlogin = () => {
|
||||||
//登录账号
|
//登录账号
|
||||||
this.setState({
|
this.setState({
|
||||||
|
@ -151,7 +120,6 @@ class NewHeader extends Component {
|
||||||
})
|
})
|
||||||
};
|
};
|
||||||
HideAddcoursestypess = (i) => {
|
HideAddcoursestypess = (i) => {
|
||||||
console.log("调用了");
|
|
||||||
this.setState({
|
this.setState({
|
||||||
Addcoursestypes: false,
|
Addcoursestypes: false,
|
||||||
mydisplay: true,
|
mydisplay: true,
|
||||||
|
@ -164,13 +132,11 @@ class NewHeader extends Component {
|
||||||
})
|
})
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
// 关闭
|
// 关闭
|
||||||
cancelModulationModels = () => {
|
cancelModulationModels = () => {
|
||||||
this.setState({ isRenders: false })
|
this.setState({ isRenders: false })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
setevaluatinghides = () => {
|
setevaluatinghides = () => {
|
||||||
this.setState({
|
this.setState({
|
||||||
setevaluatinghides: true
|
setevaluatinghides: true
|
||||||
|
@ -255,6 +221,15 @@ class NewHeader extends Component {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
checkProfile=(url)=>{
|
||||||
|
const { showCompeleteDialog , completeProfile } = this.props;
|
||||||
|
if(!completeProfile){
|
||||||
|
showCompeleteDialog && showCompeleteDialog();
|
||||||
|
}else{
|
||||||
|
window.location.href(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
addMenu=(list)=>{
|
addMenu=(list)=>{
|
||||||
return(
|
return(
|
||||||
|
@ -264,10 +239,16 @@ class NewHeader extends Component {
|
||||||
{
|
{
|
||||||
list.map((item,key)=>{
|
list.map((item,key)=>{
|
||||||
return(
|
return(
|
||||||
(item.name !=="加入课堂" && item.name !=="加入开发项目") && <Menu.Item><a href={item.url}>{item.name}</a></Menu.Item>
|
(item.name !=="加入课堂" && item.name !=="加入开发项目") &&
|
||||||
|
<Menu.Item key={item.name+key}>
|
||||||
|
<CheckProfile {...this.props} sureFunc={()=>{window.location.href=item.url}}>{item.name}</CheckProfile>
|
||||||
|
</Menu.Item>
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
<Menu.Item>
|
||||||
|
<AddProjectModal {...this.props} showNotification={this.props.showNotification}/>
|
||||||
|
</Menu.Item>
|
||||||
</Menu>
|
</Menu>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
@ -278,7 +259,7 @@ class NewHeader extends Component {
|
||||||
return(
|
return(
|
||||||
<Menu className="currentMenu">
|
<Menu className="currentMenu">
|
||||||
<Menu.Item>
|
<Menu.Item>
|
||||||
<span title={current_user && current_user.username}>{current_user && current_user.username}</span>
|
<span className="currentName" title={current_user && current_user.username}>{current_user && current_user.username}</span>
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
{
|
{
|
||||||
personal && personal.length > 0 && personal.map((item,key)=>{
|
personal && personal.length > 0 && personal.map((item,key)=>{
|
||||||
|
@ -287,25 +268,27 @@ class NewHeader extends Component {
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
{/* <li><Link to={`/settings/profile`}>设置</Link></li> */}
|
||||||
<Menu.Item><a onClick={() => this.educoderloginysl()}>退出</a></Menu.Item>
|
<Menu.Item><a onClick={() => this.educoderloginysl()}>退出</a></Menu.Item>
|
||||||
</Menu>
|
</Menu>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleVisibleChange = visible => {
|
||||||
|
this.setState({ visible });
|
||||||
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { match} = this.props;
|
const { match ,resetUserInfo ,showNotification} = this.props;
|
||||||
let current_user = this.props.user;
|
let current_user = this.props.user;
|
||||||
let { Addcoursestypes,
|
let {
|
||||||
tojoinitemtype,
|
|
||||||
tojoinclasstitle,
|
|
||||||
code_notice,
|
|
||||||
AccountProfiletype,
|
AccountProfiletype,
|
||||||
user,
|
user,
|
||||||
isRender,
|
isRender,
|
||||||
headtypesonClickbool,
|
headtypesonClickbool,
|
||||||
headtypess,
|
headtypess,
|
||||||
settings,
|
settings,
|
||||||
openSearch,
|
visible,
|
||||||
} = this.state;
|
} = this.state;
|
||||||
/*用户名称 用户头像url*/
|
/*用户名称 用户头像url*/
|
||||||
let activeIndex = false;
|
let activeIndex = false;
|
||||||
|
@ -384,7 +367,6 @@ class NewHeader extends Component {
|
||||||
}
|
}
|
||||||
|
|
||||||
let search_url = settings && settings.common && settings.common.search;
|
let search_url = settings && settings.common && settings.common.search;
|
||||||
let notice_url = settings && settings.common && settings.common.notice;
|
|
||||||
return (
|
return (
|
||||||
<div className="newHeaders" id="nHeader">
|
<div className="newHeaders" id="nHeader">
|
||||||
<div className="headerContent">
|
<div className="headerContent">
|
||||||
|
@ -416,21 +398,21 @@ class NewHeader extends Component {
|
||||||
{
|
{
|
||||||
settings.navbar && settings.navbar.map((item, key) => {
|
settings.navbar && settings.navbar.map((item, key) => {
|
||||||
var new_link = item.link;
|
var new_link = item.link;
|
||||||
var user_login = this.props.user && this.props.user.login;
|
var user_login = current_user && current_user.login;
|
||||||
var is_hidden = item.hidden
|
var is_hidden = item.hidden
|
||||||
if (new_link && (new_link.indexOf("courses") > -1 || new_link.indexOf("contests") > -1)) {
|
if (new_link && (new_link.indexOf("courses") > -1 || new_link.indexOf("contests") > -1)) {
|
||||||
if (user_login) {
|
if (user_login) {
|
||||||
if (new_link.indexOf("courses") > -1) {
|
if (new_link.indexOf("courses") > -1) {
|
||||||
new_link = new_link.replace(/courses/g, "users/" + user_login + "/courses")
|
new_link = new_link.replace(/courses/g, user_login + "/courses")
|
||||||
} else if (new_link.indexOf("contests") > -1) {
|
} else if (new_link.indexOf("contests") > -1) {
|
||||||
new_link = new_link.replace(/contests/g, "users/" + user_login + "/contests")
|
new_link = new_link.replace(/contests/g, user_login + "/contests")
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
is_hidden = true
|
is_hidden = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (user_login && (new_link && new_link.indexOf("homes") > -1)) {
|
if (user_login && (new_link && new_link.indexOf("homes") > -1)) {
|
||||||
new_link = new_link.replace(/homes/g, "users/" + user_login + "/user_activities")
|
new_link = new_link.replace(/homes/g, user_login + "/user_activities")
|
||||||
}
|
}
|
||||||
|
|
||||||
var waiLian = (new_link && str.filter(item=>new_link.indexOf(item)>-1) );
|
var waiLian = (new_link && str.filter(item=>new_link.indexOf(item)>-1) );
|
||||||
|
@ -447,24 +429,30 @@ class NewHeader extends Component {
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
<div className="head-right">
|
<div className="head-right">
|
||||||
{search_url ? this.SearchInput(openSearch,search_url):""}
|
{ search_url && <HeadSearch {...this.props}/>}
|
||||||
{
|
{
|
||||||
current_user && (current_user.main_site || current_user.login) && (settings && settings.add && settings.add.length>0)?
|
current_user && (current_user.main_site || current_user.login) && (settings && settings.add && settings.add.length>0)?
|
||||||
<Dropdown overlay={this.addMenu(settings && settings.add)} placement="bottomRight">
|
<Dropdown overlay={this.addMenu(settings && settings.add)} placement="bottomRight">
|
||||||
<i className="iconfont icon-tianjiafangda color-grey-6 ml30"></i>
|
<i className="iconfont icon-tianjiafangda color-grey-6 ml30 mr15"></i>
|
||||||
</Dropdown>:""
|
</Dropdown>:""
|
||||||
}
|
}
|
||||||
|
|
||||||
{this.props.user && this.props.user.login && notice_url ?
|
{ (settings && settings.common && settings.common.notice) && (current_user && current_user.login)?
|
||||||
<div className="ml30 edu-menu-panel">
|
<Popover
|
||||||
{user && user.login &&
|
overlayClassName="notice-popover"
|
||||||
<a href={`${notice_url}`} style={{ position: 'relative' }}>
|
placement={`bottomRight`}
|
||||||
<i className="iconfont icon-xiaoxilingdang color-grey-6"></i>
|
content={<NoticeContent visible={visible} current_user={current_user} showNotification={showNotification} resetUserInfo={resetUserInfo}/>}
|
||||||
<span className="newslight" style={{ display: this.props.Headertop === undefined ? "none" : this.props.Headertop.new_message === true ? "block" : "none" }}>
|
visible={visible}
|
||||||
</span>
|
onVisibleChange={this.handleVisibleChange}
|
||||||
</a>
|
destroyTooltipOnHide
|
||||||
}
|
>
|
||||||
</div>:""
|
<Link to={"/settings/notice"} className="message-icon">
|
||||||
|
{current_user && <Badge count={current_user.message_unread_total}>
|
||||||
|
<i className="iconfont icon-xiaoxilingdang color-grey-6 ml15 mr15"></i>
|
||||||
|
</Badge>}
|
||||||
|
</Link>
|
||||||
|
</Popover>
|
||||||
|
: ""
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
{!user || (user && !user.login) ?
|
{!user || (user && !user.login) ?
|
||||||
|
@ -477,7 +465,7 @@ class NewHeader extends Component {
|
||||||
</span>
|
</span>
|
||||||
:
|
:
|
||||||
<Dropdown placement={`bottomRight`} overlay={this.renderMenu(settings && settings.personal)}>
|
<Dropdown placement={`bottomRight`} overlay={this.renderMenu(settings && settings.personal)}>
|
||||||
<a href={`/users/${this.props.current_user && this.props.current_user.login}`}>
|
<a href={`/${this.props.current_user && this.props.current_user.login}`}>
|
||||||
<img alt="头像" src={getImageUrl(`/${user.image_url}`)} className="currentImg"></img>
|
<img alt="头像" src={getImageUrl(`/${user.image_url}`)} className="currentImg"></img>
|
||||||
</a>
|
</a>
|
||||||
</Dropdown>
|
</Dropdown>
|
||||||
|
|
|
@ -0,0 +1,262 @@
|
||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
import { Badge, Menu } from 'antd';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
import axios from 'axios';
|
||||||
|
import AppPullRefresh from './AppPullRefresh';
|
||||||
|
import { noticeSourceType } from '../common/static';
|
||||||
|
import './header.scss';
|
||||||
|
import '../SecuritySetting/notice/manager/Index.scss';
|
||||||
|
import '../SecuritySetting/Index.scss';
|
||||||
|
import '../SecuritySetting/notice/myNotice/Index.scss';
|
||||||
|
|
||||||
|
|
||||||
|
function NoticeContent({ visible, showNotification, resetUserInfo, current_user: { login } }) {
|
||||||
|
const [initialize, setInitialize] = useState(true);
|
||||||
|
const [noticeType, setNoticeType] = useState("notification");
|
||||||
|
const [letterUnreadCount, setLetterUnreadCount] = useState(0);//未读私信数量
|
||||||
|
|
||||||
|
const [noticeUnreadCount, setNoticeUnreadCount] = useState(0);//未读系统通知数量
|
||||||
|
const [noticePage, setNoticePage] = useState(0);
|
||||||
|
const [noticeUnreadList, setNoticeUnreadList] = useState([]);//未读系统通知列表
|
||||||
|
|
||||||
|
const [atUnreadCount, setAtUnreadCount] = useState();//未读@我数量
|
||||||
|
const [atPage, setAtPage] = useState(0);
|
||||||
|
const [atUnreadList, setAtUnreadList] = useState([]);//未读@我列表
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
resetUserInfo();
|
||||||
|
}, [noticeUnreadCount,atUnreadCount]);
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
setNoticePage(0);
|
||||||
|
setAtPage(0);
|
||||||
|
},[visible])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const params = {
|
||||||
|
type: noticeType,
|
||||||
|
limit: 10,
|
||||||
|
page: noticeType === "notification" ? noticePage : noticeType === "atme" ? atPage : "",
|
||||||
|
status: 1,
|
||||||
|
}
|
||||||
|
getMessageList(params);
|
||||||
|
}, [noticePage, atPage]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const params = {
|
||||||
|
type: noticeType,
|
||||||
|
limit: 10,
|
||||||
|
page: 0,
|
||||||
|
status: 1,
|
||||||
|
};
|
||||||
|
if (initialize) {
|
||||||
|
params.type = "atme"
|
||||||
|
}
|
||||||
|
visible && getMessageList(params);
|
||||||
|
}, [visible]);
|
||||||
|
|
||||||
|
|
||||||
|
function getMessageList(params) {
|
||||||
|
axios.get(`/users/${login}/messages.json`, {
|
||||||
|
params: params,
|
||||||
|
}).then((response) => {
|
||||||
|
if (response && response.data) {
|
||||||
|
setNoticeUnreadCount(response.data.unread_notification);
|
||||||
|
setAtUnreadCount(response.data.unread_atme);
|
||||||
|
if (params.type === "notification") {
|
||||||
|
let list = response.data.messages;
|
||||||
|
if (params.page !== 0) {
|
||||||
|
list = [...noticeUnreadList, ...list];
|
||||||
|
}
|
||||||
|
setNoticeUnreadList(list);
|
||||||
|
if (initialize) {
|
||||||
|
// 如果是第一次加载,根据数据量判断是否切换tab栏
|
||||||
|
setInitialize(false);
|
||||||
|
if (response.data.unread_notification === 0 && response.data.unread_atme !== 0) {
|
||||||
|
setNoticeType("atme");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (params.type === "atme") {
|
||||||
|
let list = response.data.messages;
|
||||||
|
if (params.page !== 0) {
|
||||||
|
list = [...atUnreadList, ...list];
|
||||||
|
}
|
||||||
|
setAtUnreadList(list);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function readAll() {
|
||||||
|
axios.post(`/users/${login}/messages/read.json`, {
|
||||||
|
type: noticeType,
|
||||||
|
ids: [-1]
|
||||||
|
}).then((response) => {
|
||||||
|
let data = response.data;
|
||||||
|
if (!data) return;
|
||||||
|
if (data.status === 0) {
|
||||||
|
changeReadMarkAll(noticeType);
|
||||||
|
} else {
|
||||||
|
showNotification(data.message);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function changeReadMarkAll(noticeType) {
|
||||||
|
if (noticeType === "notification") {
|
||||||
|
let list = noticeUnreadList.slice();
|
||||||
|
list.forEach(item => {
|
||||||
|
item.status = 2;
|
||||||
|
})
|
||||||
|
setNoticeUnreadList(list);
|
||||||
|
setNoticeUnreadCount(0);
|
||||||
|
} else if (noticeType === "atme") {
|
||||||
|
let list = atUnreadList.slice();
|
||||||
|
list.forEach(item => {
|
||||||
|
item.status = 2;
|
||||||
|
})
|
||||||
|
setAtUnreadList(list);
|
||||||
|
setAtUnreadCount(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// const [letter_unread_list, setLetter_unread_list] = useState([
|
||||||
|
// {
|
||||||
|
// id: 122,
|
||||||
|
// read: 0, //是否已读,0未读,1已读
|
||||||
|
// send_name: "蒋宇航", //消息发送人
|
||||||
|
// send_login: "jiangYuHang", //消息发送人的login,前端根据这个跳转到消息内页
|
||||||
|
// content: "私信内容", //最近一条未读消息的内容
|
||||||
|
// create_time: "2019-03-04 18:08", //发送时间
|
||||||
|
// },
|
||||||
|
// ]);
|
||||||
|
|
||||||
|
function readItem(item) {
|
||||||
|
axios.post(`/users/${login}/messages/read.json`, {
|
||||||
|
type: noticeType,
|
||||||
|
ids: [item.id]
|
||||||
|
}).then((response) => {
|
||||||
|
let data = response.data;
|
||||||
|
if (!data) return;
|
||||||
|
if (data.status === 0) {
|
||||||
|
changeReadMark(item);
|
||||||
|
item.notification_url && window.open(item.notification_url);
|
||||||
|
} else {
|
||||||
|
showNotification(data.message);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function changeReadMark(item) {
|
||||||
|
if (item.type === "notification") {
|
||||||
|
let list = noticeUnreadList.slice();
|
||||||
|
let index = noticeUnreadList.indexOf(item);
|
||||||
|
list[index].status = 2;
|
||||||
|
setNoticeUnreadList(list);
|
||||||
|
if (noticeUnreadCount > 0) {
|
||||||
|
setNoticeUnreadCount(noticeUnreadCount - 1);
|
||||||
|
}
|
||||||
|
} else if (item.type === "atme") {
|
||||||
|
let list = atUnreadList.slice();
|
||||||
|
let index = atUnreadList.indexOf(item);
|
||||||
|
list[index].status = 2;
|
||||||
|
setAtUnreadList(list);
|
||||||
|
if (atUnreadCount > 0) {
|
||||||
|
setAtUnreadCount(atUnreadCount - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="messageHoverDiv notice01">
|
||||||
|
<div className="sshHead hoverNotice-head">
|
||||||
|
<Menu mode="horizontal" selectedKeys={noticeType} onClick={(e) => setNoticeType(e.key)}>
|
||||||
|
<Menu.Item key="notification"><Badge count={noticeUnreadCount}>系统通知</Badge></Menu.Item>
|
||||||
|
{/* <Menu.Item key="1" id="item-private"><Badge count={letterUnreadCount}>私信</Badge></Menu.Item> */}
|
||||||
|
<Menu.Item key="atme"><Badge count={atUnreadCount}>@我</Badge></Menu.Item>
|
||||||
|
</Menu>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 系统通知 */}
|
||||||
|
{noticeType === "notification" && <AppPullRefresh
|
||||||
|
className='hoverNotice-body' // 外部添加className加以区分
|
||||||
|
onPullRefresh={() => { setNoticePage(noticePage + 1); }} //触发加载ajax的function
|
||||||
|
// type={2} // 传送加载组件的状态
|
||||||
|
count={noticeUnreadList.length} // 数据当前的总数量
|
||||||
|
pageSize={10} //
|
||||||
|
>
|
||||||
|
{
|
||||||
|
noticeUnreadList.map(item => {
|
||||||
|
return (
|
||||||
|
<div key={item.id + Math.random()} className="noticeCont-back" onClick={() => { readItem(item) }}>
|
||||||
|
<div className={`noticeCont ${item.notification_url?'pointer':''}`}>
|
||||||
|
<span style={{ visibility: item.status === 1 ? 'visible' : 'hidden' }}>
|
||||||
|
<Badge color="#FA2020" />
|
||||||
|
</span>
|
||||||
|
<i className={"iconfont " + noticeSourceType[item.source]}></i>
|
||||||
|
<div className="noticeCont-text">
|
||||||
|
<span className="content-span notice-cont-span" dangerouslySetInnerHTML={{ __html: item.content }}></span>
|
||||||
|
<span className="timeSpan">{item.time_ago}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</AppPullRefresh>
|
||||||
|
}
|
||||||
|
|
||||||
|
{/* @我 */}
|
||||||
|
{noticeType === "atme" && <AppPullRefresh
|
||||||
|
className='hoverNotice-body' // 外部添加className加以区分
|
||||||
|
onPullRefresh={() => { setAtPage(atPage + 1); }} //触发加载ajax的function
|
||||||
|
// type={1} // 传送加载组件的状态
|
||||||
|
count={atUnreadList.length} // 数据当前的总数量
|
||||||
|
pageSize={10} //
|
||||||
|
>
|
||||||
|
{atUnreadList.map(item => {
|
||||||
|
return (
|
||||||
|
<div key={item.id + Math.random()} className="noticeCont-back" onClick={() => { readItem(item) }}>
|
||||||
|
<div className="noticeCont">
|
||||||
|
<span style={{ visibility: item.status === 1 ? 'visible' : 'hidden' }}>
|
||||||
|
<Badge color="#FA2020" />
|
||||||
|
</span>
|
||||||
|
<div className="noticeCont-text">
|
||||||
|
<span className="content-span atme-cont-span" dangerouslySetInnerHTML={{ __html: "<b>" + (item.sender ? item.sender.name : '') + "</b> " + item.content + " 中@我" }}></span>
|
||||||
|
<span className="timeSpan">{item.time_ago}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</AppPullRefresh>
|
||||||
|
}
|
||||||
|
|
||||||
|
{/* 私信 */}
|
||||||
|
{/* {noticeType === "1" ? letter_unread_list.length > 0 ? letter_unread_list.map(item => {
|
||||||
|
return (
|
||||||
|
<div className="noticeCont-back">
|
||||||
|
<div className="noticeCont" style={{ height: item.content.length >= 30 && item.content.length <= 34 ? '65px' : "" }}>
|
||||||
|
<Badge color="#FA2020" />
|
||||||
|
<div className="noticeCont-text">
|
||||||
|
<span>{item.send_name}:</span>
|
||||||
|
<span className="boldSpan" dangerouslySetInnerHTML={{ __html: item.content.length >= 50 ? item.content.substr(0, 50) + "..." : item.content }}></span>
|
||||||
|
<span className="timeSpan">{item.create_time ? timeAgo(item.create_time) : "刚刚"}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}) : "暂无数据" : ""} */}
|
||||||
|
<div className="hoverNotice-buttom">
|
||||||
|
<Link to={{pathname:"/settings/notice",query:{noticeType:noticeType}}}>全部消息</Link>
|
||||||
|
{noticeUnreadCount > 0 && noticeType === "notification" && <a onClick={readAll}>所有系统消息一键已读</a>}
|
||||||
|
{atUnreadCount > 0 && noticeType === "atme" && <a onClick={readAll}>所有@我一键已读</a>}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default NoticeContent;
|
|
@ -1,7 +1,6 @@
|
||||||
|
|
||||||
.dropdownFlex{
|
.dropdownFlex{
|
||||||
display:flex;
|
display:flex;
|
||||||
padding:5px;
|
|
||||||
background:#fff;
|
background:#fff;
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
.ant-menu-vertical > .ant-menu-item{
|
.ant-menu-vertical > .ant-menu-item{
|
||||||
|
@ -9,6 +8,13 @@
|
||||||
height: 35px;
|
height: 35px;
|
||||||
line-height: 35px;
|
line-height: 35px;
|
||||||
margin:0px;
|
margin:0px;
|
||||||
|
&.ant-menu-item-selected{
|
||||||
|
background-color: #fff;
|
||||||
|
a{color: rgba(0, 0, 0, 0.65)!important;}
|
||||||
|
}
|
||||||
|
&.ant-menu-item-active{
|
||||||
|
a{color: #4cacff!important;}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.ant-menu-vertical{
|
.ant-menu-vertical{
|
||||||
border:none;
|
border:none;
|
||||||
|
@ -18,16 +24,24 @@
|
||||||
width: 34px;
|
width: 34px;
|
||||||
height: 34px;
|
height: 34px;
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
margin-left: 30px;
|
margin-left: 15px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.currentMenu{
|
.currentMenu{
|
||||||
width: 120px;
|
width: 120px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
padding:0px;
|
padding:0px;
|
||||||
|
.currentName{
|
||||||
|
padding:0px 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
li{
|
li{
|
||||||
height: 40px;
|
height: 40px;
|
||||||
line-height: 40px;
|
line-height: 40px;
|
||||||
padding:0px;
|
padding:0px!important;
|
||||||
cursor: default;
|
cursor: default;
|
||||||
&:hover{
|
&:hover{
|
||||||
background-color: #fff;
|
background-color: #fff;
|
||||||
|
@ -104,4 +118,148 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
.inviteForm{
|
||||||
|
.ant-form-item{
|
||||||
|
margin-right: 0px;
|
||||||
|
}
|
||||||
|
.ant-form-item-label{
|
||||||
|
width: 110px;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 右上角小铃铛单独样式
|
||||||
|
.notice-popover{
|
||||||
|
//popover小尖尖
|
||||||
|
.ant-popover-arrow{
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
//popover框
|
||||||
|
.ant-popover-inner-content {
|
||||||
|
width: 386px;
|
||||||
|
height: 446px;
|
||||||
|
box-shadow: 0px 4px 8px 2px rgba(212, 212, 212, 0.5);
|
||||||
|
border-radius: 4px;
|
||||||
|
margin-top: -10px;
|
||||||
|
padding: 12px 1px 12px 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageHoverDiv .ant-menu-item{
|
||||||
|
margin-right: 24px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hoverNotice-head{
|
||||||
|
margin-left: 18px;
|
||||||
|
|
||||||
|
& .ant-badge{
|
||||||
|
font-size: 14px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
&>.ant-menu-horizontal {
|
||||||
|
border-bottom: 1px solid #e8e8e8 !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.hoverNotice-body{
|
||||||
|
height: 342px;
|
||||||
|
overflow-y: scroll;
|
||||||
|
|
||||||
|
& b{
|
||||||
|
font-weight: 400;
|
||||||
|
text-shadow: 0.5px 0 0 #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.none_panels{
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-icon{
|
||||||
|
position: relative;
|
||||||
|
.ant-scroll-number{
|
||||||
|
right:12px;
|
||||||
|
padding: 0 0px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
.hoverNotice-buttom{
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 12px 18px;
|
||||||
|
a{
|
||||||
|
color: #466AFF;
|
||||||
|
&:hover{
|
||||||
|
opacity:0.85;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.noticeCont-back{
|
||||||
|
.pointer{
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
&:hover{
|
||||||
|
background: #F3F4F6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.noticeCont{
|
||||||
|
display: flex;
|
||||||
|
margin: 0 16px 0 18px;
|
||||||
|
padding: 12px 0 10px 0;
|
||||||
|
line-height: 24px;
|
||||||
|
border-bottom: 1px solid #EEEEEE;
|
||||||
|
cursor: default;
|
||||||
|
i{
|
||||||
|
font-size: 14px !important;
|
||||||
|
margin-right: 6px;
|
||||||
|
color: #333333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.boldSpan{
|
||||||
|
font-weight: 400;
|
||||||
|
text-shadow: 0.5px 0 0 #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noticeCont-text{
|
||||||
|
display: flex;
|
||||||
|
color:#333333;
|
||||||
|
flex:auto;
|
||||||
|
justify-content: space-between;
|
||||||
|
|
||||||
|
& .content-span{
|
||||||
|
word-break: break-all;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
display: -webkit-box;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
-webkit-line-clamp: 2;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
& .atme-cont-span{
|
||||||
|
width: 272px;
|
||||||
|
}
|
||||||
|
|
||||||
|
& .notice-cont-span{
|
||||||
|
width: 255px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeSpan{
|
||||||
|
font-size: 12px;
|
||||||
|
color: #666666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-name{
|
||||||
|
margin-right: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.text-center{
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
|
@ -12,8 +12,9 @@ import Loadable from "react-loadable";
|
||||||
import Loading from "../Loading";
|
import Loading from "../Loading";
|
||||||
import { ImageLayerOfCommentHOC } from "../modules/page/layers/ImageLayerOfCommentHOC";
|
import { ImageLayerOfCommentHOC } from "../modules/page/layers/ImageLayerOfCommentHOC";
|
||||||
|
|
||||||
|
|
||||||
const ProjectNew = Loadable({
|
const ProjectNew = Loadable({
|
||||||
loader: () => import("./New/Index"),
|
loader: () => import("./New/Index"),
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
});
|
});
|
||||||
const ProjectIndex = Loadable({
|
const ProjectIndex = Loadable({
|
||||||
|
@ -21,17 +22,23 @@ const ProjectIndex = Loadable({
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
});
|
});
|
||||||
|
|
||||||
const ProjectDetail = Loadable({
|
// 项目详情放在用户和组织下作为二级菜单存在
|
||||||
loader: () => import("./Main/Detail"),
|
// const ProjectDetail = Loadable({
|
||||||
loading: Loading,
|
// loader: () => import("./Main/Detail"),
|
||||||
});
|
// loading: Loading,
|
||||||
|
// });
|
||||||
|
|
||||||
const Infos = Loadable({
|
|
||||||
loader: () => import("./users/Infos"),
|
|
||||||
loading: Loading,
|
|
||||||
});
|
|
||||||
class Index extends Component {
|
class Index extends Component {
|
||||||
|
|
||||||
|
componentDidUpdate = () => {
|
||||||
|
this.props.history.listen(() => {
|
||||||
|
if (document.body.scrollTop || document.documentElement.scrollTop > 0) {
|
||||||
|
window.scrollTo(0, 0)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
render() {
|
render() {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="newMain clearfix">
|
<div className="newMain clearfix">
|
||||||
<Switch {...this.props}>
|
<Switch {...this.props}>
|
||||||
|
@ -48,30 +55,27 @@ class Index extends Component {
|
||||||
)}
|
)}
|
||||||
></Route>
|
></Route>
|
||||||
<Route
|
<Route
|
||||||
path="/projects/new"
|
path="/projects/mirror/new"
|
||||||
render={(props) => (
|
render={(props) => (
|
||||||
<ProjectNew {...this.props} {...props} />
|
<ProjectNew {...this.props} {...props} />
|
||||||
)}
|
)}
|
||||||
></Route>
|
></Route>
|
||||||
<Route
|
{/* <Route
|
||||||
path="/projects/:owner/:projectsId"
|
path="/:owner/:projectsId"
|
||||||
render={(props) => (
|
render={(props) => (
|
||||||
<ProjectDetail {...this.props} {...props} />
|
<ProjectDetail {...this.props} {...props} />
|
||||||
)}
|
)}
|
||||||
></Route>
|
></Route> */}
|
||||||
<Route
|
<Route
|
||||||
path="/projects"
|
path="/explore"
|
||||||
render={(props) => (
|
render={(props) => (
|
||||||
<ProjectIndex {...this.props} {...props} />
|
<ProjectIndex {...this.props} {...props} />
|
||||||
)}
|
)}
|
||||||
></Route>
|
></Route>
|
||||||
|
|
||||||
<Route
|
<Route
|
||||||
exact
|
|
||||||
path="/"
|
path="/"
|
||||||
render={(props) => (
|
render={(props) => (
|
||||||
this.props.current_user && this.props.current_user.login ?
|
|
||||||
<Infos {...this.props} {...props} />
|
|
||||||
:
|
|
||||||
<ProjectIndex {...this.props} {...props} />
|
<ProjectIndex {...this.props} {...props} />
|
||||||
)}
|
)}
|
||||||
></Route>
|
></Route>
|
||||||
|
@ -86,3 +90,10 @@ export default withRouter(
|
||||||
parentSelector: ".newMain",
|
parentSelector: ".newMain",
|
||||||
})(CNotificationHOC()(SnackbarHOC()(TPMIndexHOC(Index))))
|
})(CNotificationHOC()(SnackbarHOC()(TPMIndexHOC(Index))))
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// export default withRouter(
|
||||||
|
// ImageLayerOfCommentHOC({
|
||||||
|
// imgSelector: ".imageLayerParent img, .imageLayerParent .imageTarget",
|
||||||
|
// parentSelector: ".newMain",
|
||||||
|
// })(Index)
|
||||||
|
// );
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import React , { useEffect , useState } from 'react';
|
import React , { useEffect , useState } from 'react';
|
||||||
import { WhiteBack , Box , LongWidth , ShortWidth , Gap , AlignCenter , FlexAJ } from '../Component/layout';
|
import { WhiteBack , Box , LongWidth , ShortWidth , Gap , AlignCenter , FlexAJ } from '../Component/layout';
|
||||||
import { Dropdown , Menu , Divider , Spin, Button } from 'antd';
|
import { Dropdown , Menu , Divider , Spin, Button , Typography } from 'antd';
|
||||||
import { getImageUrl } from "educoder";
|
import { getImageUrl } from "educoder";
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
import { truncateCommitId } from "../common/util";
|
||||||
import CloneAddress from '../Branch/CloneAddress';
|
import CloneAddress from '../Branch/CloneAddress';
|
||||||
|
|
||||||
import SelectBranch from '../Branch/Select';
|
import SelectBranch from '../Branch/Select';
|
||||||
|
@ -19,12 +20,29 @@ import LanguagePower from '../Component/LanguagePower';
|
||||||
import DrawerPanel from '../Component/DrawerPanel';
|
import DrawerPanel from '../Component/DrawerPanel';
|
||||||
import UpdateDescModal from './sub/UpdateDescModal';
|
import UpdateDescModal from './sub/UpdateDescModal';
|
||||||
import Nodata from '../Nodata';
|
import Nodata from '../Nodata';
|
||||||
|
import Invite from './sub/Invite';
|
||||||
|
import CheckProfile from '../Component/ProfileModal/Profile';
|
||||||
|
import RenderHtml from '../../components/render-html';
|
||||||
/**
|
/**
|
||||||
* projectDetail.type:0是托管项目,1是镜像项目,2是同步镜像项目(为2时不支持在线创建、在线上传、在线修改、在线删除、创建合并请求等功能)
|
* projectDetail.type:0是托管项目,1是镜像项目,2是同步镜像项目(为2时不支持在线创建、在线上传、在线修改、在线删除、创建合并请求等功能)
|
||||||
*/
|
*/
|
||||||
|
const { Paragraph } = Typography;
|
||||||
|
function turnbar(str){
|
||||||
|
if(str && str.length>0 && str.indexOf("/")>-1){
|
||||||
|
return str.replaceAll('/','%2F');
|
||||||
|
}
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
function returnbar(str){
|
||||||
|
if(str && str.length>0 && str.indexOf("%2F")>-1){
|
||||||
|
return str.replaceAll('%2F','/');
|
||||||
|
}
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
function CoderDepot(props){
|
function CoderDepot(props){
|
||||||
const [ projectDetail , setProjectDetail ]= useState(undefined);
|
const [ projectDetail , setProjectDetail ]= useState(undefined);
|
||||||
|
const [ inviteCode , setInviteCode ] = useState(undefined);
|
||||||
const [ treeValue , setTreeValue ] = useState(undefined);
|
const [ treeValue , setTreeValue ] = useState(undefined);
|
||||||
const [ treeValuePath , setTreeValuePath ] = useState(undefined);
|
const [ treeValuePath , setTreeValuePath ] = useState(undefined);
|
||||||
const [ lastCommit,setLastCommit ] = useState(undefined);
|
const [ lastCommit,setLastCommit ] = useState(undefined);
|
||||||
|
@ -46,13 +64,35 @@ function CoderDepot(props){
|
||||||
const [ website , setWebsite ] = useState(undefined);
|
const [ website , setWebsite ] = useState(undefined);
|
||||||
const [ lesson_url , setLessonUrl ] = useState(undefined);
|
const [ lesson_url , setLessonUrl ] = useState(undefined);
|
||||||
const [ readme , setReadme ] = useState(undefined);
|
const [ readme , setReadme ] = useState(undefined);
|
||||||
const [ defaultBranch , setDefaultBranch ] = useState(undefined)
|
const [ defaultBranch , setDefaultBranch ] = useState(undefined);
|
||||||
|
const [ editReadme , setEditReadme ] = useState(false);
|
||||||
|
const [ pullsFlag , setPullsFlag ] = useState(true);
|
||||||
|
const [ issuesFlag , setIssuesFlag ] = useState(true);
|
||||||
|
const [ releaseVersions , setReleaseVersions] = useState(undefined);
|
||||||
|
|
||||||
const owner = props.match.params.owner;
|
const owner = props.match.params.owner;
|
||||||
const projectsId = props.match.params.projectsId;
|
const projectsId = props.match.params.projectsId;
|
||||||
const branchName = props.match.params.branchName;
|
let branchName = props.match.params.branchName;
|
||||||
|
branchName = returnbar(branchName);
|
||||||
const details = props.projectDetail;
|
const details = props.projectDetail;
|
||||||
let pathname = props.history.location.pathname;
|
let pathname = props.history.location.pathname;
|
||||||
|
//distribution:判断此用户是否可以创建发行版
|
||||||
|
const distribution = details && details.type != 2 && (details.permission === "Admin" || details.permission === "Owner" || details.permission === "Manager");
|
||||||
|
const { bannerList } = props;
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
if(bannerList && bannerList.length>0){
|
||||||
|
let a = bannerList.filter(i=>i.menu_name === "pulls");
|
||||||
|
let i = bannerList.filter(i=>i.menu_name === "issues");
|
||||||
|
if(a && a.length === 0){
|
||||||
|
setPullsFlag(false);
|
||||||
|
}
|
||||||
|
if(i && i.length === 0){
|
||||||
|
setIssuesFlag(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},[bannerList])
|
||||||
|
|
||||||
|
|
||||||
useEffect(()=>{
|
useEffect(()=>{
|
||||||
if(details){
|
if(details){
|
||||||
|
@ -61,6 +101,7 @@ function CoderDepot(props){
|
||||||
setWebsite(details.website);
|
setWebsite(details.website);
|
||||||
setLessonUrl(details.lesson_url);
|
setLessonUrl(details.lesson_url);
|
||||||
setDefaultBranch(details.default_branch);
|
setDefaultBranch(details.default_branch);
|
||||||
|
setInviteCode(details.invite_code);
|
||||||
}
|
}
|
||||||
},[details])
|
},[details])
|
||||||
|
|
||||||
|
@ -75,21 +116,58 @@ function CoderDepot(props){
|
||||||
|
|
||||||
useEffect(()=>{
|
useEffect(()=>{
|
||||||
if (projectsId && owner && defaultBranch){
|
if (projectsId && owner && defaultBranch){
|
||||||
if(pathname.indexOf(`/projects/${owner}/${projectsId}`) > -1 && pathname.indexOf(`/tree/${branchName}/`) > -1) {
|
let b = turnbar(branchName) ;
|
||||||
let url = pathname.split(`/tree/${branchName}/`)[1];
|
if(pathname.indexOf(`/${owner}/${projectsId}`) > -1 && pathname.indexOf(`/tree/${b}/`) > -1) {
|
||||||
|
let url = pathname.split(`/tree/${b}/`)[1];
|
||||||
setTreeValue(url);
|
setTreeValue(url);
|
||||||
getFileInfo(url,branchName);
|
getFileInfo(url,branchName);
|
||||||
|
setType("file");
|
||||||
|
// getReadmeInfo(url,branchName);
|
||||||
|
// setReadme(undefined);
|
||||||
}else{
|
}else{
|
||||||
setTreeValue(undefined);
|
setTreeValue(undefined);
|
||||||
getDirInfo(branchName || defaultBranch);
|
getDirInfo(branchName || defaultBranch);
|
||||||
|
setType("dir");
|
||||||
|
// getReadmeInfo('', branchName || defaultBranch);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},[projectsId,owner,pathname,defaultBranch])
|
},[projectsId,owner,pathname,defaultBranch])
|
||||||
|
|
||||||
|
useEffect(()=>{
|
||||||
|
axios.get(`/${owner}/${projectsId}/releases.json`).then((result)=>{
|
||||||
|
if(result && result.data){
|
||||||
|
const release = {
|
||||||
|
"list":result.data.releases,
|
||||||
|
"total_count":result.data.releases.length
|
||||||
|
}
|
||||||
|
setReleaseVersions(release);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},[])
|
||||||
|
|
||||||
|
// 获取readme信息
|
||||||
|
function getReadmeInfo(path, ref) {
|
||||||
|
axios.get(`/${owner}/${projectsId}/readme.json`, {
|
||||||
|
params:{
|
||||||
|
owner: owner,
|
||||||
|
repo: projectsId,
|
||||||
|
filepath:path,
|
||||||
|
ref:ref || branchName
|
||||||
|
}
|
||||||
|
}).then((result) => {
|
||||||
|
if (result) {
|
||||||
|
setReadme(result.data);
|
||||||
|
} else {
|
||||||
|
setReadme(undefined);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// 获取主目录列表
|
// 获取主目录列表
|
||||||
function getDirInfo(branch){
|
function getDirInfo(branch){
|
||||||
setIsSpin(true);
|
setIsSpin(true);
|
||||||
const url = `/${owner}/${projectsId}/entries.json`;
|
const url = `/${owner}/${projectsId}/entries.json`;
|
||||||
|
|
||||||
axios.get(url, {
|
axios.get(url, {
|
||||||
params: { ref: branch }
|
params: { ref: branch }
|
||||||
}).then((result) => {
|
}).then((result) => {
|
||||||
|
@ -104,7 +182,10 @@ function CoderDepot(props){
|
||||||
setLastCommitAuthor(c && c.committer);
|
setLastCommitAuthor(c && c.committer);
|
||||||
setMainFlag(true);
|
setMainFlag(true);
|
||||||
setReadOnly(true);
|
setReadOnly(true);
|
||||||
setReadme(result.data.readme);
|
// setReadme(result.data.readme);
|
||||||
|
setEditReadme(false);
|
||||||
|
setHide(true);
|
||||||
|
getReadmeInfo('', branchName || defaultBranch);
|
||||||
}
|
}
|
||||||
setTimeout(function(){setIsSpin(false);},500);
|
setTimeout(function(){setIsSpin(false);},500);
|
||||||
}).catch(error=>{setIsSpin(false);})
|
}).catch(error=>{setIsSpin(false);})
|
||||||
|
@ -116,7 +197,7 @@ function CoderDepot(props){
|
||||||
let ele = document.getElementById("ptxt");
|
let ele = document.getElementById("ptxt");
|
||||||
if(ele){
|
if(ele){
|
||||||
let h = ele.offsetHeight;
|
let h = ele.offsetHeight;
|
||||||
if( h > 18 ) setHideBtn(true)
|
if( h > 35 ) setHideBtn(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},[projectDetail,lastCommit])
|
},[projectDetail,lastCommit])
|
||||||
|
@ -137,16 +218,20 @@ function CoderDepot(props){
|
||||||
setDirInfo(undefined);
|
setDirInfo(undefined);
|
||||||
setFileInfo(en);
|
setFileInfo(en);
|
||||||
setType(en.type);
|
setType(en.type);
|
||||||
|
setReadme(undefined);
|
||||||
}else{
|
}else{
|
||||||
setFileInfo(undefined);
|
setFileInfo(undefined);
|
||||||
setDirInfo(en);
|
setDirInfo(en);
|
||||||
setType("dir");
|
setType("dir");
|
||||||
|
getReadmeInfo(path, branchName || defaultBranch);
|
||||||
}
|
}
|
||||||
let c = result.data.last_commit
|
let c = result.data.last_commit
|
||||||
setLastCommit(c && c.commit);
|
setLastCommit(c && c.commit);
|
||||||
setLastCommitAuthor(c && c.committer);
|
setLastCommitAuthor(c && c.committer);
|
||||||
setMainFlag(false);
|
setMainFlag(false);
|
||||||
setReadOnly(true);
|
setReadOnly(true);
|
||||||
|
setReadOnly(!editReadme);
|
||||||
|
setHide(true);
|
||||||
}
|
}
|
||||||
setTimeout(function(){setIsSpin(false);},500)
|
setTimeout(function(){setIsSpin(false);},500)
|
||||||
}).catch(error=>{setIsSpin(false);})
|
}).catch(error=>{setIsSpin(false);})
|
||||||
|
@ -154,17 +239,26 @@ function CoderDepot(props){
|
||||||
|
|
||||||
// 切换分支或者标签
|
// 切换分支或者标签
|
||||||
function changeBranch(value){
|
function changeBranch(value){
|
||||||
let url = `/projects/${owner}/${projectsId}${value && `/tree/${value}`}${treeValue ? `/${treeValue}`:""}`;
|
let checkvalue = turnbar(value);
|
||||||
|
let url = `/${owner}/${projectsId}${value && `/tree/${checkvalue}`}${treeValue ? `/${treeValue}`:""}`;
|
||||||
props.history.push(url);
|
props.history.push(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 文件相关的下拉项
|
// 文件相关的下拉项
|
||||||
const fileMenu =(
|
function fileMenu(){
|
||||||
<Menu>
|
let b = branchName || defaultBranch;
|
||||||
<Menu.Item><a onClick={()=>urlLink(`/projects/${owner}/${projectsId}/${branchName || defaultBranch}/uploadfile${treeValue === undefined ? "" : `/${treeValue}`}`)}>上传文件</a></Menu.Item>
|
let checkvalue = turnbar(b);
|
||||||
<Menu.Item><a onClick={()=>urlLink(`/projects/${owner}/${projectsId}/${branchName || defaultBranch}/newfile${treeValue === undefined ? "" : `/${treeValue}`}`)}>新建文件</a></Menu.Item>
|
return (
|
||||||
|
<Menu className="fileMenu">
|
||||||
|
<Menu.Item>
|
||||||
|
<CheckProfile {...props} sureFunc={()=>urlLink(`/${owner}/${projectsId}/${checkvalue}/uploadfile${treeValue === undefined ? "" : `/${treeValue}`}`)}>上传文件</CheckProfile>
|
||||||
|
</Menu.Item>
|
||||||
|
<Menu.Item>
|
||||||
|
<CheckProfile {...props} sureFunc={()=>urlLink(`/${owner}/${projectsId}/${checkvalue}/newfile${treeValue === undefined ? "" : `/${treeValue}`}`)}>新建文件</CheckProfile>
|
||||||
|
</Menu.Item>
|
||||||
</Menu>
|
</Menu>
|
||||||
)
|
)
|
||||||
|
}
|
||||||
|
|
||||||
function getPathUrl(array,index){
|
function getPathUrl(array,index){
|
||||||
if(array && array.length>0 && index){
|
if(array && array.length>0 && index){
|
||||||
|
@ -179,26 +273,35 @@ function CoderDepot(props){
|
||||||
function returnMain(){
|
function returnMain(){
|
||||||
setTreeValue(undefined);
|
setTreeValue(undefined);
|
||||||
let branch = branchName || defaultBranch;
|
let branch = branchName || defaultBranch;
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/tree/${branch}`);
|
let checkvalue = turnbar(branch);
|
||||||
|
props.history.push(`/${owner}/${projectsId}/tree/${checkvalue}`);
|
||||||
};
|
};
|
||||||
// 子目录路径返回链接
|
// 子目录路径返回链接
|
||||||
function returnUlr(url){
|
function returnUlr(url){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/tree${branchName?`/${branchName}`:""}/${url}`);
|
let enBranch = turnbar(branchName);
|
||||||
|
props.history.push(`/${owner}/${projectsId}/tree${enBranch?`/${enBranch}`:""}/${url}`);
|
||||||
}
|
}
|
||||||
// 点击跳转到子目录
|
// 点击跳转到子目录
|
||||||
function goToSubRoot(path,type,filename){
|
function goToSubRoot(path,type,filename){
|
||||||
setType(type);
|
if(type!=="submodule"){
|
||||||
props.history.push(`/projects/${owner}/${projectsId}${`/tree/${branchName || defaultBranch}`}${path?`/${path}`:""}`);
|
let enBranch = branchName || defaultBranch;
|
||||||
|
let checkvalue = turnbar(enBranch);
|
||||||
|
setType(type);
|
||||||
|
props.history.push(`/${owner}/${projectsId}${`/tree/${checkvalue}`}${path?`/${path}`:""}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function onEdit(readOnly){
|
function onEdit(readOnly){
|
||||||
setReadOnly(readOnly);
|
setReadOnly(readOnly);
|
||||||
|
setEditReadme(false);
|
||||||
}
|
}
|
||||||
function ChangeFile(path, readOnly){
|
function ChangeFile(path, readOnly){
|
||||||
//点击直接跳转页面 加载一次路由
|
//点击直接跳转页面 加载一次路由
|
||||||
props.history.push(`/projects/${owner}/${projectsId}/tree/${branchName || defaultBranch}/${path}`);
|
let enBranch = branchName || defaultBranch;
|
||||||
|
let checkvalue = turnbar(enBranch);
|
||||||
|
props.history.push(`/${owner}/${projectsId}/tree/${checkvalue}/${path}`);
|
||||||
setType("file");
|
setType("file");
|
||||||
setReadOnly(readOnly);
|
setEditReadme(true);
|
||||||
};
|
};
|
||||||
|
|
||||||
function changeHide(hide){
|
function changeHide(hide){
|
||||||
|
@ -214,17 +317,12 @@ function CoderDepot(props){
|
||||||
}
|
}
|
||||||
|
|
||||||
const downloadMenu = (
|
const downloadMenu = (
|
||||||
<div className="downMenu">
|
<CloneAddress
|
||||||
<div style={{padding:"20px",borderBottom:"1px solid #eee"}}>
|
http_url={projectDetail && projectDetail.clone_url}
|
||||||
<CloneAddress
|
ssh_url = {projectDetail && projectDetail.ssh_url}
|
||||||
http_url={projectDetail && projectDetail.clone_url}
|
zip_url={zip_url}
|
||||||
showNotification={props.showNotification}/>
|
tar_url={tar_url}
|
||||||
</div>
|
showNotification={props.showNotification}/>
|
||||||
<Menu className="edu-txt-center">
|
|
||||||
<Menu.Item><a href={zip_url}>下载 ZIP</a></Menu.Item>
|
|
||||||
<Menu.Item><a href={tar_url}>下载 TAR.GZ</a></Menu.Item>
|
|
||||||
</Menu>
|
|
||||||
</div>
|
|
||||||
)
|
)
|
||||||
// 确认修改简介、website、实践课程链接
|
// 确认修改简介、website、实践课程链接
|
||||||
function okUpdate(d,w,l){
|
function okUpdate(d,w,l){
|
||||||
|
@ -239,10 +337,14 @@ function CoderDepot(props){
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
let n = fileInfo && fileInfo.name;
|
let n = fileInfo && fileInfo.name;
|
||||||
const mdFlag = n && n.substring(n.length-3,n.length) === ".md";
|
const mdFlag = n && n.substring(n.length-3,n.length) === ".md";
|
||||||
|
|
||||||
|
const { current_user } = props;
|
||||||
|
const baseOper = current_user && current_user.login && issuesFlag;
|
||||||
|
const baseOperate = projectDetail && projectDetail.permission && projectDetail.permission !=="Reporter" && projectDetail.type !== 2 && pullsFlag;
|
||||||
|
const fileOperate = type === "dir" && projectDetail && projectDetail.type !== 2 && ((projectDetail.permission && projectDetail.permission !=="Reporter") || (current_user && current_user.admin));
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<WhiteBack>
|
<WhiteBack>
|
||||||
<UpdateDescModal desc={desc} website={website} lesson_url={lesson_url} visible={openModal} onCancel={()=>setOpenModal(false)} onOk={okUpdate}/>
|
<UpdateDescModal desc={desc} website={website} lesson_url={lesson_url} visible={openModal} onCancel={()=>setOpenModal(false)} onOk={okUpdate}/>
|
||||||
|
@ -261,7 +363,7 @@ function CoderDepot(props){
|
||||||
list = {mainFlag ? dirInfo : undefined}
|
list = {mainFlag ? dirInfo : undefined}
|
||||||
/>
|
/>
|
||||||
<div className="drawerBtn" onClick={()=>setVisible(true)}>
|
<div className="drawerBtn" onClick={()=>setVisible(true)}>
|
||||||
<i className="iconfont icon-youjiantou font-16"></i>
|
<i className="iconfont icon-zuohuaicon font-14"></i>
|
||||||
<span>目录</span>
|
<span>目录</span>
|
||||||
</div>
|
</div>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
|
@ -274,7 +376,7 @@ function CoderDepot(props){
|
||||||
<div className="panelmenu">
|
<div className="panelmenu">
|
||||||
<FlexAJ>
|
<FlexAJ>
|
||||||
<AlignCenter>
|
<AlignCenter>
|
||||||
<div className="mr20">
|
<div className="mr30">
|
||||||
{
|
{
|
||||||
props && props.platform ?
|
props && props.platform ?
|
||||||
<SelectBranch
|
<SelectBranch
|
||||||
|
@ -287,66 +389,89 @@ function CoderDepot(props){
|
||||||
branchList={projectDetail && projectDetail.branches && projectDetail.branches.list}
|
branchList={projectDetail && projectDetail.branches && projectDetail.branches.list}
|
||||||
></SelectBranch>
|
></SelectBranch>
|
||||||
:
|
:
|
||||||
<span>分支:<span className="color-grey-6">{branchName || defaultBranch}</span></span>
|
<span>分支:<span className="color-grey-6">{branchName || defaultBranch}</span></span>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
<AlignCenter className="mr20">
|
{
|
||||||
<Link to={`/projects/${owner}/${projectsId}/branchs`} className="color-grey-9">
|
treeValuePath && treeValuePath.length > 0 ?
|
||||||
<i className="iconfont icon-fenzhi2 font-18 color-grey-9 mr3"></i>
|
<Path
|
||||||
<span className="color-grey-6 mr3">{projectDetail && projectDetail.branches && projectDetail.branches.total_count}个</span>分支
|
identifier={projectDetail && projectDetail.identifier}
|
||||||
</Link>
|
treeValuePath={treeValuePath}
|
||||||
</AlignCenter>
|
returnUlr={returnUlr}
|
||||||
<AlignCenter className="mr20">
|
returnMain={returnMain}
|
||||||
<Link to={`/projects/${owner}/${projectsId}/tag`} className="color-grey-9">
|
getPathUrl={getPathUrl}
|
||||||
<i className="iconfont icon-biaoqian3 font-16 color-grey-9 mr3"></i>
|
/>
|
||||||
<span className="color-grey-6 mr3">{projectDetail && projectDetail.tags && projectDetail.tags.total_count}个</span>标签
|
:
|
||||||
</Link>
|
<React.Fragment>
|
||||||
</AlignCenter>
|
<AlignCenter className="mr20">
|
||||||
|
<Link to={`/${owner}/${projectsId}/branches`} className="iconBtn">
|
||||||
|
<i className="iconfont icon-master_icon font-16"></i>
|
||||||
|
<span>分支</span>
|
||||||
|
<span>{projectDetail && projectDetail.branches_count}</span>
|
||||||
|
</Link>
|
||||||
|
</AlignCenter>
|
||||||
|
<AlignCenter className="mr20">
|
||||||
|
<Link to={`/${owner}/${projectsId}/tags`} className="iconBtn">
|
||||||
|
<i className="iconfont icon-biaoqianicon font-16"></i>
|
||||||
|
<span>标签</span>
|
||||||
|
<span>{projectDetail && projectDetail.tags_count}</span>
|
||||||
|
</Link>
|
||||||
|
</AlignCenter>
|
||||||
|
</React.Fragment>
|
||||||
|
}
|
||||||
</AlignCenter>
|
</AlignCenter>
|
||||||
<AlignCenter>
|
<AlignCenter className="depotBtn">
|
||||||
<div className="mr20 addOptionBtn">
|
{
|
||||||
{
|
(baseOperate || baseOper) &&
|
||||||
projectDetail.type !== 2 &&
|
<div className="addOptionBtn">
|
||||||
<a onClick={()=>urlLink(`/projects/${owner}/${projectsId}/pulls/new`)} >+ 合并请求</a>
|
{
|
||||||
}
|
baseOperate &&
|
||||||
<a onClick={()=>urlLink(`/projects/${owner}/${projectsId}/issues/new`)} >+ 任务</a>
|
<CheckProfile {...props} sureFunc={()=>urlLink(`/${owner}/${projectsId}/compare/master...${branchName || defaultBranch}`)} >+ 合并请求</CheckProfile>
|
||||||
</div>
|
}
|
||||||
{ type === "dir" && projectDetail.type !== 2 &&
|
{
|
||||||
<Dropdown overlay={fileMenu} className="mr20" trigger={['click']}>
|
baseOper &&
|
||||||
<Button type="default">文件 <i className="iconfont icon-sanjiaoxing-down ml3 font-14 color-grey-9"></i></Button>
|
<CheckProfile {...props} sureFunc={()=>urlLink(`/${owner}/${projectsId}/issues/new`)} >+ 易修</CheckProfile>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
{ fileOperate &&
|
||||||
|
<Dropdown
|
||||||
|
overlay={fileMenu()}
|
||||||
|
className="mr10"
|
||||||
|
trigger={['click']}
|
||||||
|
getPopupContainer={document.parentNode}
|
||||||
|
>
|
||||||
|
<a>文件 <i className="iconfont icon-sanjiaoxing-down ml3 font-14 color-grey-6 mr-5"></i></a>
|
||||||
</Dropdown>
|
</Dropdown>
|
||||||
}
|
}
|
||||||
|
|
||||||
<Dropdown overlay={downloadMenu} placement="bottomRight" trigger={['click']}>
|
<Dropdown overlay={downloadMenu} placement="bottomRight" trigger={['click']}>
|
||||||
<Button type={'primary'}>下载 <i className="iconfont icon-sanjiaoxing-down ml3 font-14 color-white"></i></Button>
|
<Button type={'primary'}>下载 <i className="iconfont icon-sanjiaoxing-down ml3 font-14 color-white mr-3"></i></Button>
|
||||||
</Dropdown>
|
</Dropdown>
|
||||||
</AlignCenter>
|
</AlignCenter>
|
||||||
</FlexAJ>
|
</FlexAJ>
|
||||||
{
|
{
|
||||||
dirInfo || fileInfo ?
|
(dirInfo && dirInfo.length>0) || fileInfo ?
|
||||||
<div className="listtable">
|
<div className="listtable">
|
||||||
{
|
{
|
||||||
lastCommit &&
|
lastCommit &&
|
||||||
<div className="listtablehead">
|
<div className="listtablehead">
|
||||||
<User url={getImageUrl(`/${lastCommitAuthor && lastCommitAuthor.image_url}`)} name={lastCommitAuthor && lastCommitAuthor.name} id={lastCommitAuthor && lastCommitAuthor.id} login={lastCommitAuthor && lastCommitAuthor.login}/>
|
<User url={getImageUrl(`/${lastCommitAuthor && lastCommitAuthor.image_url}`)} name={lastCommitAuthor && lastCommitAuthor.name} id={lastCommitAuthor && lastCommitAuthor.id} login={lastCommitAuthor && lastCommitAuthor.login}/>
|
||||||
<div className={hideBtn && hide ? "ellipsistxt hide" :"ellipsistxt"}><pre id="ptxt">{lastCommit && lastCommit.message}</pre></div>
|
<div className={hideBtn && hide ? "ellipsistxt hidetxt" :"ellipsistxt"}>
|
||||||
|
<pre id="ptxt"><Link to={`/${owner}/${projectsId}/commits/${truncateCommitId(lastCommit.sha)}`}><RenderHtml value={lastCommit.message}/></Link></pre>
|
||||||
|
</div>
|
||||||
{ hideBtn && <span className="ellipsis" onClick={()=>changeHide(hide)}><i className="iconfont icon-shenglvehao"></i></span> }
|
{ hideBtn && <span className="ellipsis" onClick={()=>changeHide(hide)}><i className="iconfont icon-shenglvehao"></i></span> }
|
||||||
|
|
||||||
<span className="ml12 color-grey-9 mt3">{lastCommit && lastCommit.time_from_now}</span>
|
<span className="ml20 color-grey-6 font-12 mt3">{lastCommit.time_from_now}</span>
|
||||||
{ commitCount ? <Link to={`/projects/${owner}/${projectsId}/commits`} className="ml12 color-grey-9"><i className="iconfont icon-tijiao mr3 font-17 color-grey-9"></i>{commitCount}次提交</Link>:"" }
|
{
|
||||||
|
commitCount ?
|
||||||
|
<Link to={`/${owner}/${projectsId}/commits/branch/${turnbar(branchName || defaultBranch)}`} className="ml20 color-grey-3"style={{height:"28px",lineHeight:"28px"}}>
|
||||||
|
<i className="iconfont icon-tijiaoicon mr3 font-16"></i><span style={{fontWeight:"500"}}>{commitCount}次提交</span>
|
||||||
|
</Link>:""
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
<ul className="listtablebody">
|
<ul className="listtablebody">
|
||||||
{
|
|
||||||
treeValuePath && treeValuePath.length > 0 &&
|
|
||||||
<Path
|
|
||||||
identifier={projectDetail && projectDetail.identifier}
|
|
||||||
treeValuePath={treeValuePath}
|
|
||||||
returnUlr={returnUlr}
|
|
||||||
returnMain={returnMain}
|
|
||||||
getPathUrl={getPathUrl}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
{
|
{
|
||||||
dirInfo && dirInfo.length > 0 &&
|
dirInfo && dirInfo.length > 0 &&
|
||||||
dirInfo.map((item,key)=>{
|
dirInfo.map((item,key)=>{
|
||||||
|
@ -377,64 +502,80 @@ function CoderDepot(props){
|
||||||
: ""
|
: ""
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
(dirInfo && dirInfo.length === 0) && (fileInfo && fileInfo.length === 0) ? <Nodata _html="暂未发现文件"/> :""
|
(dirInfo && dirInfo.length === 0) && !fileInfo ? <Nodata _html="暂未发现文件"/> :""
|
||||||
}
|
}
|
||||||
{/* readme文件显示(显示文件详情时不显示readme文件) */}
|
{/* readme文件显示(显示文件详情时不显示readme文件) */}
|
||||||
{ dirInfo && (readme && readme.content) ? <ReadMe ChangeFile={ChangeFile} readme={readme} operate={props && (props.isManager || props.isDeveloper) && projectDetail.type !==2 } history={props.history} /> :"" }
|
{ (readme && readme.content) ? <ReadMe ChangeFile={ChangeFile} readme={readme} operate={props && (props.isManager || props.isDeveloper) && projectDetail.type !==2 } history={props.history} /> :"" }
|
||||||
</div>
|
</div>
|
||||||
</LongWidth>
|
</LongWidth>
|
||||||
{
|
{
|
||||||
!fileInfo &&
|
(!(treeValuePath && treeValuePath.length > 0) && !fileInfo) &&
|
||||||
<ShortWidth>
|
<ShortWidth>
|
||||||
<Gap style={{paddingLeft:"30px"}}>
|
<Gap style={{paddingLeft:"30px"}}>
|
||||||
<div className="panelmenu">
|
<div className="panelmenu">
|
||||||
<FlexAJ className="font-18 color-grey-6 mb20" style={{lineHeight:"28px"}}>简介
|
<FlexAJ className="font-18 color-ooo mb20" style={{lineHeight:"28px"}}>关于
|
||||||
{projectDetail.permission && (projectDetail.permission==="Admin" || projectDetail.permission==="Owner") && <i onClick={()=>setOpenModal(true)} className="iconfont icon-anquanshezhi color-grey-9 font-15"></i>}
|
{
|
||||||
|
projectDetail.permission && (projectDetail.permission==="Admin" || projectDetail.permission==="Owner" || projectDetail.permission==="Manager") &&
|
||||||
|
<i onClick={()=>setOpenModal(true)} className="iconfont icon-a-shezhi color-grey-9 font-15"></i>
|
||||||
|
}
|
||||||
</FlexAJ>
|
</FlexAJ>
|
||||||
{desc && <p className="font-14 color-grey-9 mb15 task-hide-2" style={{lineHeight:"22px",WebkitLineClamp:"4",textAlign:"justify",wordBreak:"break-all"}}>{desc}</p>}
|
{desc && <p className="font-14 color-grey-3 mb15 task-hide-2" style={{lineHeight:"24px",WebkitLineClamp:"4",textAlign:"justify",wordBreak:"break-all"}}>{desc}</p>}
|
||||||
{
|
{
|
||||||
website &&
|
website &&
|
||||||
<p className="color-grey-6 df">
|
<div className="color-grey-6 df pinfos mb5">
|
||||||
<i className="iconfont icon-lianjie2 font-15 mr10 color-grey-9"></i>
|
<i className="iconfont icon-lianjie2 font-15 mr10"></i>
|
||||||
<a href={website} className="color-grey-6" target="_blank" style={{wordBreak:"break-all",lineHeight:"20px",marginTop:"5px",textDecoration:"underline"}}>{website}</a>
|
<a href={website} target="_blank" style={{wordBreak:"break-all",lineHeight:"20px",marginTop:"5px",textDecoration:"underline"}}>{website}</a>
|
||||||
</p>
|
</div>
|
||||||
}
|
}
|
||||||
<p>
|
<div className="pinfos mb5">
|
||||||
<i className="iconfont icon-wenjian4 font-15 mr10 color-grey-9"></i>
|
<i className="iconfont icon-zishuwenjian_icon font-15 mr10"></i>
|
||||||
<a href="#readme" className="color-grey-6">README.md</a>
|
<a href="#readme">README.md</a>
|
||||||
</p>
|
</div>
|
||||||
<p className="color-grey-6">
|
<div className="color-grey-6 mb5">
|
||||||
<i className="iconfont icon-dataBase font-15 mr10 color-grey-6"></i>
|
<i className="iconfont icon-neicunicon font-15 mr10"></i>
|
||||||
<span>{projectDetail && projectDetail.size}</span>
|
<span>{projectDetail && projectDetail.size}</span>
|
||||||
</p>
|
</div>
|
||||||
{
|
{
|
||||||
projectDetail && projectDetail.license_name &&
|
projectDetail && projectDetail.license_name &&
|
||||||
<p className="color-grey-6">
|
<div className="pinfos">
|
||||||
<i className="iconfont icon-tianping font-16 mr10 color-grey-3"></i>
|
<i className="iconfont icon-xieyiicon font-16 mr10"></i>
|
||||||
<span>{projectDetail.license_name}</span>
|
<Link to={`/${owner}/${projectsId}/tree/${branchName || defaultBranch}/LICENSE`} className="color-grey-6">{projectDetail.license_name}</Link>
|
||||||
</p>
|
</div>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
{
|
||||||
|
inviteCode &&
|
||||||
|
<div>
|
||||||
|
<Divider />
|
||||||
|
<Invite code={inviteCode}/>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
{
|
{
|
||||||
lesson_url &&
|
lesson_url &&
|
||||||
<div>
|
<div>
|
||||||
<Divider />
|
<Divider />
|
||||||
<p className="font-16 color-grey-6">实践课程</p>
|
<p className="font-16 color-ooo">实践课程</p>
|
||||||
<a href={lesson_url} target="_blank" className="color-grey-6" style={{textDecoration:"underline"}}>{lesson_url}</a>
|
<a href={lesson_url} target="_blank" className="color-grey-6" style={{textDecoration:"underline",wordBreak:"break-all"}}>{lesson_url}</a>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
{/* 发布 */}
|
{/* 发布 */}
|
||||||
{
|
{
|
||||||
projectDetail && projectDetail.release_versions &&
|
releaseVersions &&
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<Divider />
|
<Divider />
|
||||||
<Releases owner={owner} projectsId={projectsId} releaseVersions={projectDetail.release_versions} history={props.history}/>
|
<Releases
|
||||||
|
owner={owner}
|
||||||
|
projectsId={projectsId}
|
||||||
|
releaseVersions={releaseVersions}
|
||||||
|
history={props.history}
|
||||||
|
distribution={distribution}
|
||||||
|
/>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
}
|
}
|
||||||
{/* 贡献者 */}
|
{/* 贡献者 */}
|
||||||
{
|
{
|
||||||
projectDetail && projectDetail.contributors &&
|
projectDetail && projectDetail.contributors && projectDetail.contributors.total_count >0 &&
|
||||||
<Contributors contributors={projectDetail && projectDetail.contributors} owner={owner} projectsId={projectsId} />
|
<Contributors contributors={projectDetail.contributors} owner={owner} projectsId={projectsId} />
|
||||||
}
|
}
|
||||||
{/* 语言 */}
|
{/* 语言 */}
|
||||||
{ projectDetail && projectDetail.languages &&
|
{ projectDetail && projectDetail.languages &&
|
||||||
|
|
|
@ -2,20 +2,26 @@ import React from 'react';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
import { truncateCommitId } from '../common/util';
|
import { truncateCommitId } from '../common/util';
|
||||||
|
|
||||||
|
const typeIco = {
|
||||||
|
"submodule":"icon-file-submodule font-17",
|
||||||
|
"file":'icon-wenjian6 font-15 color-blue-file',
|
||||||
|
"dir":"icon-wenjianjia4 font-15 color-blue_4C"
|
||||||
|
}
|
||||||
|
|
||||||
function CoderDepotCatalogue({item , goToSubRoot , owner , projectsId }){
|
function CoderDepotCatalogue({item , goToSubRoot , owner , projectsId }){
|
||||||
return(
|
return(
|
||||||
<li>
|
<li>
|
||||||
<span>
|
<span>
|
||||||
<a onClick={()=>goToSubRoot(item.path,item.type,item.name)}>
|
<a onClick={()=>goToSubRoot(item.path,item.type,item.name)} className={item.type === "submodule" && "submoduleStyle"}>
|
||||||
<i className={item.type === 'dir' ? "iconfont icon-wenjianjia1 color-green-file font-15 mr5":"iconfont icon-wenjia color-green-file font-15 mr5"}></i>{item.name}
|
<i className={`iconfont ${typeIco[`${item.type}`]} mr8`}></i>{item.name}
|
||||||
</a>
|
</a>
|
||||||
</span>
|
</span>
|
||||||
<span title="init project">
|
<span title="init project">
|
||||||
<Link to={`/projects/${owner}/${projectsId}/commits/${truncateCommitId(`${item.commit && item.commit.sha}`)}`} title={item.commit && item.commit.message}>
|
<Link to={`/${owner}/${projectsId}/commits/${truncateCommitId(`${item.commit && item.commit.sha}`)}`} title={item.commit && item.commit.message}>
|
||||||
{item.commit && item.commit.message}
|
{item.commit && item.commit.message}
|
||||||
</Link>
|
</Link>
|
||||||
</span>
|
</span>
|
||||||
<span>{item.commit && item.commit.time_from_now}</span>
|
<span title={item.commit && item.commit.created_at}>{item.commit && item.commit.time_from_now}</span>
|
||||||
</li>
|
</li>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import { result } from 'lodash';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import RenderHtml from '../../components/render-html';
|
import RenderHtml from '../../components/render-html';
|
||||||
import { Dropdown , Menu , Spin } from 'antd';
|
import { AlignCenter } from '../Component/layout';
|
||||||
import { Link } from 'react-router-dom';
|
import { Dropdown , Anchor , Spin } from 'antd';
|
||||||
|
|
||||||
|
import ReadmeCatelogue from './sub/ReadmeCatelogue';
|
||||||
const $ = window.$;
|
const $ = window.$;
|
||||||
|
|
||||||
function CoderDepotReadme({ operate , history , readme , ChangeFile }){
|
function CoderDepotReadme({ operate , history , readme , ChangeFile }){
|
||||||
|
@ -22,44 +24,45 @@ function CoderDepotReadme({ operate , history , readme , ChangeFile }){
|
||||||
const anchor = el.id;
|
const anchor = el.id;
|
||||||
const level = el.tagName.replace("H", "");
|
const level = el.tagName.replace("H", "");
|
||||||
const href = `#${anchor}`;
|
const href = `#${anchor}`;
|
||||||
return { href:`${path}${href}`,text:el.textContent , level:level }
|
return { href:`${href}`,text:el.textContent , level:level }
|
||||||
});
|
});
|
||||||
setMenuList(items);
|
setMenuList(items);
|
||||||
},[content])
|
},[content])
|
||||||
|
|
||||||
function menu(){
|
function menu(){
|
||||||
if(menuList && menuList.length > 0){
|
if(menuList && menuList.length > 0){
|
||||||
let hash = history.location.hash;
|
|
||||||
return(
|
return(
|
||||||
<Menu className="menuslist">
|
<ReadmeCatelogue menuList={menuList} hash={history.location.hash}/>
|
||||||
{
|
|
||||||
menuList.map((item,key)=>{
|
|
||||||
return(
|
|
||||||
<Menu.Item key={item.id} className={decodeURI(hash).indexOf(item.text)>-1 ?"active":""}><Link to={`${item.href}`} style={{paddingLeft:`${item.level *10}px`}} title={item.text}>{item.text}</Link></Menu.Item>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</Menu>
|
|
||||||
)
|
)
|
||||||
}else{
|
}else{
|
||||||
return <Spin />
|
return <Spin />
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return(
|
return(
|
||||||
<div className="commonBox" id="readme">
|
<div className="commonBox readBox" id="readme">
|
||||||
<div className="commonBox-title">
|
<Anchor offsetTop={70} targetOffset={160}>
|
||||||
<Dropdown overlay={menu()}>
|
<div className="commonBox-title boxTitle">
|
||||||
<i className="iconfont icon-zhangjie1 font-16 color-grey-3 mr10"></i>
|
<AlignCenter>
|
||||||
</Dropdown>
|
<Dropdown overlay={menu()} trigger={['hover']} overlayClassName="menuslist">
|
||||||
<span className="commonBox-title-read">README.md</span>
|
<span className="catelogue">
|
||||||
{
|
<i className="iconfont icon-muluicon font-12 mr5"></i>
|
||||||
operate ?
|
<span>目录</span>
|
||||||
<a className="ml20 pull-right" onClick={() =>ChangeFile(readme && readme.path, false)}>
|
</span>
|
||||||
<i className="iconfont icon-bianji6 font-16 color-blue"></i>
|
</Dropdown>
|
||||||
</a>
|
|
||||||
:""
|
<span className="commonBox-title-read"><a href="#readme ">README.md</a></span>
|
||||||
}
|
|
||||||
</div>
|
</AlignCenter>
|
||||||
|
{
|
||||||
|
operate ?
|
||||||
|
<a className="ml20 pull-right" onClick={() =>ChangeFile(readme && readme.path, false)}>
|
||||||
|
<i className="iconfont icon-a-bianji font-17 color-grey-6"></i>
|
||||||
|
</a>
|
||||||
|
:""
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</Anchor>
|
||||||
{
|
{
|
||||||
content &&
|
content &&
|
||||||
<div className="commonBox-info">
|
<div className="commonBox-info">
|
||||||
|
|
|
@ -1,21 +1,36 @@
|
||||||
import React , { Component } from 'react';
|
import React , { Component } from 'react';
|
||||||
import { Spin , Pagination } from 'antd';
|
import { Spin , Pagination, Timeline } from 'antd';
|
||||||
import { getImageUrl } from 'educoder';
|
import { getImageUrl } from 'educoder';
|
||||||
import { truncateCommitId } from '../common/util';
|
import { truncateCommitId ,timeFormat } from '../common/util';
|
||||||
|
import { AlignTop } from '../Component/layout';
|
||||||
import SelectBranch from '../Branch/Select';
|
import SelectBranch from '../Branch/Select';
|
||||||
import Nodata from '../Nodata';
|
import Nodata from '../Nodata';
|
||||||
|
|
||||||
|
import User from '../Component/User';
|
||||||
|
import RenderHtml from '../../components/render-html.jsx';
|
||||||
|
import Tree from './img/tree.png';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import {Link} from "react-router-dom";
|
import {Link} from "react-router-dom";
|
||||||
|
import CopyTool from '../Component/CopyTool';
|
||||||
|
|
||||||
|
import './tree/Index.scss'
|
||||||
|
|
||||||
|
function returnbar(str){
|
||||||
|
if(str && str.length>0 && str.indexOf("%2F")>-1){
|
||||||
|
return str.replaceAll('%2F','/');
|
||||||
|
}
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
//代码库--提交页面
|
||||||
class CoderRootCommit extends Component{
|
class CoderRootCommit extends Component{
|
||||||
constructor(props){
|
constructor(props){
|
||||||
super(props)
|
super(props);
|
||||||
this.state={
|
this.state={
|
||||||
commitDatas:undefined,
|
commitDatas:undefined,
|
||||||
dataCount:undefined,
|
dataCount:undefined,
|
||||||
limit:20,
|
limit:10,
|
||||||
page:1,
|
page: 1,
|
||||||
isSpining:false,
|
isSpining:false,
|
||||||
branchList:undefined
|
branchList:undefined
|
||||||
}
|
}
|
||||||
|
@ -43,13 +58,28 @@ class CoderRootCommit extends Component{
|
||||||
this.Init();
|
this.Init();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
UrlParamHash(url){
|
||||||
|
const params = {};
|
||||||
|
let h;
|
||||||
|
let hash = url.slice(url.indexOf('?')+1).split('&');
|
||||||
|
for(let i = 0; i<hash.length;i++){
|
||||||
|
h = hash[i].split('=');
|
||||||
|
params[h[0]] = h[1];
|
||||||
|
}
|
||||||
|
return params;
|
||||||
|
}
|
||||||
|
|
||||||
Init =()=>{
|
Init =()=>{
|
||||||
const { branchName } = this.props.match.params;
|
const { branchName } = this.props.match.params;
|
||||||
const { page , limit } = this.state;
|
const { limit } = this.state;
|
||||||
|
const {search} = this.props.location;
|
||||||
|
const realPage = (search && this.UrlParamHash(search).page) ? parseInt(this.UrlParamHash(search).page) : 1;
|
||||||
this.setState({
|
this.setState({
|
||||||
isSpining:true
|
isSpining:true,
|
||||||
|
page:realPage
|
||||||
})
|
})
|
||||||
this.getCommitList( branchName , page , limit );
|
this.getCommitList( branchName , realPage , limit );
|
||||||
}
|
}
|
||||||
|
|
||||||
getCommitList=(branch , page , limit)=>{
|
getCommitList=(branch , page , limit)=>{
|
||||||
|
@ -60,7 +90,7 @@ class CoderRootCommit extends Component{
|
||||||
const url = `/${owner}/${projectsId}/commits.json`;
|
const url = `/${owner}/${projectsId}/commits.json`;
|
||||||
axios.get(url,{
|
axios.get(url,{
|
||||||
params:{
|
params:{
|
||||||
sha:branch,
|
sha:returnbar(branch),
|
||||||
page,
|
page,
|
||||||
limit
|
limit
|
||||||
}
|
}
|
||||||
|
@ -78,7 +108,8 @@ class CoderRootCommit extends Component{
|
||||||
image_url:item.author && item.author.image_url,
|
image_url:item.author && item.author.image_url,
|
||||||
sha:item.sha,
|
sha:item.sha,
|
||||||
time_from_now:item.time_from_now,
|
time_from_now:item.time_from_now,
|
||||||
message:item.message
|
message:item.message,
|
||||||
|
timestamp:item.timestamp
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
this.setState({
|
this.setState({
|
||||||
|
@ -93,19 +124,18 @@ class CoderRootCommit extends Component{
|
||||||
// 切换分支 search:tag为根据标签搜索
|
// 切换分支 search:tag为根据标签搜索
|
||||||
changeBranch=(value)=>{
|
changeBranch=(value)=>{
|
||||||
const { projectsId , owner } = this.props.match.params;
|
const { projectsId , owner } = this.props.match.params;
|
||||||
this.props.history.push(`/projects/${owner}/${projectsId}/commits/branch/${value}`);
|
this.props.history.push(`/${owner}/${projectsId}/commits/branch/${value}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
ChangePage=(page)=>{
|
ChangePage=(page)=>{
|
||||||
const { branchName } = this.props.match.params;
|
this.props.history.push({pathname: this.props.history.location.pathname,search: `page=${page}`})
|
||||||
const { limit } = this.state;
|
|
||||||
this.getCommitList(branchName , page , limit);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
render(){
|
render(){
|
||||||
const { commitDatas , dataCount , limit , page , isSpining , branchList } = this.state;
|
const { commitDatas , dataCount , limit , page , isSpining , branchList } = this.state;
|
||||||
const { projectDetail, commit_class , defaultBranch } = this.props;
|
const { projectDetail, commit_class , defaultBranch } = this.props;
|
||||||
const { projectsId , owner , branchName } = this.props.match.params;
|
const { projectsId , owner , branchName } = this.props.match.params;
|
||||||
let branch = branchName || defaultBranch;
|
let branch = returnbar(branchName || defaultBranch);
|
||||||
return(
|
return(
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<div className={"main"}style={{padding:"0px",border:"none"}}>
|
<div className={"main"}style={{padding:"0px",border:"none"}}>
|
||||||
|
@ -121,46 +151,50 @@ class CoderRootCommit extends Component{
|
||||||
></SelectBranch>
|
></SelectBranch>
|
||||||
</div>
|
</div>
|
||||||
<Spin spinning={isSpining}>
|
<Spin spinning={isSpining}>
|
||||||
<div className="commonBox">
|
<Timeline className="commitList">
|
||||||
<div className="commonBox-title">
|
{
|
||||||
<div className="f-wrap-between" style={{alignItems:"center"}}>
|
commitDatas && commitDatas.length > 0 && commitDatas.map((item,k)=>{
|
||||||
<span className="font-16">{dataCount}次提交代码({branch})</span>
|
return(
|
||||||
</div>
|
<Timeline.Item key={k} dot={page ===1 && k===0 ?<span className="new-conmmit">最新</span>:<i className="iconfont icon-a-yuanquan2x"></i>}>
|
||||||
</div>
|
<div className="commitList-item f-wrap-between">
|
||||||
<div className="commitList">
|
<div>
|
||||||
{
|
<AlignTop>
|
||||||
commitDatas && commitDatas.length > 0 && commitDatas.map((item,k)=>{
|
<div className="commitDesc"><Link to={`/${owner}/${projectsId}/commits/${truncateCommitId(`${item.sha}`)}`} className="font-14 color-grey-3 font-bd"><RenderHtml value={item.message}/></Link></div>
|
||||||
return(
|
</AlignTop>
|
||||||
<div key={k}>
|
<p className="f-wrap-alignCenter mt15 pb5">
|
||||||
<p className="f-wrap-alignCenter">
|
<User
|
||||||
<Link to={`/projects/${owner}/${projectsId}/commits/${truncateCommitId(`${item.sha}`)}`} className="commitKey" style={{marginLeft:0}}>{truncateCommitId(`${item.sha}`)}</Link>
|
id={item.id}
|
||||||
<Link to={`/projects/${owner}/${projectsId}/commits/${truncateCommitId(`${item.sha}`)}`} className="flex1 ml20 font-16 color-grey-3">{item.message}</Link>
|
url={(item.image_url && getImageUrl(`/${item.image_url}`)) || "https://dss3.bdstatic.com/70cFv8Sh_Q1YnxGkpoWK1HF6hhy/it/u=3025493530,1989042357&fm=26&gp=0.jpg"}
|
||||||
</p>
|
name={item.name}
|
||||||
<p className="f-wrap-alignCenter mt15">
|
login={item.login}
|
||||||
{
|
/>
|
||||||
item.id ?
|
{item.timestamp && <label className="font-14 color-grey-3 ml3">提交于 {timeFormat(item.timestamp)}</label>}
|
||||||
<Link to={`/users/${item.login}`} className="show-user-link">
|
</p>
|
||||||
{item.image_url?<img src={getImageUrl(`/${item.image_url}`)} alt="" width="28px" height="28px" className="mr15 radius"/>:""}
|
</div>
|
||||||
<label className="font-14 color-grey-6" style={{verticalAlign:'middle'}}>{item.name ?`${item.name}:`:""}提交于 {item.time_from_now}</label>
|
<div>
|
||||||
</Link>:
|
<div className="treecopy">
|
||||||
<span className="show-user-link">
|
<div>
|
||||||
{item.image_url?<img src={getImageUrl(`/${item.image_url}`)} alt="" width="28px" height="28px" className="mr15 radius"/>:""}
|
<span className="treecopy-cont shadow">
|
||||||
<label className="font-14 color-grey-6" style={{verticalAlign:'middle'}}>{item.name ?`${item.name}:`:""}提交于 {item.time_from_now}</label>
|
<img src={Tree} alt="sha" width={"16px"}/>
|
||||||
</span>
|
<Link to={`/${owner}/${projectsId}/commits/${truncateCommitId(`${item.sha}`)}`}>{truncateCommitId(`${item.sha}`)}</Link>
|
||||||
}
|
<input type="text" id={`value${k}`} value={`${truncateCommitId(`${item.sha}`)}`}/>
|
||||||
|
</span>
|
||||||
</p>
|
<CopyTool beforeText="复制commit id" afterText="复制成功" inputId={`value${k}`}/>
|
||||||
|
</div>
|
||||||
|
<button className="btn-83" onClick={()=>{window.location.href=`/${owner}/${projectsId}/tree/${truncateCommitId(item.sha)}`}}>浏览文件</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
</Timeline.Item>
|
||||||
})
|
)
|
||||||
}
|
})
|
||||||
{commitDatas && commitDatas.length === 0 && <Nodata _html="暂无数据"/>}
|
}
|
||||||
</div>
|
{commitDatas && commitDatas.length === 0 && <Nodata _html="暂无数据"/>}
|
||||||
</div>
|
</Timeline>
|
||||||
{
|
{
|
||||||
dataCount > limit ?
|
dataCount > limit ?
|
||||||
<div className="edu-txt-center pt30 mb30">
|
<div className="edu-txt-center pt30 mb30">
|
||||||
<Pagination simple defaultCurrent={page} total={dataCount} pageSize={limit} onChange={this.ChangePage}></Pagination>
|
<Pagination simple current={page} total={dataCount} pageSize={limit} onChange={this.ChangePage}></Pagination>
|
||||||
</div>
|
</div>
|
||||||
:""
|
:""
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
import React, { Component } from "react";
|
import React, { Component } from "react";
|
||||||
import { Popconfirm , Select } from "antd";
|
import { Popconfirm , Select , Dropdown , Spin , Anchor } from "antd";
|
||||||
import "./list.css";
|
import "./list.scss";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import Meditor from "../Newfile/m_editor";
|
import Meditor from "../Newfile/m_editor";
|
||||||
import RenderHtml from "../../components/render-html";
|
import RenderHtml from "../../components/render-html";
|
||||||
|
import ReadmeCatelogue from "./sub/ReadmeCatelogue";
|
||||||
|
|
||||||
|
const $ = window.$;
|
||||||
function bytesToSize(bytes) {
|
function bytesToSize(bytes) {
|
||||||
if (bytes === 0) return "0 B";
|
if (bytes === 0) return "0 B";
|
||||||
let k = 1024,
|
let k = 1024,
|
||||||
|
@ -19,11 +21,13 @@ class CoderRootFileDetail extends Component {
|
||||||
value: undefined,
|
value: undefined,
|
||||||
language: undefined,
|
language: undefined,
|
||||||
languages: undefined,
|
languages: undefined,
|
||||||
description: props.detail.content
|
description: props.detail.content,
|
||||||
|
menuList:undefined
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount = () => {
|
componentDidMount = () => {
|
||||||
|
window.scrollTo(0, 0);
|
||||||
const { detail , mdFlag } = this.props;
|
const { detail , mdFlag } = this.props;
|
||||||
this.setState({
|
this.setState({
|
||||||
value: detail.content,
|
value: detail.content,
|
||||||
|
@ -149,7 +153,7 @@ class CoderRootFileDetail extends Component {
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
if (result) {
|
if (result) {
|
||||||
this.props.showNotification("删除成功!");
|
this.props.showNotification("删除成功!");
|
||||||
this.props.history.push(`/projects/${owner}/${projectsId}`);
|
this.props.history.push(`/${owner}/${projectsId}`);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
|
@ -169,6 +173,31 @@ class CoderRootFileDetail extends Component {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
renderMenulist=()=>{
|
||||||
|
const { description } = this.state;
|
||||||
|
if(description){
|
||||||
|
const items = $.map($("#files-md").find("h1,h2,h3,h4,h5,h6"), function (el, _) {
|
||||||
|
const anchor = el.id;
|
||||||
|
const level = el.tagName.replace("H", "");
|
||||||
|
const href = `#${anchor}`;
|
||||||
|
return { href:`${href}`,text:el.textContent , level:level }
|
||||||
|
});
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
menu=()=>{
|
||||||
|
const menuList = this.renderMenulist();
|
||||||
|
if(menuList && menuList.length > 0){
|
||||||
|
return(
|
||||||
|
<ReadmeCatelogue menuList={menuList} hash={this.props.history.location.hash}/>
|
||||||
|
)
|
||||||
|
}else{
|
||||||
|
return <Spin />
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const {
|
const {
|
||||||
readOnly,
|
readOnly,
|
||||||
|
@ -186,79 +215,88 @@ class CoderRootFileDetail extends Component {
|
||||||
const Option = Select.Option;
|
const Option = Select.Option;
|
||||||
return (
|
return (
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<div className="grid-item branchTitle">
|
<Anchor className="griditemAnchor" offsetTop={70}>
|
||||||
<div className="grid-item">
|
<div className="griditemCate">
|
||||||
<span className="ml20 color-grey-6 font-16">
|
{
|
||||||
|
md && readOnly &&
|
||||||
|
<Dropdown overlay={this.menu()} trigger={['hover']} overlayClassName="menuslist">
|
||||||
|
<span className="catelogue mr20">
|
||||||
|
<i className="iconfont icon-muluicon font-12 mr5"></i>
|
||||||
|
<span>目录</span>
|
||||||
|
</span>
|
||||||
|
</Dropdown>
|
||||||
|
}
|
||||||
|
<span className="color-grey-6 font-16">
|
||||||
{bytesToSize(detail && detail.size)}
|
{bytesToSize(detail && detail.size)}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<p className="text-right">
|
<p className="text-right">
|
||||||
{flag && platform && (
|
{flag && platform && (
|
||||||
<div>
|
<div>
|
||||||
{readOnly ? (
|
{readOnly ? (
|
||||||
<span>
|
<span>
|
||||||
{
|
{
|
||||||
!detail.direct_download?
|
!detail.direct_download?
|
||||||
<span>
|
<span>
|
||||||
<a onClick={() => this.DownLoadFile(detail.download_url)} className="ml20">
|
<a onClick={() => this.DownLoadFile(detail.download_url)} className="ml20">
|
||||||
<i className="iconfont icon-xiazai1 font-15 color-grey-6"></i>
|
<i className="iconfont icon-xiazai1 font-15 color-grey-6"></i>
|
||||||
</a>
|
|
||||||
{
|
|
||||||
type !==2 &&
|
|
||||||
<a onClick={() => this.EditFile(false)} className="ml20">
|
|
||||||
<i className="iconfont icon-bianji1 font-15 color-grey-6"></i>
|
|
||||||
</a>
|
</a>
|
||||||
}
|
{
|
||||||
</span>:""
|
type !==2 &&
|
||||||
}
|
<a onClick={() => this.EditFile(false)} className="ml20">
|
||||||
</span>
|
<i className="iconfont icon-bianji1 font-15 color-grey-6"></i>
|
||||||
) : (
|
</a>
|
||||||
<React.Fragment>
|
}
|
||||||
<Select
|
</span>:""
|
||||||
showSearch={true}
|
}
|
||||||
placeholder={"请选择文本语言"}
|
</span>
|
||||||
style={{ width: 200 }}
|
) : (
|
||||||
value={language}
|
<React.Fragment>
|
||||||
onChange={this.select_language}
|
<Select
|
||||||
>
|
showSearch={true}
|
||||||
<Option value={undefined}>请选择文本语言</Option>
|
placeholder={"请选择文本语言"}
|
||||||
{languages &&
|
style={{ width: 200 }}
|
||||||
languages.map((item, key) => {
|
value={language}
|
||||||
return (
|
onChange={this.select_language}
|
||||||
<Option value={item} key={key}>
|
>
|
||||||
{item}
|
<Option value={undefined}>请选择文本语言</Option>
|
||||||
</Option>
|
{languages &&
|
||||||
);
|
languages.map((item, key) => {
|
||||||
})}
|
return (
|
||||||
</Select>
|
<Option value={item} key={key}>
|
||||||
<button
|
{item}
|
||||||
type="button"
|
</Option>
|
||||||
className="ant-btn ant-btn-sm ml20"
|
);
|
||||||
onClick={() => this.EditFile(true)}
|
})}
|
||||||
>
|
</Select>
|
||||||
<span>取 消</span>
|
<button
|
||||||
</button>
|
type="button"
|
||||||
</React.Fragment>
|
className="ant-btn ant-btn-sm ml20"
|
||||||
)}
|
onClick={() => this.EditFile(true)}
|
||||||
{
|
>
|
||||||
type !==2 &&
|
<span>取 消</span>
|
||||||
<Popconfirm
|
</button>
|
||||||
title="确认删除这个文件?"
|
</React.Fragment>
|
||||||
className="ml20"
|
)}
|
||||||
okText="确定"
|
{
|
||||||
cancelText="取消"
|
type !==2 &&
|
||||||
onConfirm={this.deleteFile}
|
<Popconfirm
|
||||||
>
|
title="确认删除这个文件?"
|
||||||
<a>
|
className="ml20"
|
||||||
<i className="iconfont icon-shanchu font-15 color-grey-6"></i>
|
okText="确定"
|
||||||
</a>
|
cancelText="取消"
|
||||||
</Popconfirm>
|
onConfirm={this.deleteFile}
|
||||||
}
|
>
|
||||||
|
<a>
|
||||||
</div>
|
<i className="iconfont icon-shanchu font-15 color-grey-6"></i>
|
||||||
)}
|
</a>
|
||||||
</p>
|
</Popconfirm>
|
||||||
</div>
|
}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</p>
|
||||||
|
</Anchor>
|
||||||
<div>
|
<div>
|
||||||
{detail.image_type ? (
|
{detail.image_type ? (
|
||||||
<div className="edu-txt-center pt20 pb20">
|
<div className="edu-txt-center pt20 pb20">
|
||||||
|
@ -272,7 +310,7 @@ class CoderRootFileDetail extends Component {
|
||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
md && readOnly ?
|
md && readOnly ?
|
||||||
<div className="files-md">
|
<div className="files-md" id="files-md">
|
||||||
<RenderHtml className="file-md imageLayerParent" value={description} url={this.props.history.location}/>
|
<RenderHtml className="file-md imageLayerParent" value={description} url={this.props.history.location}/>
|
||||||
</div>
|
</div>
|
||||||
:
|
:
|
||||||
|
@ -285,6 +323,7 @@ class CoderRootFileDetail extends Component {
|
||||||
readOnly={readOnly}
|
readOnly={readOnly}
|
||||||
editorType="update"
|
editorType="update"
|
||||||
currentBranch={currentBranch}
|
currentBranch={currentBranch}
|
||||||
|
descName={detail && `Update ${detail.name}`}
|
||||||
></Meditor>
|
></Meditor>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
import React , { Component } from 'react';
|
import React , { Component } from 'react';
|
||||||
import { Route , Switch } from 'react-router-dom';
|
import { Route , Switch } from 'react-router-dom';
|
||||||
import Top from './DetailTop';
|
// import Top from './DetailTop';
|
||||||
import Loadable from 'react-loadable';
|
import Loadable from 'react-loadable';
|
||||||
import Loading from '../../Loading';
|
import Loading from '../../Loading';
|
||||||
import axios from 'axios';
|
import './Index.scss';
|
||||||
|
|
||||||
const FileNew = Loadable({
|
const FileNew = Loadable({
|
||||||
loader: () => import('../Newfile/Index'),
|
loader: () => import('../Newfile/Index'),
|
||||||
|
@ -18,25 +18,25 @@ const CoderRootCommit = Loadable({
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
})
|
})
|
||||||
const CoderRootBranch = Loadable({
|
const CoderRootBranch = Loadable({
|
||||||
loader: () => import('./CoderRootBranch'),
|
loader: () => import('./tree/Index'),
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
})
|
})
|
||||||
const CoderRootTag = Loadable({
|
const CoderRootTag = Loadable({
|
||||||
loader: () => import('./CoderRootTag'),
|
loader: () => import('./tag/Index'),
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
})
|
})
|
||||||
const CoderRootVersion = Loadable({
|
const CoderRootVersion = Loadable({
|
||||||
loader: () => import('../Version/version'),
|
loader: () => import('./version/Index'),
|
||||||
loading: Loading,
|
|
||||||
})
|
|
||||||
const CoderRootVersionNew = Loadable({
|
|
||||||
loader: () => import('../Version/New'),
|
|
||||||
loading: Loading,
|
|
||||||
})
|
|
||||||
const CoderRootVersionUpdate = Loadable({
|
|
||||||
loader: () => import('../Version/New'),
|
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
})
|
})
|
||||||
|
// const CoderRootVersionNew = Loadable({
|
||||||
|
// loader: () => import('./version/New'),
|
||||||
|
// loading: Loading,
|
||||||
|
// })
|
||||||
|
// const CoderRootVersionUpdate = Loadable({
|
||||||
|
// loader: () => import('./version/New'),
|
||||||
|
// loading: Loading,
|
||||||
|
// })
|
||||||
const Diff = Loadable({
|
const Diff = Loadable({
|
||||||
loader: () => import('./Diff'),
|
loader: () => import('./Diff'),
|
||||||
loading: Loading,
|
loading: Loading,
|
||||||
|
@ -50,89 +50,94 @@ class CoderRootIndex extends Component{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount=()=>{
|
// componentDidMount=()=>{
|
||||||
this.Init();
|
// this.Init();
|
||||||
}
|
// }
|
||||||
componentDidUpdate=(prevProps)=>{
|
// componentDidUpdate=(prevProps)=>{
|
||||||
const { location } = this.props;
|
// const { location } = this.props;
|
||||||
const prevlocation = prevProps && prevProps.location;
|
// const prevlocation = prevProps && prevProps.location;
|
||||||
if (location !== prevlocation) {
|
// if (location !== prevlocation) {
|
||||||
this.Init();
|
// this.Init();
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
Init=()=>{
|
// Init=()=>{
|
||||||
const { branchName } = this.props.match.params;
|
// const { branchName } = this.props.match.params;
|
||||||
const { defaultBranch } = this.props;
|
// const { defaultBranch } = this.props;
|
||||||
this.getTopCount(branchName || defaultBranch);
|
// this.getTopCount(branchName || defaultBranch);
|
||||||
}
|
// }
|
||||||
|
|
||||||
// 获取<Top />组件里要显示的数据
|
// 获取<Top />组件里要显示的数据
|
||||||
getTopCount=(branch)=>{
|
// getTopCount=(branch)=>{
|
||||||
const { projectsId , owner } = this.props.match.params;
|
// const { projectsId , owner } = this.props.match.params;
|
||||||
const url = `/${owner}/${projectsId}/top_counts.json`;
|
// const url = `/${owner}/${projectsId}/top_counts.json`;
|
||||||
axios.get(url,{params:{
|
// axios.get(url,{params:{
|
||||||
ref:branch
|
// ref:branch
|
||||||
}}).then(result=>{
|
// }}).then(result=>{
|
||||||
if(result){
|
// if(result){
|
||||||
this.setState({
|
// this.setState({
|
||||||
coderCount:result.data
|
// coderCount:result.data
|
||||||
})
|
// })
|
||||||
}
|
// }
|
||||||
}).catch(error=>{console.log(error);})
|
// }).catch(error=>{console.log(error);})
|
||||||
}
|
// }
|
||||||
render(){
|
render(){
|
||||||
return(
|
return(
|
||||||
<div>
|
<div className="coderSubPage">
|
||||||
<Top {...this.props} {...this.state}/>
|
{/* <Top {...this.props} {...this.state}/> */}
|
||||||
<Switch {...this.props}>
|
<Switch {...this.props}>
|
||||||
{/* 新建文件 */}
|
{/* 新建文件 */}
|
||||||
<Route path="/projects/:owner/:projectsId/:branch/newfile/:path"
|
<Route path="/:owner/:projectsId/:branch/newfile/:path"
|
||||||
render={
|
render={
|
||||||
(props) => (<FileNew {...this.props} {...props} {...this.state} />)
|
(props) => (<FileNew {...this.props} {...props} {...this.state} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/:branch/uploadfile"
|
<Route path="/:owner/:projectsId/:branch/uploadfile"
|
||||||
render={
|
render={
|
||||||
(props) => (<UploadFile {...this.props} {...props} {...this.state} />)
|
(props) => (<UploadFile {...this.props} {...props} {...this.state} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/:branch/newfile"
|
<Route path="/:owner/:projectsId/:branch/newfile"
|
||||||
render={
|
render={
|
||||||
(props) => (<FileNew {...this.props} {...props} {...this.state} getTopCount={this.getTopCount} />)
|
(props) => (<FileNew {...this.props} {...props} {...this.state} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/commits/:sha"
|
<Route path="/:owner/:projectsId/commits/branch/:branchName"
|
||||||
|
render={
|
||||||
|
() => (<CoderRootCommit {...this.props} {...this.state} commit_class="main" />)
|
||||||
|
}
|
||||||
|
></Route>
|
||||||
|
<Route path="/:owner/:projectsId/commits/:sha"
|
||||||
render={
|
render={
|
||||||
(props) => (<Diff {...this.props} {...props} {...this.state}/>)
|
(props) => (<Diff {...this.props} {...props} {...this.state}/>)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/commits"
|
<Route path="/:owner/:projectsId/commits"
|
||||||
render={
|
render={
|
||||||
() => (<CoderRootCommit {...this.props} {...this.state} commit_class="main" getTopCount={this.getTopCount} />)
|
() => (<CoderRootCommit {...this.props} {...this.state} commit_class="main" />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/releases/:versionId/update"
|
{/* <Route path="/:owner/:projectsId/releases/:versionId/update"
|
||||||
render={
|
render={
|
||||||
(props) => (<CoderRootVersionUpdate {...this.props} {...this.state} {...props} />)
|
(props) => (<CoderRootVersionUpdate {...this.props} {...this.state} {...props} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/releases/new"
|
<Route path="/:owner/:projectsId/releases/new"
|
||||||
render={
|
render={
|
||||||
() => (<CoderRootVersionNew {...this.props} {...this.state} />)
|
() => (<CoderRootVersionNew {...this.props} {...this.state} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route> */}
|
||||||
<Route path="/projects/:owner/:projectsId/releases"
|
<Route path="/:owner/:projectsId/releases"
|
||||||
render={
|
render={
|
||||||
() => (<CoderRootVersion {...this.props} {...this.state} />)
|
() => (<CoderRootVersion {...this.props} {...this.state} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/tag"
|
<Route path="/:owner/:projectsId/tags"
|
||||||
render={
|
render={
|
||||||
() => (<CoderRootTag {...this.props} {...this.state} />)
|
() => (<CoderRootTag {...this.props} {...this.state} />)
|
||||||
}
|
}
|
||||||
></Route>
|
></Route>
|
||||||
<Route path="/projects/:owner/:projectsId/branchs"
|
<Route path="/:owner/:projectsId/branches"
|
||||||
render={
|
render={
|
||||||
() => (<CoderRootBranch {...this.props} {...this.state} />)
|
() => (<CoderRootBranch {...this.props} {...this.state} />)
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ export default (( props, { projectDetail }) => {
|
||||||
}, [owner, projectsId]);
|
}, [owner, projectsId]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="main">
|
<div className="main" style={{padding:"0px",border:"none"}}>
|
||||||
<Spin spinning={isSpin}>
|
<Spin spinning={isSpin}>
|
||||||
<div style={{minHeight:"400px"}}>
|
<div style={{minHeight:"400px"}}>
|
||||||
{
|
{
|
||||||
|
@ -34,7 +34,7 @@ export default (( props, { projectDetail }) => {
|
||||||
<div className="div_table">
|
<div className="div_table">
|
||||||
<ul className="ul_thead">
|
<ul className="ul_thead">
|
||||||
<li>
|
<li>
|
||||||
<span className="flex1">标签名</span>
|
<span className="flex1">标记名</span>
|
||||||
<span>提交信息</span>
|
<span>提交信息</span>
|
||||||
<span className="ul_tbody_forth">下载</span>
|
<span className="ul_tbody_forth">下载</span>
|
||||||
</li>
|
</li>
|
||||||
|
@ -49,7 +49,7 @@ export default (( props, { projectDetail }) => {
|
||||||
<span className="font-16">{item.name}</span>
|
<span className="font-16">{item.name}</span>
|
||||||
</span>
|
</span>
|
||||||
<span className="ul_tbody_third">
|
<span className="ul_tbody_third">
|
||||||
<Link to={`/projects/${owner}/${projectsId}/commits/${truncateCommitId(`${item.id}`)}`} className="commitKey" style={{ "marginLeft": 0 }}>{truncateCommitId(`${item.id}`)}</Link>
|
<Link to={`/${owner}/${projectsId}/commits/${truncateCommitId(`${item.id}`)}`} className="commitKey" style={{ "marginLeft": 0 }}>{truncateCommitId(`${item.id}`)}</Link>
|
||||||
</span>
|
</span>
|
||||||
<span className="ul_tbody_forth">
|
<span className="ul_tbody_forth">
|
||||||
<a href={item.tarball_url} style={{ color: "#4CC1DA" }} className="mr30"><i className="iconfont icon-TAR font-18 mr5"></i>TAR</a>
|
<a href={item.tarball_url} style={{ color: "#4CC1DA" }} className="mr30"><i className="iconfont icon-TAR font-18 mr5"></i>TAR</a>
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,28 @@
|
||||||
|
import React from 'react';
|
||||||
|
import { Route, Switch } from "react-router-dom";
|
||||||
|
import { withRouter } from "react-router";
|
||||||
|
import Loadable from "react-loadable";
|
||||||
|
import Loading from "../../Loading";
|
||||||
|
import { SnackbarHOC } from "educoder";
|
||||||
|
import { CNotificationHOC } from "../../modules/courses/common/CNotificationHOC";
|
||||||
|
import { TPMIndexHOC } from "../../modules/tpm/TPMIndexHOC";
|
||||||
|
|
||||||
|
// forge项目详情
|
||||||
|
const ProjectDetail = Loadable({
|
||||||
|
loader: () => import("../Main/Detail"),
|
||||||
|
loading: Loading,
|
||||||
|
});
|
||||||
|
export default withRouter(
|
||||||
|
(CNotificationHOC()(SnackbarHOC()(TPMIndexHOC((props) => {
|
||||||
|
return (
|
||||||
|
<Switch>
|
||||||
|
<Route
|
||||||
|
path="/:owner/:projectsId"
|
||||||
|
render={(p) => (
|
||||||
|
<ProjectDetail {...props} {...p} />
|
||||||
|
)}
|
||||||
|
></Route>
|
||||||
|
</Switch>
|
||||||
|
)
|
||||||
|
}))))
|
||||||
|
)
|
|
@ -12,19 +12,19 @@ class DetailTop extends Component {
|
||||||
{
|
{
|
||||||
platform ?
|
platform ?
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/commits`} className={pathname.indexOf("/commits") > 0 ? "active" : ""}>
|
<Link to={`/${owner}/${projectsId}/commits`} className={pathname.indexOf("/commits") > 0 ? "active" : ""}>
|
||||||
<i className="iconfont icon-tijiaojilu font-20 mr3 font-bd"></i>
|
<i className="iconfont icon-tijiaojilu font-20 mr3 font-bd"></i>
|
||||||
<span>{(coderCount && coderCount.commits_count) || 0}</span>个提交
|
<span>{(coderCount && coderCount.commits_count) || 0}</span>个提交
|
||||||
</Link>
|
</Link>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/branchs`} className={pathname.indexOf("/branchs") > 0 ? "active" : ""}>
|
<Link to={`/${owner}/${projectsId}/branches`} className={pathname.indexOf("/branches") > 0 ? "active" : ""}>
|
||||||
<i className="iconfont icon-fenzhi1 font-18 mr3"></i>
|
<i className="iconfont icon-fenzhi1 font-18 mr3"></i>
|
||||||
<span>{(coderCount && coderCount.branches_count) || 0}</span>个分支
|
<span>{(coderCount && coderCount.branches_count) || 0}</span>个分支
|
||||||
</Link>
|
</Link>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/tag`} className={pathname.indexOf("/tag") > 0 ? "active" : ""}>
|
<Link to={`/${owner}/${projectsId}/tags`} className={pathname.indexOf("/tags") > 0 ? "active" : ""}>
|
||||||
<i className="iconfont icon-biaoqian3 font-18 mr3"></i>
|
<i className="iconfont icon-biaoqian3 font-18 mr3"></i>
|
||||||
<span>{(coderCount && coderCount.tags_count) || 0}</span>个标签
|
<span>{(coderCount && coderCount.tags_count) || 0}</span>个标签
|
||||||
</Link>
|
</Link>
|
||||||
<Link to={`/projects/${owner}/${projectsId}/releases`} className={pathname.indexOf("/releases") > 0 ? "active" : ""}>
|
<Link to={`/${owner}/${projectsId}/releases`} className={pathname.indexOf("/releases") > 0 ? "active" : ""}>
|
||||||
<i className="iconfont icon-fahangban font-18 mr3"></i>
|
<i className="iconfont icon-fahangban font-18 mr3"></i>
|
||||||
<span>{(coderCount && coderCount.version_releasesed_count) || 0}</span>个发行版
|
<span>{(coderCount && coderCount.version_releasesed_count) || 0}</span>个发行版
|
||||||
</Link>
|
</Link>
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue