diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/package.json b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/package.json
index 3029a00c3f..ba015d08d1 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/package.json
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/package.json
@@ -16,10 +16,11 @@
"react": "^16.8.4",
"react-dom": "^16.8.4",
"react-highlight-words": "^0.16.0",
+ "react-router": "latest",
"react-router-config": "^5.0.0",
"react-router-dom": "latest",
"react-scripts": "2.1.8",
- "react-router": "latest"
+ "redux-thunk": "^2.3.0"
},
"devDependencies": {
"@babel/core": "^7.0.0",
@@ -37,6 +38,7 @@
"html-loader": "^0.5.5",
"html-webpack-plugin": "^3.2.0",
"img-loader": "^3.0.1",
+ "json-loader": "^0.5.7",
"less": "^3.9.0",
"less-loader": "^4.1.0",
"mini-css-extract-plugin": "^0.5.0",
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/public/conf/config.json b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/public/conf/config.json
index 63678cb541..c2e66dda14 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/public/conf/config.json
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/public/conf/config.json
@@ -3,10 +3,16 @@
"type": "default",
"value": "lightBaseTheme"
},
- "config": {
+ "serverConfig": {
"hostname": "localhost",
"httpsPort": "9443",
- "apiPort": "8243"
+ "invokerUri": "/api/application-mgt-handler/v1.0/invoke",
+ "loginUri": "/api/application-mgt-handler/v1.0/login"
},
- "serverUrl" : "https://localhost:9443"
+ "serverUrl" : "https://localhost:9443",
+ "defaultPlatformIcons": {
+ "default": "http://www.newdesignfile.com/postpic/2015/08/square-app-icon-blue_77131.png",
+ "android": "",
+ "ios" : ""
+ }
}
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/App.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/App.js
index be1c130124..9c532989fa 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/App.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/App.js
@@ -3,27 +3,33 @@ import "antd/dist/antd.css";
import RouteWithSubRoutes from "./components/RouteWithSubRoutes";
import {
BrowserRouter as Router,
- Link,
+ Link, Redirect, Switch,
} from 'react-router-dom';
class App extends React.Component {
routes;
+
constructor(props) {
super(props);
this.routes = props.routes;
}
- render() {
- return (
-
-
- {this.routes.map((route) => (
-
- ))}
-
-
- );
- }
+ render() {
+ console.log(this.routes);
+ return (
+
+
+
+
+ {this.routes.map((route) => (
+
+ ))}
+
+
+
+
+ );
+ }
}
export default App;
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/AppCard.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/AppCard.js
new file mode 100644
index 0000000000..d965e05d7f
--- /dev/null
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/AppCard.js
@@ -0,0 +1,47 @@
+import {
+ Skeleton, Switch, Card, Icon, Avatar, Typography
+} from 'antd';
+import React from "react";
+import config from "../../public/conf/config.json";
+
+const { Meta } = Card;
+const { Text } = Typography;
+
+class AppCard extends React.Component {
+
+ constructor(props){
+ super(props);
+ }
+
+
+ render() {
+ const defaultPlatformIcons = config.defaultPlatformIcons;
+ let icon = defaultPlatformIcons.default;
+ if(defaultPlatformIcons.hasOwnProperty(this.props.platform)){
+ icon = defaultPlatformIcons[this.props.platform];
+ }
+ let descriptionText = this.props.description;
+ if(descriptionText.length>50){
+ descriptionText = descriptionText.substring(0,50)+"...";
+ }
+ const description = (
+
+
{descriptionText}
+
{this.props.type}
+
{this.props.subType}
+
+ );
+
+ return (
+ , , ]}>
+ }
+ title={this.props.name}
+ description={description}
+ />
+
+ );
+ }
+}
+
+export default AppCard;
\ No newline at end of file
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/AppList.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/AppList.js
new file mode 100644
index 0000000000..d6c54f4039
--- /dev/null
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/AppList.js
@@ -0,0 +1,41 @@
+import React from "react";
+import AppCard from "./AppCard";
+import {Col, Row} from "antd";
+import {connect} from "react-redux";
+import {getApps} from "../js/actions";
+
+// connecting state.articles with the component
+const mapStateToProps= state => {
+ return {apps : state.apps}
+};
+
+
+class ConnectedAppList extends React.Component {
+ constructor(props) {
+ super(props);
+ }
+ componentDidMount() {
+ this.props.getApps();
+ }
+
+ render() {
+ return (
+
+ {this.props.apps.map(app => (
+
+
+
+ ))}
+
+ );
+ }
+}
+
+const AppList = connect(mapStateToProps,{getApps})(ConnectedAppList);
+
+export default AppList;
\ No newline at end of file
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/RouteWithSubRoutes.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/RouteWithSubRoutes.js
index c172b28edd..41cb84caf8 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/RouteWithSubRoutes.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/components/RouteWithSubRoutes.js
@@ -8,7 +8,7 @@ class RouteWithSubRoutes extends React.Component{
}
render() {
return(
- (
+ (
)}/>
);
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/index.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/index.js
index df386140c4..829941613a 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/index.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/index.js
@@ -14,19 +14,23 @@ import {Provider} from "react-redux";
const routes = [
{
path: '/publisher/login',
+ exact: true,
component: Login
},
{
- path: '/publisher',
+ path: '/publisher/apps',
+ exact: false,
component: Dashboard,
routes: [
{
path: '/publisher/apps',
- component: Apps
+ component: Apps,
+ exact: true
},
{
- path: '/publisher/new-app',
- component: AddNewApp
+ path: '/publisher/apps/new-app',
+ component: AddNewApp,
+ exact: true
}
]
}
@@ -35,7 +39,7 @@ const routes = [
ReactDOM.render(
-
+
,
document.getElementById('root'));
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/actions/index.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/actions/index.js
new file mode 100644
index 0000000000..05da75eb70
--- /dev/null
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/actions/index.js
@@ -0,0 +1,31 @@
+import axios from "axios";
+import {GET_APPS} from "../constants/action-types";
+import config from "../../../public/conf/config.json";
+
+export function getApps() {
+
+ return (dispatch) => {
+ const request = "method=post&content-type=application/json&payload={}&api-endpoint=/application-mgt-publisher/v1.0/applications";
+
+ return axios.post('https://'+config.serverConfig.hostname+':'+config.serverConfig.httpsPort+config.serverConfig.invokerUri, request
+ ).then(res => {
+ if (res.status === 200) {
+ let apps = [];
+
+ if(res.data.data.hasOwnProperty("applications")){
+ apps = res.data.data.applications;
+ }
+ console.log(res.data);
+ dispatch({type: GET_APPS, payload: apps});
+ }
+
+ }).catch(function (error) {
+ if (error.response.status === 401) {
+ window.location.href = 'https://localhost:9443/publisher/login';
+ }
+ });
+
+ };
+
+
+}
\ No newline at end of file
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/constants/action-types.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/constants/action-types.js
index 2cd535bbc2..e270ea0c6a 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/constants/action-types.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/constants/action-types.js
@@ -1 +1,3 @@
-export const LOGIN = "LOGIN";
\ No newline at end of file
+export const LOGIN = "LOGIN";
+export const GET_APPS = "GET_APPS";
+
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/reducers/index.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/reducers/index.js
index 08eb66ed6d..3093402642 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/reducers/index.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/reducers/index.js
@@ -1,8 +1,16 @@
+import {GET_APPS} from "../constants/action-types";
+
const initialState = {
-
+ apps: []
};
function rootReducer(state = initialState, action) {
+ if (action.type === GET_APPS) {
+ console.log(11);
+ return Object.assign({}, state, {
+ apps: action.payload
+ });
+ }
return state;
}
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/store/index.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/store/index.js
index 7209a4b021..04957eb39a 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/store/index.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/js/store/index.js
@@ -1,4 +1,5 @@
-import { createStore } from "redux";
+import { createStore, applyMiddleware } from "redux";
import rootReducer from "../reducers/index";
-const store = createStore(rootReducer);
+import thunk from "redux-thunk";
+const store = createStore(rootReducer, applyMiddleware(thunk));
export default store;
\ No newline at end of file
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/Login.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/Login.js
index e49b031118..8953e32352 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/Login.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/Login.js
@@ -2,6 +2,7 @@ import React from "react";
import {Typography, Row, Col, Form, Icon, Input, Button, Checkbox} from 'antd';
import styles from './Login.less';
import axios from 'axios';
+import config from "../../public/conf/config.json";
const {Title} = Typography;
const {Text} = Typography;
@@ -58,7 +59,7 @@ class NormalLoginForm extends React.Component {
});
console.log('Received values of form: ', values);
let data = "username=" + values.username + "&password=" + values.password + "&platform=publisher";
- axios.post('https://localhost:9443/api/application-mgt-handler/v1.0/login', data
+ axios.post('https://'+config.serverConfig.hostname+':'+config.serverConfig.httpsPort+config.serverConfig.loginUri, data
).then(res => {
if (res.status === 200) {
window.location = res.data.url;
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/Dashboard.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/Dashboard.js
index 9d8569b12a..396e335f2e 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/Dashboard.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/Dashboard.js
@@ -31,14 +31,14 @@ class Dashboard extends React.Component {
defaultSelectedKeys={['2']}
style={{lineHeight: '64px'}}
>
- Apps
- Apps
- Add New App
+ Apps
+ Apps
+ Add New App
-
+
{this.state.routes.map((route) => (
))}
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/Apps.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/Apps.js
index 3a2994b184..92d366a3ae 100644
--- a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/Apps.js
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/Apps.js
@@ -1,9 +1,8 @@
import React from "react";
import "antd/dist/antd.css";
-import {Table, Divider, Tag, Card, PageHeader, Typography, Avatar,Input, Button, Icon, Row, Col} from "antd";
-import Highlighter from 'react-highlight-words';
+import {PageHeader, Typography,Input, Button, Row, Col} from "antd";
+import AppList from "../../../components/AppList";
-const Paragraph = Typography;
const Search = Input.Search;
const routes = [
@@ -22,188 +21,32 @@ const routes = [
];
-
-
-const data = [{
- key: '1',
- icon: 'https://gw.alipayobjects.com/zos/rmsportal/zOsKZmFRdUtvpqCImOVY.png',
- name: 'John Brown',
- platform: 'android',
- type: 'Enterprise',
- status: 'published',
- version: '13.0.0.1',
- updated_at: '27-03-2019 08:27'
-},{
- key: '2',
- icon: 'http://aztechbeat.com/wp-content/uploads/2014/04/confide-app-icon.png',
- name: 'Lorem Ipsum',
- platform: 'ios',
- type: 'Enterprise',
- status: 'published',
- version: '2.3.1.2',
- updated_at: '27-03-2019 09:45'
-},{
- key: '3',
- icon: 'https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcRx2Xx1-hnH16EGZHUlT06nOcfGODPoboA2TXKaBVtODto4lJtK',
- name: 'Lorem Ipsum',
- platform: 'ios',
- type: 'Enterprise',
- status: 'removed',
- version: '4.1.1.0',
- updated_at: '27-03-2019 09:46'
-}];
-
class Apps extends React.Component {
routes;
-
-
- state = {
- searchText: '',
- };
-
-
constructor(props) {
super(props);
this.routes = props.routes;
- }
- getColumnSearchProps = (dataIndex) => ({
- filterDropdown: ({
- setSelectedKeys, selectedKeys, confirm, clearFilters,
- }) => (
-
- { this.searchInput = node; }}
- placeholder={`Search ${dataIndex}`}
- value={selectedKeys[0]}
- onChange={e => setSelectedKeys(e.target.value ? [e.target.value] : [])}
- onPressEnter={() => this.handleSearch(selectedKeys, confirm)}
- style={{ width: 188, marginBottom: 8, display: 'block' }}
- />
-
-
-
- ),
- filterIcon: filtered => ,
- onFilter: (value, record) => record[dataIndex].toString().toLowerCase().includes(value.toLowerCase()),
- onFilterDropdownVisibleChange: (visible) => {
- if (visible) {
- setTimeout(() => this.searchInput.select());
- }
- },
- render: (text) => (
-
- ),
- })
-
- handleSearch = (selectedKeys, confirm) => {
- confirm();
- this.setState({ searchText: selectedKeys[0] });
- }
-
- handleReset = (clearFilters) => {
- clearFilters();
- this.setState({ searchText: '' });
}
-
render() {
- const columns = [{
- title: '',
- dataIndex: 'icon',
- key: 'icon',
- render: text => ,
- }, {
- title: 'Name',
- dataIndex: 'name',
- key: 'name',
- render: text => {text},
- ...this.getColumnSearchProps('name'),
- }, {
- title: 'Platform',
- dataIndex: 'platform',
- key: 'platform',
- }, {
- title: 'Type',
- dataIndex: 'type',
- key: 'type',
- }, {
- title: 'Status',
- key: 'status',
- dataIndex: 'status',
- render: tag => {
- let color;
- switch (tag) {
- case 'published':
- color = 'green';
- break;
- case 'removed':
- color = 'red'
- break;
- case 'default':
- color = 'blue'
- }
- return {tag.toUpperCase()};
- },
- }, {
- title: 'Published Version',
- dataIndex: 'version',
- key: 'version',
- }, {
- title: 'Last Updated',
- dataIndex: 'updated_at',
- key: 'updated_at',
- },{
- title: 'Action',
- key: 'action',
- render: () => (
-
- Edit
-
- Manage
-
- ),
- }];
-
return (
-
-
-
-
- console.log(value)}
- style={{ width: 200}}
- />
-
-
-
-
-
+
+
+ console.log(value)}
+ style={{ width: 200}}
+ />
+
+
+
+
diff --git a/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/OldApps.js b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/OldApps.js
new file mode 100644
index 0000000000..75750bfc36
--- /dev/null
+++ b/components/application-mgt/org.wso2.carbon.device.application.mgt.publisher.ui/react-app/src/pages/dashboard/apps/OldApps.js
@@ -0,0 +1,226 @@
+import React from "react";
+import "antd/dist/antd.css";
+import {Table, Divider, Tag, Card, PageHeader, Typography, Avatar,Input, Button, Icon, Row, Col} from "antd";
+import Highlighter from 'react-highlight-words';
+import axios from "axios";
+
+const Paragraph = Typography;
+const Search = Input.Search;
+
+const routes = [
+ {
+ path: 'index',
+ breadcrumbName: 'Publisher',
+ },
+ {
+ path: 'first',
+ breadcrumbName: 'Dashboard',
+ },
+ {
+ path: 'second',
+ breadcrumbName: 'OldApps',
+ },
+];
+
+
+class OldApps extends React.Component {
+ routes;
+
+ state = {
+ searchText: '',
+ };
+
+
+ constructor(props) {
+ super(props);
+ this.routes = props.routes;
+ this.state = {
+ data: []
+ };
+
+ this.loadData = this.loadData.bind(this);
+ }
+
+ loadData(){
+ const thisComponent = this;
+ const request = "method=post&content-type=application/json&payload={}&api-endpoint=/application-mgt-publisher/v1.0/applications";
+ axios.post('https://localhost:9443/api/application-mgt-handler/v1.0/invoke', request
+ ).then(res => {
+ if(res.status === 200){
+ console.log(res.status);
+ let apps = [];
+ res.data['data']['applications'].forEach(function (app) {
+ apps.push({
+ key: app.id,
+ icon: app["applicationReleases"][0]["iconPath"],
+ name: app.name,
+ platform: 'undefined',
+ type: app.type,
+ status: 'undefined',
+ version: 'undefined',
+ updated_at: 'undefined'
+ });
+ });
+ thisComponent.setState({
+ data : apps
+ })
+ }
+
+ }).catch(function (error) {
+ if(error.response.status === 401){
+ window.location.href = 'https://localhost:9443/publisher/login';
+ }
+ });
+ }
+
+ componentDidMount() {
+ this.loadData();
+
+ }
+
+ getColumnSearchProps = (dataIndex) => ({
+ filterDropdown: ({
+ setSelectedKeys, selectedKeys, confirm, clearFilters,
+ }) => (
+
+ { this.searchInput = node; }}
+ placeholder={`Search ${dataIndex}`}
+ value={selectedKeys[0]}
+ onChange={e => setSelectedKeys(e.target.value ? [e.target.value] : [])}
+ onPressEnter={() => this.handleSearch(selectedKeys, confirm)}
+ style={{ width: 188, marginBottom: 8, display: 'block' }}
+ />
+
+
+
+ ),
+ filterIcon: filtered => ,
+ onFilter: (value, record) => record[dataIndex].toString().toLowerCase().includes(value.toLowerCase()),
+ onFilterDropdownVisibleChange: (visible) => {
+ if (visible) {
+ setTimeout(() => this.searchInput.select());
+ }
+ },
+ render: (text) => (
+
+ ),
+ });
+
+ handleSearch = (selectedKeys, confirm) => {
+ confirm();
+ this.setState({ searchText: selectedKeys[0] });
+ };
+
+ handleReset = (clearFilters) => {
+ clearFilters();
+ this.setState({ searchText: '' });
+ };
+
+
+ render() {
+ const columns = [{
+ title: '',
+ dataIndex: 'icon',
+ key: 'icon',
+ render: text => ,
+ }, {
+ title: 'Name',
+ dataIndex: 'name',
+ key: 'name',
+ render: text => {text},
+ ...this.getColumnSearchProps('name'),
+ }, {
+ title: 'Platform',
+ dataIndex: 'platform',
+ key: 'platform',
+ }, {
+ title: 'Type',
+ dataIndex: 'type',
+ key: 'type',
+ }, {
+ title: 'Status',
+ key: 'status',
+ dataIndex: 'status',
+ render: tag => {
+ let color;
+ switch (tag) {
+ case 'published':
+ color = 'green';
+ break;
+ case 'removed':
+ color = 'red';
+ break;
+ case 'default':
+ color = 'blue';
+ }
+ return {tag.toUpperCase()};
+ },
+ }, {
+ title: 'Published Version',
+ dataIndex: 'version',
+ key: 'version',
+ }, {
+ title: 'Last Updated',
+ dataIndex: 'updated_at',
+ key: 'updated_at',
+ },{
+ title: 'Action',
+ key: 'action',
+ render: () => (
+
+ Edit
+
+ Manage
+
+ ),
+ }];
+
+ return (
+
+
+
+
+
+
+
+ console.log(value)}
+ style={{ width: 200}}
+ />
+
+
+
+
+
+
+
+
+
+ );
+ }
+}
+
+export default OldApps;