提交 7957220e9db2b447e8efde898a5255ed2549111c

作者 愚道
1 个父辈 70ebdf1a

feat: refactor monitor block of ant-design-pro

1 export default { 1 export default {
2 plugins: [ 2 plugins: [
3 - ['umi-plugin-block-dev', {}], 3 + ['umi-plugin-block-dev', {
  4 + layout: 'ant-design-pro',
  5 + }],
4 ['umi-plugin-react', { 6 ['umi-plugin-react', {
5 dva: true, 7 dva: true,
6 locale: true, 8 locale: true,
1 -import React, { PureComponent } from 'react';  
2 -import { connect } from 'dva';  
3 -import styles from './GridContent.less';  
4 -  
5 -class GridContent extends PureComponent {  
6 - render() {  
7 - const { contentWidth, children } = this.props;  
8 - let className = `${styles.main}`;  
9 - if (contentWidth === 'Fixed') {  
10 - className = `${styles.main} ${styles.wide}`;  
11 - }  
12 - return <div className={className}>{children}</div>;  
13 - }  
14 -}  
15 -  
16 -export default connect(({ setting }) => ({  
17 - contentWidth: setting.contentWidth,  
18 -}))(GridContent);  
1 -.main {  
2 - width: 100%;  
3 - height: 100%;  
4 - min-height: 100%;  
5 - transition: 0.3s;  
6 - &.wide {  
7 - max-width: 1200px;  
8 - margin: 0 auto;  
9 - }  
10 -}  
1 -import { stringify } from 'qs';  
2 -import request from '@/utils/request';  
3 -  
4 -export async function queryProjectNotice() {  
5 - return request('/api/project/notice');  
6 -}  
7 -  
8 -export async function queryActivities() {  
9 - return request('/api/activities');  
10 -}  
11 -  
12 -export async function queryRule(params) {  
13 - return request(`/api/rule?${stringify(params)}`);  
14 -}  
15 -  
16 -export async function removeRule(params) {  
17 - return request('/api/rule', {  
18 - method: 'POST',  
19 - body: {  
20 - ...params,  
21 - method: 'delete',  
22 - },  
23 - });  
24 -}  
25 -  
26 -export async function addRule(params) {  
27 - return request('/api/rule', {  
28 - method: 'POST',  
29 - body: {  
30 - ...params,  
31 - method: 'post',  
32 - },  
33 - });  
34 -}  
35 -  
36 -export async function updateRule(params) {  
37 - return request('/api/rule', {  
38 - method: 'POST',  
39 - body: {  
40 - ...params,  
41 - method: 'update',  
42 - },  
43 - });  
44 -}  
45 -  
46 -export async function fakeSubmitForm(params) {  
47 - return request('/api/forms', {  
48 - method: 'POST',  
49 - body: params,  
50 - });  
51 -}  
52 -  
53 -export async function fakeChartData() {  
54 - return request('/api/fake_chart_data');  
55 -}  
56 -  
57 -export async function queryTags() {  
58 - return request('/api/tags');  
59 -}  
60 -  
61 -export async function queryBasicProfile() {  
62 - return request('/api/profile/basic');  
63 -}  
64 -  
65 -export async function queryAdvancedProfile() {  
66 - return request('/api/profile/advanced');  
67 -}  
68 -  
69 -export async function queryFakeList(params) {  
70 - return request(`/api/fake_list?${stringify(params)}`);  
71 -}  
72 -  
73 -export async function removeFakeList(params) {  
74 - const { count = 5, ...restParams } = params;  
75 - return request(`/api/fake_list?count=${count}`, {  
76 - method: 'POST',  
77 - body: {  
78 - ...restParams,  
79 - method: 'delete',  
80 - },  
81 - });  
82 -}  
83 -  
84 -export async function addFakeList(params) {  
85 - const { count = 5, ...restParams } = params;  
86 - return request(`/api/fake_list?count=${count}`, {  
87 - method: 'POST',  
88 - body: {  
89 - ...restParams,  
90 - method: 'post',  
91 - },  
92 - });  
93 -}  
94 -  
95 -export async function updateFakeList(params) {  
96 - const { count = 5, ...restParams } = params;  
97 - return request(`/api/fake_list?count=${count}`, {  
98 - method: 'POST',  
99 - body: {  
100 - ...restParams,  
101 - method: 'update',  
102 - },  
103 - });  
104 -}  
105 -  
106 -export async function fakeAccountLogin(params) {  
107 - return request('/api/login/account', {  
108 - method: 'POST',  
109 - body: params,  
110 - });  
111 -}  
112 -  
113 -export async function fakeRegister(params) {  
114 - return request('/api/register', {  
115 - method: 'POST',  
116 - body: params,  
117 - });  
118 -}  
119 -  
120 -export async function queryNotices() {  
121 - return request('/api/notices');  
122 -}  
123 -  
124 -export async function getFakeCaptcha(mobile) {  
125 - return request(`/api/captcha?mobile=${mobile}`);  
126 -}  
1 -import RenderAuthorized from 'ant-design-pro/lib/Authorized';  
2 -import { getAuthority } from './authority';  
3 -  
4 -let Authorized = RenderAuthorized(getAuthority()); // eslint-disable-line  
5 -  
6 -// Reload the rights component  
7 -const reloadAuthorized = () => {  
8 - Authorized = RenderAuthorized(getAuthority());  
9 -};  
10 -  
11 -export { reloadAuthorized };  
12 -export default Authorized;  
1 -// use localStorage to store the authority info, which might be sent from server in actual project.  
2 -export function getAuthority(str) {  
3 - // return localStorage.getItem('antd-pro-authority') || ['admin', 'user'];  
4 - const authorityString =  
5 - typeof str === 'undefined' ? localStorage.getItem('antd-pro-authority') : str;  
6 - // authorityString could be admin, "admin", ["admin"]  
7 - let authority;  
8 - try {  
9 - authority = JSON.parse(authorityString);  
10 - } catch (e) {  
11 - authority = authorityString;  
12 - }  
13 - if (typeof authority === 'string') {  
14 - return [authority];  
15 - }  
16 - return authority || ['admin'];  
17 -}  
18 -  
19 -export function setAuthority(authority) {  
20 - const proAuthority = typeof authority === 'string' ? [authority] : authority;  
21 - return localStorage.setItem('antd-pro-authority', JSON.stringify(proAuthority));  
22 -}  
1 -import fetch from 'dva/fetch';  
2 -import { notification } from 'antd';  
3 -import router from 'umi/router';  
4 -import hash from 'hash.js';  
5 -import { isAntdPro } from './utils';  
6 -  
7 -const codeMessage = {  
8 - 200: '服务器成功返回请求的数据。',  
9 - 201: '新建或修改数据成功。',  
10 - 202: '一个请求已经进入后台排队(异步任务)。',  
11 - 204: '删除数据成功。',  
12 - 400: '发出的请求有错误,服务器没有进行新建或修改数据的操作。',  
13 - 401: '用户没有权限(令牌、用户名、密码错误)。',  
14 - 403: '用户得到授权,但是访问是被禁止的。',  
15 - 404: '发出的请求针对的是不存在的记录,服务器没有进行操作。',  
16 - 406: '请求的格式不可得。',  
17 - 410: '请求的资源被永久删除,且不会再得到的。',  
18 - 422: '当创建一个对象时,发生一个验证错误。',  
19 - 500: '服务器发生错误,请检查服务器。',  
20 - 502: '网关错误。',  
21 - 503: '服务不可用,服务器暂时过载或维护。',  
22 - 504: '网关超时。',  
23 -};  
24 -  
25 -const checkStatus = response => {  
26 - if (response.status >= 200 && response.status < 300) {  
27 - return response;  
28 - }  
29 - const errortext = codeMessage[response.status] || response.statusText;  
30 - notification.error({  
31 - message: `请求错误 ${response.status}: ${response.url}`,  
32 - description: errortext,  
33 - });  
34 - const error = new Error(errortext);  
35 - error.name = response.status;  
36 - error.response = response;  
37 - throw error;  
38 -};  
39 -  
40 -const cachedSave = (response, hashcode) => {  
41 - /**  
42 - * Clone a response data and store it in sessionStorage  
43 - * Does not support data other than json, Cache only json  
44 - */  
45 - const contentType = response.headers.get('Content-Type');  
46 - if (contentType && contentType.match(/application\/json/i)) {  
47 - // All data is saved as text  
48 - response  
49 - .clone()  
50 - .text()  
51 - .then(content => {  
52 - sessionStorage.setItem(hashcode, content);  
53 - sessionStorage.setItem(`${hashcode}:timestamp`, Date.now());  
54 - });  
55 - }  
56 - return response;  
57 -};  
58 -  
59 -/**  
60 - * Requests a URL, returning a promise.  
61 - *  
62 - * @param {string} url The URL we want to request  
63 - * @param {object} [option] The options we want to pass to "fetch"  
64 - * @return {object} An object containing either "data" or "err"  
65 - */  
66 -export default function request(url, option) {  
67 - const options = {  
68 - expirys: isAntdPro(),  
69 - ...option,  
70 - };  
71 - /**  
72 - * Produce fingerprints based on url and parameters  
73 - * Maybe url has the same parameters  
74 - */  
75 - const fingerprint = url + (options.body ? JSON.stringify(options.body) : '');  
76 - const hashcode = hash  
77 - .sha256()  
78 - .update(fingerprint)  
79 - .digest('hex');  
80 -  
81 - const defaultOptions = {  
82 - credentials: 'include',  
83 - };  
84 - const newOptions = { ...defaultOptions, ...options };  
85 - if (  
86 - newOptions.method === 'POST' ||  
87 - newOptions.method === 'PUT' ||  
88 - newOptions.method === 'DELETE'  
89 - ) {  
90 - if (!(newOptions.body instanceof FormData)) {  
91 - newOptions.headers = {  
92 - Accept: 'application/json',  
93 - 'Content-Type': 'application/json; charset=utf-8',  
94 - ...newOptions.headers,  
95 - };  
96 - newOptions.body = JSON.stringify(newOptions.body);  
97 - } else {  
98 - // newOptions.body is FormData  
99 - newOptions.headers = {  
100 - Accept: 'application/json',  
101 - ...newOptions.headers,  
102 - };  
103 - }  
104 - }  
105 -  
106 - const expirys = options.expirys && 60;  
107 - // options.expirys !== false, return the cache,  
108 - if (options.expirys !== false) {  
109 - const cached = sessionStorage.getItem(hashcode);  
110 - const whenCached = sessionStorage.getItem(`${hashcode}:timestamp`);  
111 - if (cached !== null && whenCached !== null) {  
112 - const age = (Date.now() - whenCached) / 1000;  
113 - if (age < expirys) {  
114 - const response = new Response(new Blob([cached]));  
115 - return response.json();  
116 - }  
117 - sessionStorage.removeItem(hashcode);  
118 - sessionStorage.removeItem(`${hashcode}:timestamp`);  
119 - }  
120 - }  
121 - return fetch(url, newOptions)  
122 - .then(checkStatus)  
123 - .then(response => cachedSave(response, hashcode))  
124 - .then(response => {  
125 - // DELETE and 204 do not return data by default  
126 - // using .json will report an error.  
127 - if (newOptions.method === 'DELETE' || response.status === 204) {  
128 - return response.text();  
129 - }  
130 - return response.json();  
131 - })  
132 - .catch(e => {  
133 - const status = e.name;  
134 - if (status === 401) {  
135 - // @HACK  
136 - /* eslint-disable no-underscore-dangle */  
137 - window.g_app._store.dispatch({  
138 - type: 'login/logout',  
139 - });  
140 - return;  
141 - }  
142 - // environment should not be used  
143 - if (status === 403) {  
144 - router.push('/exception/403');  
145 - return;  
146 - }  
147 - if (status <= 504 && status >= 500) {  
148 - router.push('/exception/500');  
149 - return;  
150 - }  
151 - if (status >= 404 && status < 422) {  
152 - router.push('/exception/404');  
153 - }  
154 - });  
155 -}  
1 -import moment from 'moment';  
2 -import React from 'react';  
3 -import nzh from 'nzh/cn';  
4 -import { parse, stringify } from 'qs';  
5 -  
6 -export function fixedZero(val) {  
7 - return val * 1 < 10 ? `0${val}` : val;  
8 -}  
9 -  
10 -export function getTimeDistance(type) {  
11 - const now = new Date();  
12 - const oneDay = 1000 * 60 * 60 * 24;  
13 -  
14 - if (type === 'today') {  
15 - now.setHours(0);  
16 - now.setMinutes(0);  
17 - now.setSeconds(0);  
18 - return [moment(now), moment(now.getTime() + (oneDay - 1000))];  
19 - }  
20 -  
21 - if (type === 'week') {  
22 - let day = now.getDay();  
23 - now.setHours(0);  
24 - now.setMinutes(0);  
25 - now.setSeconds(0);  
26 -  
27 - if (day === 0) {  
28 - day = 6;  
29 - } else {  
30 - day -= 1;  
31 - }  
32 -  
33 - const beginTime = now.getTime() - day * oneDay;  
34 -  
35 - return [moment(beginTime), moment(beginTime + (7 * oneDay - 1000))];  
36 - }  
37 -  
38 - if (type === 'month') {  
39 - const year = now.getFullYear();  
40 - const month = now.getMonth();  
41 - const nextDate = moment(now).add(1, 'months');  
42 - const nextYear = nextDate.year();  
43 - const nextMonth = nextDate.month();  
44 -  
45 - return [  
46 - moment(`${year}-${fixedZero(month + 1)}-01 00:00:00`),  
47 - moment(moment(`${nextYear}-${fixedZero(nextMonth + 1)}-01 00:00:00`).valueOf() - 1000),  
48 - ];  
49 - }  
50 -  
51 - const year = now.getFullYear();  
52 - return [moment(`${year}-01-01 00:00:00`), moment(`${year}-12-31 23:59:59`)];  
53 -}  
54 -  
55 -export function getPlainNode(nodeList, parentPath = '') {  
56 - const arr = [];  
57 - nodeList.forEach(node => {  
58 - const item = node;  
59 - item.path = `${parentPath}/${item.path || ''}`.replace(/\/+/g, '/');  
60 - item.exact = true;  
61 - if (item.children && !item.component) {  
62 - arr.push(...getPlainNode(item.children, item.path));  
63 - } else {  
64 - if (item.children && item.component) {  
65 - item.exact = false;  
66 - }  
67 - arr.push(item);  
68 - }  
69 - });  
70 - return arr;  
71 -}  
72 -  
73 -export function digitUppercase(n) {  
74 - return nzh.toMoney(n);  
75 -}  
76 -  
77 -function getRelation(str1, str2) {  
78 - if (str1 === str2) {  
79 - console.warn('Two path are equal!'); // eslint-disable-line  
80 - }  
81 - const arr1 = str1.split('/');  
82 - const arr2 = str2.split('/');  
83 - if (arr2.every((item, index) => item === arr1[index])) {  
84 - return 1;  
85 - }  
86 - if (arr1.every((item, index) => item === arr2[index])) {  
87 - return 2;  
88 - }  
89 - return 3;  
90 -}  
91 -  
92 -function getRenderArr(routes) {  
93 - let renderArr = [];  
94 - renderArr.push(routes[0]);  
95 - for (let i = 1; i < routes.length; i += 1) {  
96 - // 去重  
97 - renderArr = renderArr.filter(item => getRelation(item, routes[i]) !== 1);  
98 - // 是否包含  
99 - const isAdd = renderArr.every(item => getRelation(item, routes[i]) === 3);  
100 - if (isAdd) {  
101 - renderArr.push(routes[i]);  
102 - }  
103 - }  
104 - return renderArr;  
105 -}  
106 -  
107 -/**  
108 - * Get router routing configuration  
109 - * { path:{name,...param}}=>Array<{name,path ...param}>  
110 - * @param {string} path  
111 - * @param {routerData} routerData  
112 - */  
113 -export function getRoutes(path, routerData) {  
114 - let routes = Object.keys(routerData).filter(  
115 - routePath => routePath.indexOf(path) === 0 && routePath !== path  
116 - );  
117 - // Replace path to '' eg. path='user' /user/name => name  
118 - routes = routes.map(item => item.replace(path, ''));  
119 - // Get the route to be rendered to remove the deep rendering  
120 - const renderArr = getRenderArr(routes);  
121 - // Conversion and stitching parameters  
122 - const renderRoutes = renderArr.map(item => {  
123 - const exact = !routes.some(route => route !== item && getRelation(route, item) === 1);  
124 - return {  
125 - exact,  
126 - ...routerData[`${path}${item}`],  
127 - key: `${path}${item}`,  
128 - path: `${path}${item}`,  
129 - };  
130 - });  
131 - return renderRoutes;  
132 -}  
133 -  
134 -export function getPageQuery() {  
135 - return parse(window.location.href.split('?')[1]);  
136 -}  
137 -  
138 -export function getQueryPath(path = '', query = {}) {  
139 - const search = stringify(query);  
140 - if (search.length) {  
141 - return `${path}?${search}`;  
142 - }  
143 - return path;  
144 -}  
145 -  
146 -/* eslint no-useless-escape:0 */  
147 -const reg = /(((^https?:(?:\/\/)?)(?:[-;:&=\+\$,\w]+@)?[A-Za-z0-9.-]+(?::\d+)?|(?:www.|[-;:&=\+\$,\w]+@)[A-Za-z0-9.-]+)((?:\/[\+~%\/.\w-_]*)?\??(?:[-\+=&;%@.\w_]*)#?(?:[\w]*))?)$/;  
148 -  
149 -export function isUrl(path) {  
150 - return reg.test(path);  
151 -}  
152 -  
153 -export function formatWan(val) {  
154 - const v = val * 1;  
155 - if (!v || Number.isNaN(v)) return '';  
156 -  
157 - let result = val;  
158 - if (val > 10000) {  
159 - result = Math.floor(val / 10000);  
160 - result = (  
161 - <span>  
162 - {result}  
163 - <span  
164 - style={{  
165 - position: 'relative',  
166 - top: -2,  
167 - fontSize: 14,  
168 - fontStyle: 'normal',  
169 - marginLeft: 2,  
170 - }}  
171 - >  
172 -  
173 - </span>  
174 - </span>  
175 - );  
176 - }  
177 - return result;  
178 -}  
179 -  
180 -// 给官方演示站点用,用于关闭真实开发环境不需要使用的特性  
181 -export function isAntdPro() {  
182 - return window.location.hostname === 'preview.pro.ant.design';  
183 -}  
1 -.textOverflow() {  
2 - overflow: hidden;  
3 - text-overflow: ellipsis;  
4 - word-break: break-all;  
5 - white-space: nowrap;  
6 -}  
7 -  
8 -.textOverflowMulti(@line: 3, @bg: #fff) {  
9 - overflow: hidden;  
10 - position: relative;  
11 - line-height: 1.5em;  
12 - max-height: @line * 1.5em;  
13 - text-align: justify;  
14 - margin-right: -1em;  
15 - padding-right: 1em;  
16 - &:before {  
17 - background: @bg;  
18 - content: '...';  
19 - padding: 0 1px;  
20 - position: absolute;  
21 - right: 14px;  
22 - bottom: 0;  
23 - }  
24 - &:after {  
25 - background: white;  
26 - content: '';  
27 - margin-top: 0.2em;  
28 - position: absolute;  
29 - right: 14px;  
30 - width: 1em;  
31 - height: 1em;  
32 - }  
33 -}  
34 -  
35 -// mixins for clearfix  
36 -// ------------------------  
37 -.clearfix() {  
38 - zoom: 1;  
39 - &:before,  
40 - &:after {  
41 - content: ' ';  
42 - display: table;  
43 - }  
44 - &:after {  
45 - clear: both;  
46 - visibility: hidden;  
47 - font-size: 0;  
48 - height: 0;  
49 - }  
50 -}  
@@ -11,15 +11,17 @@ @@ -11,15 +11,17 @@
11 "url": "https://github.com/umijs/umi-blocks/monitor" 11 "url": "https://github.com/umijs/umi-blocks/monitor"
12 }, 12 },
13 "dependencies": { 13 "dependencies": {
14 - "react": "^16.6.3",  
15 - "dva": "^2.4.0",  
16 - "antd": "^3.10.9",  
17 "ant-design-pro": "^2.1.1", 14 "ant-design-pro": "^2.1.1",
18 - "numeral": "^2.0.6",  
19 - "qs": "^6.6.0", 15 + "antd": "^3.10.9",
  16 + "dva": "^2.4.0",
20 "hash.js": "^1.1.5", 17 "hash.js": "^1.1.5",
  18 + "mockjs": "^1.0.1-beta3",
21 "moment": "^2.22.2", 19 "moment": "^2.22.2",
22 - "nzh": "^1.0.3" 20 + "numeral": "^2.0.6",
  21 + "nzh": "^1.0.3",
  22 + "qs": "^6.6.0",
  23 + "react": "^16.6.3",
  24 + "umi-request": "^1.0.0"
23 }, 25 },
24 "devDependencies": { 26 "devDependencies": {
25 "umi": "^2.3.0-beta.1", 27 "umi": "^2.3.0-beta.1",
1 -import moment from 'moment';  
2 -  
3 -// mock data  
4 -const visitData = [];  
5 -const beginDay = new Date().getTime();  
6 -  
7 -const fakeY = [7, 5, 4, 2, 4, 7, 5, 6, 5, 9, 6, 3, 1, 5, 3, 6, 5];  
8 -for (let i = 0; i < fakeY.length; i += 1) {  
9 - visitData.push({  
10 - x: moment(new Date(beginDay + 1000 * 60 * 60 * 24 * i)).format('YYYY-MM-DD'),  
11 - y: fakeY[i],  
12 - });  
13 -}  
14 -  
15 -const visitData2 = [];  
16 -const fakeY2 = [1, 6, 4, 8, 3, 7, 2];  
17 -for (let i = 0; i < fakeY2.length; i += 1) {  
18 - visitData2.push({  
19 - x: moment(new Date(beginDay + 1000 * 60 * 60 * 24 * i)).format('YYYY-MM-DD'),  
20 - y: fakeY2[i],  
21 - });  
22 -}  
23 -  
24 -const salesData = [];  
25 -for (let i = 0; i < 12; i += 1) {  
26 - salesData.push({  
27 - x: `${i + 1}月`,  
28 - y: Math.floor(Math.random() * 1000) + 200,  
29 - });  
30 -}  
31 -const searchData = [];  
32 -for (let i = 0; i < 50; i += 1) {  
33 - searchData.push({  
34 - index: i + 1,  
35 - keyword: `搜索关键词-${i}`,  
36 - count: Math.floor(Math.random() * 1000),  
37 - range: Math.floor(Math.random() * 100),  
38 - status: Math.floor((Math.random() * 10) % 2),  
39 - });  
40 -}  
41 -const salesTypeData = [  
42 - {  
43 - x: '家用电器',  
44 - y: 4544,  
45 - },  
46 - {  
47 - x: '食用酒水',  
48 - y: 3321,  
49 - },  
50 - {  
51 - x: '个护健康',  
52 - y: 3113,  
53 - },  
54 - {  
55 - x: '服饰箱包',  
56 - y: 2341,  
57 - },  
58 - {  
59 - x: '母婴产品',  
60 - y: 1231,  
61 - },  
62 - {  
63 - x: '其他',  
64 - y: 1231,  
65 - },  
66 -];  
67 -  
68 -const salesTypeDataOnline = [  
69 - {  
70 - x: '家用电器',  
71 - y: 244,  
72 - },  
73 - {  
74 - x: '食用酒水',  
75 - y: 321,  
76 - },  
77 - {  
78 - x: '个护健康',  
79 - y: 311,  
80 - },  
81 - {  
82 - x: '服饰箱包',  
83 - y: 41,  
84 - },  
85 - {  
86 - x: '母婴产品',  
87 - y: 121,  
88 - },  
89 - {  
90 - x: '其他',  
91 - y: 111,  
92 - },  
93 -];  
94 -  
95 -const salesTypeDataOffline = [  
96 - {  
97 - x: '家用电器',  
98 - y: 99,  
99 - },  
100 - {  
101 - x: '食用酒水',  
102 - y: 188,  
103 - },  
104 - {  
105 - x: '个护健康',  
106 - y: 344,  
107 - },  
108 - {  
109 - x: '服饰箱包',  
110 - y: 255,  
111 - },  
112 - {  
113 - x: '其他',  
114 - y: 65,  
115 - },  
116 -];  
117 -  
118 -const offlineData = [];  
119 -for (let i = 0; i < 10; i += 1) {  
120 - offlineData.push({  
121 - name: `Stores ${i}`,  
122 - cvr: Math.ceil(Math.random() * 9) / 10,  
123 - });  
124 -}  
125 -const offlineChartData = [];  
126 -for (let i = 0; i < 20; i += 1) {  
127 - offlineChartData.push({  
128 - x: new Date().getTime() + 1000 * 60 * 30 * i,  
129 - y1: Math.floor(Math.random() * 100) + 10,  
130 - y2: Math.floor(Math.random() * 100) + 10,  
131 - });  
132 -}  
133 -  
134 -const radarOriginData = [  
135 - {  
136 - name: '个人',  
137 - ref: 10,  
138 - koubei: 8,  
139 - output: 4,  
140 - contribute: 5,  
141 - hot: 7,  
142 - },  
143 - {  
144 - name: '团队',  
145 - ref: 3,  
146 - koubei: 9,  
147 - output: 6,  
148 - contribute: 3,  
149 - hot: 1,  
150 - },  
151 - {  
152 - name: '部门',  
153 - ref: 4,  
154 - koubei: 1,  
155 - output: 6,  
156 - contribute: 5,  
157 - hot: 7,  
158 - },  
159 -];  
160 -  
161 -const radarData = [];  
162 -const radarTitleMap = {  
163 - ref: '引用',  
164 - koubei: '口碑',  
165 - output: '产量',  
166 - contribute: '贡献',  
167 - hot: '热度',  
168 -};  
169 -radarOriginData.forEach(item => {  
170 - Object.keys(item).forEach(key => {  
171 - if (key !== 'name') {  
172 - radarData.push({  
173 - name: item.name,  
174 - label: radarTitleMap[key],  
175 - value: item[key],  
176 - });  
177 - }  
178 - });  
179 -});  
180 -  
181 -const getFakeChartData = {  
182 - visitData,  
183 - visitData2,  
184 - salesData,  
185 - searchData,  
186 - offlineData,  
187 - offlineChartData,  
188 - salesTypeData,  
189 - salesTypeDataOnline,  
190 - salesTypeDataOffline,  
191 - radarData,  
192 -}; 1 +import mockjs from 'mockjs';
193 2
194 export default { 3 export default {
195 - 'GET /api/fake_chart_data': getFakeChartData, 4 + 'GET /api/tags': mockjs.mock({
  5 + 'list|100': [{ name: '@city', 'value|1-100': 150, 'type|0-2': 1 }],
  6 + }),
196 }; 7 };
1 -module.exports = {  
2 - navTheme: 'dark', // theme for nav menu  
3 - primaryColor: '#1890FF', // primary color of ant design  
4 - layout: 'sidemenu', // nav menu position: sidemenu or topmenu  
5 - contentWidth: 'Fluid', // layout of content: Fluid or Fixed, only works when layout is topmenu  
6 - fixedHeader: false, // sticky header  
7 - autoHideHeader: false, // auto hide header  
8 - fixSiderbar: false, // sticky siderbar  
9 -};  
@@ -2,17 +2,15 @@ import React, { PureComponent } from 'react'; @@ -2,17 +2,15 @@ import React, { PureComponent } from 'react';
2 import { connect } from 'dva'; 2 import { connect } from 'dva';
3 import { formatMessage, FormattedMessage } from 'umi/locale'; 3 import { formatMessage, FormattedMessage } from 'umi/locale';
4 import { Row, Col, Card, Tooltip } from 'antd'; 4 import { Row, Col, Card, Tooltip } from 'antd';
5 -import { Pie, WaterWave, Gauge, TagCloud } from 'ant-design-pro/lib/Charts';  
6 -import NumberInfo from 'ant-design-pro/lib/NumberInfo'; 5 +import { NumberInfo, Charts } from 'ant-design-pro';
7 import CountDown from 'ant-design-pro/lib/CountDown'; 6 import CountDown from 'ant-design-pro/lib/CountDown';
8 import ActiveChart from '@/components/ActiveChart'; 7 import ActiveChart from '@/components/ActiveChart';
9 import numeral from 'numeral'; 8 import numeral from 'numeral';
10 -import GridContent from '@/components/PageHeaderWrapper/GridContent';  
11 9
12 -import Authorized from '@/utils/Authorized';  
13 -import styles from './Monitor.less'; 10 +import styles from './style.less';
  11 +
  12 +const { Pie, WaterWave, Gauge, TagCloud } = Charts;
14 13
15 -const { Secured } = Authorized;  
16 14
17 const targetTime = new Date().getTime() + 3900000; 15 const targetTime = new Date().getTime() + 3900000;
18 16
@@ -22,7 +20,6 @@ const havePermissionAsync = new Promise(resolve => { @@ -22,7 +20,6 @@ const havePermissionAsync = new Promise(resolve => {
22 setTimeout(() => resolve(), 300); 20 setTimeout(() => resolve(), 300);
23 }); 21 });
24 22
25 -@Secured(havePermissionAsync)  
26 @connect(({ monitor, loading }) => ({ 23 @connect(({ monitor, loading }) => ({
27 monitor, 24 monitor,
28 loading: loading.models.monitor, 25 loading: loading.models.monitor,
@@ -40,7 +37,7 @@ class Monitor extends PureComponent { @@ -40,7 +37,7 @@ class Monitor extends PureComponent {
40 const { tags } = monitor; 37 const { tags } = monitor;
41 38
42 return ( 39 return (
43 - <GridContent> 40 + <React.Fragment>
44 <Row gutter={24}> 41 <Row gutter={24}>
45 <Col xl={18} lg={24} md={24} sm={24} xs={24} style={{ marginBottom: 24 }}> 42 <Col xl={18} lg={24} md={24} sm={24} xs={24} style={{ marginBottom: 24 }}>
46 <Card 43 <Card
@@ -237,7 +234,7 @@ class Monitor extends PureComponent { @@ -237,7 +234,7 @@ class Monitor extends PureComponent {
237 </Card> 234 </Card>
238 </Col> 235 </Col>
239 </Row> 236 </Row>
240 - </GridContent> 237 + </React.Fragment>
241 ); 238 );
242 } 239 }
243 } 240 }
  1 +export default {
  2 + 'app.monitor.trading-activity': 'Real-Time Trading Activity',
  3 + 'app.monitor.total-transactions': 'Total transactions today',
  4 + 'app.monitor.sales-target': 'Sales target completion rate',
  5 + 'app.monitor.remaining-time': 'Remaining time of activity',
  6 + 'app.monitor.total-transactions-per-second': 'Total transactions per second',
  7 + 'app.monitor.activity-forecast': 'Activity forecast',
  8 + 'app.monitor.efficiency': 'Efficiency',
  9 + 'app.monitor.ratio': 'Ratio',
  10 + 'app.monitor.proportion-per-category': 'Proportion Per Category',
  11 + 'app.monitor.fast-food': 'Fast food',
  12 + 'app.monitor.western-food': 'Western food',
  13 + 'app.monitor.hot-pot': 'Hot pot',
  14 + 'app.monitor.waiting-for-implementation': 'Waiting for implementation',
  15 + 'app.monitor.popular-searches': 'Popular Searches',
  16 + 'app.monitor.resource-surplus': 'Resource Surplus',
  17 + 'app.monitor.fund-surplus': 'Fund Surplus',
  18 +};
  1 +export default {
  2 + 'app.monitor.trading-activity': 'Real-Time Trading Activity',
  3 + 'app.monitor.total-transactions': 'Total transactions today',
  4 + 'app.monitor.sales-target': 'Sales target completion rate',
  5 + 'app.monitor.remaining-time': 'Remaining time of activity',
  6 + 'app.monitor.total-transactions-per-second': 'Total transactions per second',
  7 + 'app.monitor.activity-forecast': 'Activity forecast',
  8 + 'app.monitor.efficiency': 'Efficiency',
  9 + 'app.monitor.ratio': 'Ratio',
  10 + 'app.monitor.proportion-per-category': 'Proportion Per Category',
  11 + 'app.monitor.fast-food': 'Fast food',
  12 + 'app.monitor.western-food': 'Western food',
  13 + 'app.monitor.hot-pot': 'Hot pot',
  14 + 'app.monitor.waiting-for-implementation': 'Waiting for implementation',
  15 + 'app.monitor.popular-searches': 'Popular Searches',
  16 + 'app.monitor.resource-surplus': 'Resource Surplus',
  17 + 'app.monitor.fund-surplus': 'Fund Surplus',
  18 + 'app.exception.back': 'Back to home',
  19 +};
  1 +export default {
  2 + 'app.monitor.trading-activity': '活动实时交易情况',
  3 + 'app.monitor.total-transactions': '今日交易总额',
  4 + 'app.monitor.sales-target': '销售目标完成率',
  5 + 'app.monitor.remaining-time': '活动剩余时间',
  6 + 'app.monitor.total-transactions-per-second': '每秒交易总额',
  7 + 'app.monitor.activity-forecast': '活动情况预测',
  8 + 'app.monitor.efficiency': '券核效率',
  9 + 'app.monitor.ratio': '跳出率',
  10 + 'app.monitor.proportion-per-category': '各品类占比',
  11 + 'app.monitor.fast-food': '中式快餐',
  12 + 'app.monitor.western-food': '西餐',
  13 + 'app.monitor.hot-pot': '火锅',
  14 + 'app.monitor.waiting-for-implementation': 'Waiting for implementation',
  15 + 'app.monitor.popular-searches': '热门搜索',
  16 + 'app.monitor.resource-surplus': '资源剩余',
  17 + 'app.monitor.fund-surplus': '补贴资金剩余',
  18 +};
  1 +export default {
  2 + 'app.monitor.trading-activity': '活動實時交易情況',
  3 + 'app.monitor.total-transactions': '今日交易總額',
  4 + 'app.monitor.sales-target': '銷售目標完成率',
  5 + 'app.monitor.remaining-time': '活動剩余時間',
  6 + 'app.monitor.total-transactions-per-second': '每秒交易總額',
  7 + 'app.monitor.activity-forecast': '活動情況預測',
  8 + 'app.monitor.efficiency': '券核效率',
  9 + 'app.monitor.ratio': '跳出率',
  10 + 'app.monitor.proportion-per-category': '各品類占比',
  11 + 'app.monitor.fast-food': '中式快餐',
  12 + 'app.monitor.western-food': '西餐',
  13 + 'app.monitor.hot-pot': '火鍋',
  14 + 'app.monitor.waiting-for-implementation': 'Waiting for implementation',
  15 + 'app.monitor.popular-searches': '熱門搜索',
  16 + 'app.monitor.resource-surplus': '資源剩余',
  17 + 'app.monitor.fund-surplus': '補貼資金剩余',
  18 +};
1 -import { queryTags } from '@/services/api'; 1 +import { queryTags } from './service';
2 2
3 export default { 3 export default {
4 namespace: 'monitor', 4 namespace: 'monitor',
1 -import { message } from 'antd';  
2 -import defaultSettings from '../defaultSettings';  
3 -  
4 -let lessNodesAppended;  
5 -const updateTheme = primaryColor => {  
6 - // Don't compile less in production!  
7 - if (APP_TYPE !== 'site') {  
8 - return;  
9 - }  
10 - // Determine if the component is remounted  
11 - if (!primaryColor) {  
12 - return;  
13 - }  
14 - const hideMessage = message.loading('正在编译主题!', 0);  
15 - function buildIt() {  
16 - if (!window.less) {  
17 - return;  
18 - }  
19 - setTimeout(() => {  
20 - window.less  
21 - .modifyVars({  
22 - '@primary-color': primaryColor,  
23 - })  
24 - .then(() => {  
25 - hideMessage();  
26 - })  
27 - .catch(() => {  
28 - message.error('Failed to update theme');  
29 - hideMessage();  
30 - });  
31 - }, 200);  
32 - }  
33 - if (!lessNodesAppended) {  
34 - // insert less.js and color.less  
35 - const lessStyleNode = document.createElement('link');  
36 - const lessConfigNode = document.createElement('script');  
37 - const lessScriptNode = document.createElement('script');  
38 - lessStyleNode.setAttribute('rel', 'stylesheet/less');  
39 - lessStyleNode.setAttribute('href', '/color.less');  
40 - lessConfigNode.innerHTML = `  
41 - window.less = {  
42 - async: true,  
43 - env: 'production',  
44 - javascriptEnabled: true  
45 - };  
46 - `;  
47 - lessScriptNode.src = 'https://gw.alipayobjects.com/os/lib/less.js/3.8.1/less.min.js';  
48 - lessScriptNode.async = true;  
49 - lessScriptNode.onload = () => {  
50 - buildIt();  
51 - lessScriptNode.onload = null;  
52 - };  
53 - document.body.appendChild(lessStyleNode);  
54 - document.body.appendChild(lessConfigNode);  
55 - document.body.appendChild(lessScriptNode);  
56 - lessNodesAppended = true;  
57 - } else {  
58 - buildIt();  
59 - }  
60 -};  
61 -  
62 -const updateColorWeak = colorWeak => {  
63 - document.body.className = colorWeak ? 'colorWeak' : '';  
64 -};  
65 -  
66 -export default {  
67 - namespace: 'setting',  
68 - state: defaultSettings,  
69 - reducers: {  
70 - getSetting(state) {  
71 - const setting = {};  
72 - const urlParams = new URL(window.location.href);  
73 - Object.keys(state).forEach(key => {  
74 - if (urlParams.searchParams.has(key)) {  
75 - const value = urlParams.searchParams.get(key);  
76 - setting[key] = value === '1' ? true : value;  
77 - }  
78 - });  
79 - const { primaryColor, colorWeak } = setting;  
80 - if (state.primaryColor !== primaryColor) {  
81 - updateTheme(primaryColor);  
82 - }  
83 - updateColorWeak(colorWeak);  
84 - return {  
85 - ...state,  
86 - ...setting,  
87 - };  
88 - },  
89 - changeSetting(state, { payload }) {  
90 - const urlParams = new URL(window.location.href);  
91 - Object.keys(defaultSettings).forEach(key => {  
92 - if (urlParams.searchParams.has(key)) {  
93 - urlParams.searchParams.delete(key);  
94 - }  
95 - });  
96 - Object.keys(payload).forEach(key => {  
97 - if (key === 'collapse') {  
98 - return;  
99 - }  
100 - let value = payload[key];  
101 - if (value === true) {  
102 - value = 1;  
103 - }  
104 - if (defaultSettings[key] !== value) {  
105 - urlParams.searchParams.set(key, value);  
106 - }  
107 - });  
108 - const { primaryColor, colorWeak, contentWidth } = payload;  
109 - if (state.primaryColor !== primaryColor) {  
110 - updateTheme(primaryColor);  
111 - }  
112 - if (state.contentWidth !== contentWidth && window.dispatchEvent) {  
113 - window.dispatchEvent(new Event('resize'));  
114 - }  
115 - updateColorWeak(colorWeak);  
116 - window.history.replaceState(null, 'setting', urlParams.href);  
117 - return {  
118 - ...state,  
119 - ...payload,  
120 - };  
121 - },  
122 - },  
123 -};  
  1 +import request from 'umi-request';
  2 +
  3 +export async function queryTags() {
  4 + return request('/api/tags');
  5 +}
1 @import '~antd/lib/style/themes/default.less'; 1 @import '~antd/lib/style/themes/default.less';
2 -@import '~@/utils/utils.less';  
3 2
4 .mapChart { 3 .mapChart {
5 padding-top: 24px; 4 padding-top: 24px;
注册登录 后发表评论