satori
3 years ago
8 changed files with 413 additions and 104 deletions
@ -0,0 +1,10 @@
|
||||
# FROM node:8.10-alpine |
||||
FROM node:12 |
||||
WORKDIR /usr/src/app |
||||
COPY package.json ./ |
||||
# RUN apt install git python3 |
||||
RUN npm install --only=production |
||||
RUN npm i react@16.14.0 |
||||
RUN npm audit fix |
||||
COPY . . |
||||
CMD [ "npm", "start" ] |
@ -0,0 +1,128 @@
|
||||
.number-asset { |
||||
font-size: 50px; |
||||
font-weight: bold; |
||||
text-align: center; |
||||
} |
||||
|
||||
.text-bold { |
||||
font-weight: bold; |
||||
} |
||||
|
||||
.view-rectangle { |
||||
display: flex; |
||||
flex-direction: row; |
||||
width: 100%; |
||||
justify-content: space-between; |
||||
} |
||||
|
||||
.view1 { |
||||
width: 220px; |
||||
height: 100px; |
||||
padding: 10px; |
||||
margin-bottom: -20px; |
||||
|
||||
justify-content: center; |
||||
align-items: center; |
||||
text-align: center; |
||||
background-color: #008B8B; |
||||
} |
||||
|
||||
.view2 { |
||||
width: 220px; |
||||
height: 100px; |
||||
padding: 10px; |
||||
|
||||
justify-content: center; |
||||
align-items: center; |
||||
text-align: center; |
||||
background-color: #8B008B; |
||||
} |
||||
|
||||
.view3 { |
||||
width: 220px; |
||||
height: 100px; |
||||
padding: 10px; |
||||
|
||||
justify-content: center; |
||||
align-items: center; |
||||
text-align: center; |
||||
background-color: #7CFC00; |
||||
} |
||||
|
||||
.view4 { |
||||
width: 220px; |
||||
height: 100px; |
||||
padding: 10px; |
||||
|
||||
justify-content: center; |
||||
align-items: center; |
||||
text-align: center; |
||||
background-color: #FF0000; |
||||
} |
||||
|
||||
.view5 { |
||||
width: 220px; |
||||
height: 100px; |
||||
padding: 10px; |
||||
|
||||
justify-content: center; |
||||
align-items: center; |
||||
text-align: center; |
||||
background-color: #4682B4; |
||||
} |
||||
|
||||
.number-style { |
||||
font-size: 30px; |
||||
color: black; |
||||
font-weight: bold; |
||||
margin-bottom:15px; |
||||
} |
||||
|
||||
.number-style1 { |
||||
font-size: 30px; |
||||
color: #FFFFFF; |
||||
font-weight: bold; |
||||
margin-bottom:15px; |
||||
} |
||||
|
||||
.daily-info-card { |
||||
min-width: 130px; |
||||
width: 220px; |
||||
height: 100px; |
||||
padding: 10px; |
||||
justify-content: center; |
||||
align-items: center; |
||||
text-align: center; |
||||
margin: 4px; |
||||
border-radius: 8px; |
||||
} |
||||
|
||||
|
||||
.dashboard-container { |
||||
overflow-x: auto; |
||||
} |
||||
|
||||
.maptable-window-button-container { |
||||
float: right; |
||||
right: 0px; |
||||
} |
||||
|
||||
.maptable-close, .maptable-maximize, .maptable-minimize { |
||||
cursor: pointer; |
||||
padding: 4px; |
||||
} |
||||
|
||||
.maptable-header { |
||||
margin-bottom: -10px; |
||||
} |
||||
|
||||
.maptable-title { |
||||
font-size: 24px; |
||||
font-weight: 700; |
||||
text-align: left; |
||||
margin-left: 20px; |
||||
} |
||||
|
||||
.maptable-close:hover, .maptable-maximize:hover, .maptable-minimize:hover { |
||||
color: #20a8d8; |
||||
} |
@ -0,0 +1,41 @@
|
||||
import React, { useState, useEffect } from 'react'; |
||||
import ReactDOM from 'react-dom'; |
||||
import { Pie } from '@ant-design/plots'; |
||||
|
||||
const PieChart = () => { |
||||
const data = [ |
||||
{ |
||||
type: 'FTTH', |
||||
value: 27, |
||||
}, |
||||
{ |
||||
type: 'Konstruksi', |
||||
value: 25, |
||||
}, |
||||
]; |
||||
const config = { |
||||
appendPadding: 10, |
||||
data, |
||||
angleField: 'value', |
||||
colorField: 'type', |
||||
radius: 0.9, |
||||
label: { |
||||
type: 'inner', |
||||
offset: '-30%', |
||||
content: ({ percent }) => `${(percent * 100).toFixed(0)}%`, |
||||
style: { |
||||
fontSize: 14, |
||||
textAlign: 'center', |
||||
}, |
||||
}, |
||||
interactions: [ |
||||
{ |
||||
type: 'element-active', |
||||
}, |
||||
], |
||||
}; |
||||
|
||||
return <Pie {...config} />; |
||||
}; |
||||
|
||||
export default PieChart; |
@ -0,0 +1,114 @@
|
||||
import '../../../node_modules/react-grid-layout/css/styles.css'; |
||||
import '../../../node_modules/react-resizable/css/styles.css'; |
||||
import './Dashboard.css'; |
||||
import PieChart from './PieChart'; |
||||
import { BASE_OSPRO } from '../../const/ApiConst'; |
||||
import ContentLoader from "react-content-loader" |
||||
import React, { useEffect, useState } from 'react'; |
||||
import axios from 'axios' |
||||
import moment from 'moment'; |
||||
import numeral from 'numeral'; |
||||
import { Table, Tree, Row, Col, Space, Card} from 'antd'; |
||||
import { Pie } from '@ant-design/plots'; |
||||
import { NotificationContainer, NotificationManager } from 'react-notifications'; |
||||
import { formatRibuanDecimal, DATE_TIME_FORMAT } from '../../const/CustomFunc.js'; |
||||
import { Badge } from 'reactstrap'; |
||||
|
||||
const token = localStorage.getItem("token") |
||||
const HEADER = { |
||||
headers: { |
||||
"Content-Type": "application/json", |
||||
"Authorization": `Bearer ${token}`, |
||||
"Access-Control-Allow-Origin": "*" |
||||
} |
||||
} |
||||
|
||||
const Dashboard = () => { |
||||
const token = localStorage.getItem("token") |
||||
const HEADER = { |
||||
headers: { |
||||
"Content-Type": "application/json", |
||||
"Authorization": `Bearer ${token}` |
||||
} |
||||
} |
||||
|
||||
const [dataTable, setDataTable] = useState([]) |
||||
const getProjects = async () => { |
||||
const URL = `${BASE_OSPRO}/api/project/list` |
||||
const result = await axios.get(URL, HEADER).then(res => res).catch(err => err.response) |
||||
if (result.data.code !== 200) { |
||||
NotificationManager.error('Belum ada data proyek!', 'Failed'); |
||||
} |
||||
console.log("res ", result.data.data) |
||||
setDataTable(result.data.data); |
||||
} |
||||
|
||||
const columns = [ |
||||
{ |
||||
title: 'Project', |
||||
dataIndex: 'nama', |
||||
key: 'nama', |
||||
render: (text) => <a>{text}</a>, |
||||
}, |
||||
{ |
||||
title: 'Planned Interval', |
||||
dataIndex: 'plannedInterval', |
||||
key: 'plannedInterval', |
||||
}, |
||||
{ |
||||
title: 'Planned Cost', |
||||
dataIndex: 'plannedCost', |
||||
key: 'plannedCost', |
||||
render: (text) => <a>{ formatRibuanDecimal(text) }</a>, |
||||
}, |
||||
{ |
||||
title: 'Actual Cost', |
||||
dataIndex: 'actualCost', |
||||
key: 'actualCost', |
||||
render: (text) => <a>{ formatRibuanDecimal(text) }</a>, |
||||
}, |
||||
{ |
||||
title: 'Cost Variance', |
||||
dataIndex: 'costVariance', |
||||
key: 'costVariance', |
||||
render: (text) => <a>{ formatRibuanDecimal(text) }</a>, |
||||
}, |
||||
{ |
||||
title: 'Cost Health', |
||||
dataIndex: 'costHealth', |
||||
key: 'costHealth', |
||||
render: (text) => { |
||||
if(text == "on-budget") { |
||||
return <Badge color="success">On Budget</Badge> |
||||
} else if(text == "warning") { |
||||
return <Badge color="warning">Warning</Badge> |
||||
} else { |
||||
return <Badge color="danger">Danger</Badge> |
||||
} |
||||
} |
||||
}, |
||||
{ |
||||
title: 'Progress', |
||||
dataIndex: 'progress', |
||||
key: 'progress', |
||||
}, |
||||
]; |
||||
|
||||
useEffect(() => { |
||||
getProjects(); |
||||
}, []) |
||||
|
||||
|
||||
return ( |
||||
<> |
||||
<Row gutter={[16, 16]}> |
||||
<Col span={24}> |
||||
<Table columns={columns} dataSource={dataTable} /> |
||||
</Col> |
||||
</Row> |
||||
</> |
||||
|
||||
); |
||||
} |
||||
|
||||
export default Dashboard; |
Loading…
Reference in new issue