admin_core 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitattributes +2 -0
- data/.gitignore +53 -0
- data/.rspec +2 -0
- data/.rubocop.yml +27 -0
- data/CHANGELOG.md +15 -0
- data/Gemfile +3 -0
- data/README.md +48 -0
- data/Rakefile +31 -0
- data/admin_core.gemspec +32 -0
- data/client/.babelrc +19 -0
- data/client/.eslintignore +3 -0
- data/client/.eslintrc.yml +20 -0
- data/client/.flowconfig +7 -0
- data/client/.gitignore +64 -0
- data/client/README.md +3 -0
- data/client/admin-core.scss +8 -0
- data/client/flow-typed/npm/axios_v0.16.x.js +120 -0
- data/client/flow-typed/npm/classnames_v2.x.x.js +16 -0
- data/client/flow-typed/npm/lodash_v4.x.x.js +514 -0
- data/client/flow-typed/npm/react-router-dom_v4.x.x.js +166 -0
- data/client/flow-typed/npm/reactstrap_vx.x.x.js +536 -0
- data/client/package.json +60 -0
- data/client/src/.eslintrc.yml +23 -0
- data/client/src/AdminCore.jsx +44 -0
- data/client/src/components/Breadcrumb.jsx +18 -0
- data/client/src/components/Header.jsx +45 -0
- data/client/src/components/Pagination.jsx +72 -0
- data/client/src/components/ResourceFilters.jsx +87 -0
- data/client/src/components/ResourceForm.jsx +103 -0
- data/client/src/components/ResourcesCollection.jsx +41 -0
- data/client/src/components/Sidebar.jsx +90 -0
- data/client/src/decls.js +119 -0
- data/client/src/http-client.js +18 -0
- data/client/src/main.js +9 -0
- data/client/src/resource-field/BelongsTo.jsx +26 -0
- data/client/src/resource-field/Boolean.jsx +43 -0
- data/client/src/resource-field/Date.jsx +29 -0
- data/client/src/resource-field/DateTime.jsx +29 -0
- data/client/src/resource-field/Enum.jsx +34 -0
- data/client/src/resource-field/Number.jsx +28 -0
- data/client/src/resource-field/String.jsx +28 -0
- data/client/src/resource-field/Text.jsx +27 -0
- data/client/src/resource-field-renderer.js +45 -0
- data/client/src/resource-filter/Boolean.jsx +22 -0
- data/client/src/resource-filter/Number.jsx +45 -0
- data/client/src/resource-filter/String.jsx +46 -0
- data/client/src/resource-filter-renderer.js +17 -0
- data/client/src/resource-page/Base.js +36 -0
- data/client/src/resource-page/Edit.jsx +48 -0
- data/client/src/resource-page/Index.jsx +141 -0
- data/client/src/resource-page/New.jsx +48 -0
- data/client/src/resource-page/Show.jsx +116 -0
- data/client/webpack.config.js +26 -0
- data/client/yarn.lock +3816 -0
- data/lib/admin_core/base_controller.rb +114 -0
- data/lib/admin_core/base_resource_manager.rb +24 -0
- data/lib/admin_core/configuration.rb +20 -0
- data/lib/admin_core/engine.rb +6 -0
- data/lib/admin_core/errors.rb +17 -0
- data/lib/admin_core/resource_field/base.rb +69 -0
- data/lib/admin_core/resource_field/belongs_to.rb +38 -0
- data/lib/admin_core/resource_field/boolean.rb +18 -0
- data/lib/admin_core/resource_field/date.rb +18 -0
- data/lib/admin_core/resource_field/date_time.rb +18 -0
- data/lib/admin_core/resource_field/enum.rb +26 -0
- data/lib/admin_core/resource_field/number.rb +18 -0
- data/lib/admin_core/resource_field/string.rb +18 -0
- data/lib/admin_core/resource_field/text.rb +23 -0
- data/lib/admin_core/resource_field_builder.rb +48 -0
- data/lib/admin_core/resource_filter/base.rb +63 -0
- data/lib/admin_core/resource_filter/boolean.rb +17 -0
- data/lib/admin_core/resource_filter/number.rb +25 -0
- data/lib/admin_core/resource_filter/string.rb +27 -0
- data/lib/admin_core/resource_filter_builder.rb +37 -0
- data/lib/admin_core/resource_manager/buildable.rb +42 -0
- data/lib/admin_core/resource_manager/convert.rb +95 -0
- data/lib/admin_core/resource_manager/has_many_fields.rb +71 -0
- data/lib/admin_core/resource_manager/permission.rb +35 -0
- data/lib/admin_core/resource_manager/searchable.rb +57 -0
- data/lib/admin_core/resource_page/base.rb +17 -0
- data/lib/admin_core/resource_page/edit.rb +22 -0
- data/lib/admin_core/resource_page/index.rb +52 -0
- data/lib/admin_core/resource_page/new.rb +26 -0
- data/lib/admin_core/resource_page/show.rb +22 -0
- data/lib/admin_core/resource_router.rb +58 -0
- data/lib/admin_core/resource_search.rb +22 -0
- data/lib/admin_core/rspec/matchers.rb +18 -0
- data/lib/admin_core/rspec/resource_field_spec_helper.rb +54 -0
- data/lib/admin_core/version.rb +11 -0
- data/lib/admin_core/view_object/sidebar_dropdown.rb +21 -0
- data/lib/admin_core/view_object/sidebar_link.rb +24 -0
- data/lib/admin_core/view_object/sidebar_resource_link.rb +23 -0
- data/lib/admin_core/view_object/sidebar_title.rb +18 -0
- data/lib/admin_core.rb +69 -0
- data/lib/generators/admin_core/install_generator.rb +39 -0
- data/lib/generators/admin_core/resource_manager_generator.rb +166 -0
- data/lib/generators/admin_core/templates/admin-core.css +1 -0
- data/lib/generators/admin_core/templates/admin-core.js +38196 -0
- data/lib/generators/admin_core/templates/controller.rb.erb +4 -0
- data/lib/generators/admin_core/templates/initializer.rb.erb +3 -0
- data/lib/generators/admin_core/templates/resource_manager.rb.erb +33 -0
- data/lib/generators/admin_core/templates/view.html.erb +58 -0
- data/sample/.gitignore +21 -0
- data/sample/Gemfile +35 -0
- data/sample/Gemfile.lock +147 -0
- data/sample/README.md +24 -0
- data/sample/Rakefile +6 -0
- data/sample/app/assets/config/manifest.js +2 -0
- data/sample/app/assets/images/.keep +0 -0
- data/sample/app/assets/stylesheets/application.css +15 -0
- data/sample/app/controllers/admin/application_controller.rb +4 -0
- data/sample/app/controllers/admin/tweets_controller.rb +4 -0
- data/sample/app/controllers/admin/users_controller.rb +4 -0
- data/sample/app/controllers/application_controller.rb +3 -0
- data/sample/app/controllers/concerns/.keep +0 -0
- data/sample/app/helpers/application_helper.rb +2 -0
- data/sample/app/jobs/application_job.rb +2 -0
- data/sample/app/models/admin/tweet.rb +35 -0
- data/sample/app/models/admin/user.rb +41 -0
- data/sample/app/models/application_record.rb +3 -0
- data/sample/app/models/concerns/.keep +0 -0
- data/sample/app/models/tweet.rb +3 -0
- data/sample/app/models/user.rb +3 -0
- data/sample/app/views/admin/application.html.erb +64 -0
- data/sample/app/views/layouts/application.html.erb +13 -0
- data/sample/bin/bundle +3 -0
- data/sample/bin/rails +4 -0
- data/sample/bin/rake +4 -0
- data/sample/bin/setup +34 -0
- data/sample/bin/update +29 -0
- data/sample/config/application.rb +25 -0
- data/sample/config/boot.rb +3 -0
- data/sample/config/database.yml +25 -0
- data/sample/config/environment.rb +5 -0
- data/sample/config/environments/development.rb +42 -0
- data/sample/config/environments/production.rb +69 -0
- data/sample/config/environments/test.rb +36 -0
- data/sample/config/initializers/admin_core.rb +8 -0
- data/sample/config/initializers/application_controller_renderer.rb +6 -0
- data/sample/config/initializers/backtrace_silencers.rb +7 -0
- data/sample/config/initializers/cookies_serializer.rb +5 -0
- data/sample/config/initializers/filter_parameter_logging.rb +4 -0
- data/sample/config/initializers/inflections.rb +16 -0
- data/sample/config/initializers/mime_types.rb +4 -0
- data/sample/config/initializers/new_framework_defaults.rb +24 -0
- data/sample/config/initializers/session_store.rb +3 -0
- data/sample/config/initializers/wrap_parameters.rb +14 -0
- data/sample/config/locales/en.yml +23 -0
- data/sample/config/routes.rb +6 -0
- data/sample/config/secrets.yml +22 -0
- data/sample/config.ru +5 -0
- data/sample/db/migrate/20170417055257_create_users.rb +10 -0
- data/sample/db/migrate/20170417055412_create_tweets.rb +9 -0
- data/sample/db/schema.rb +31 -0
- data/sample/db/seeds.rb +7 -0
- data/sample/lib/assets/.keep +0 -0
- data/sample/lib/tasks/.keep +0 -0
- data/sample/log/.keep +0 -0
- data/sample/public/404.html +67 -0
- data/sample/public/422.html +67 -0
- data/sample/public/500.html +66 -0
- data/sample/public/apple-touch-icon-precomposed.png +0 -0
- data/sample/public/apple-touch-icon.png +0 -0
- data/sample/public/bundle.min.js +27 -0
- data/sample/public/bundle.min.js.map +1 -0
- data/sample/public/favicon.ico +0 -0
- data/sample/public/javascripts/admin-core.js +38196 -0
- data/sample/public/robots.txt +5 -0
- data/sample/public/stylesheets/admin-core.css +1 -0
- data/sample/tmp/.keep +0 -0
- data/sample/vendor/assets/stylesheets/.keep +0 -0
- metadata +368 -0
data/client/package.json
ADDED
@@ -0,0 +1,60 @@
|
|
1
|
+
{
|
2
|
+
"name": "admin-core-js",
|
3
|
+
"version": "0.0.1",
|
4
|
+
"engines": {
|
5
|
+
"node": "~6.10",
|
6
|
+
"npm": "~3.10",
|
7
|
+
"yarn": "~0.21"
|
8
|
+
},
|
9
|
+
"description": "Flexible admin framework for Rails",
|
10
|
+
"main": "lib/AdminCore.js",
|
11
|
+
"author": "Yuku Takahashi <yuku@qiita.com>",
|
12
|
+
"scripts": {
|
13
|
+
"build": "yarn run clean && run-p build:*",
|
14
|
+
"build:dist": "run-p build:dist:*",
|
15
|
+
"build:dist:js": "webpack && uglifyjs dist/admin-core.js -o dist/admin-core.min.js --source-map dist/admin-core.min.js.map --source-map-url admin-core.min.js.map",
|
16
|
+
"build:dist:css": "mkdir -p dist && node-sass --output-style compressed admin-core.scss > dist/admin-core.css",
|
17
|
+
"build:lib": "babel src -d lib -s && rm -fr lib/main.*",
|
18
|
+
"clean": "rm -fr dist lib",
|
19
|
+
"test": "flow"
|
20
|
+
},
|
21
|
+
"repository": {
|
22
|
+
"type": "git",
|
23
|
+
"url": "git+https://github.com/increments/admin_core.git"
|
24
|
+
},
|
25
|
+
"bugs": {
|
26
|
+
"url": "https://github.com/increments/admin_core/issues"
|
27
|
+
},
|
28
|
+
"homepage": "https://github.com/increments/admin_core#readme",
|
29
|
+
"dependencies": {
|
30
|
+
"axios": "^0.16.1",
|
31
|
+
"classnames": "^2.2.5",
|
32
|
+
"coreui.io": "^1.0.0-alpha.4",
|
33
|
+
"lodash.omit": "^4.5.0",
|
34
|
+
"lodash.reduce": "^4.6.0",
|
35
|
+
"lodash.topairs": "^4.3.0",
|
36
|
+
"react": "^15.5.4",
|
37
|
+
"react-addons-css-transition-group": "^15.5.2",
|
38
|
+
"react-addons-transition-group": "^15.5.2",
|
39
|
+
"react-dom": "^15.5.4",
|
40
|
+
"react-router-dom": "^4.1.1",
|
41
|
+
"reactstrap": "^4.5.0"
|
42
|
+
},
|
43
|
+
"devDependencies": {
|
44
|
+
"babel-cli": "^6.24.1",
|
45
|
+
"babel-core": "^6.24.1",
|
46
|
+
"babel-eslint": "^7.2.2",
|
47
|
+
"babel-loader": "^6.4.1",
|
48
|
+
"babel-preset-env": "^1.4.0",
|
49
|
+
"babel-preset-react": "^6.24.1",
|
50
|
+
"eslint": "^3.19.0",
|
51
|
+
"eslint-plugin-flowtype": "^2.30.4",
|
52
|
+
"eslint-plugin-react": "^6.10.3",
|
53
|
+
"flow-bin": "^0.44.0",
|
54
|
+
"flow-typed": "^2.0.0",
|
55
|
+
"node-sass": "^4.5.2",
|
56
|
+
"npm-run-all": "^4.0.2",
|
57
|
+
"uglify-js": "^2.8.22",
|
58
|
+
"webpack": "^2.4.1"
|
59
|
+
}
|
60
|
+
}
|
@@ -0,0 +1,23 @@
|
|
1
|
+
---
|
2
|
+
env:
|
3
|
+
browser: true
|
4
|
+
|
5
|
+
plugins:
|
6
|
+
- flowtype
|
7
|
+
- react
|
8
|
+
|
9
|
+
rules:
|
10
|
+
flowtype/boolean-style: error
|
11
|
+
flowtype/define-flow-type: error
|
12
|
+
flowtype/delimiter-dangle: [error, always]
|
13
|
+
flowtype/generic-spacing: [error, never]
|
14
|
+
flowtype/no-dupe-keys: error
|
15
|
+
flowtype/no-primitive-constructor-types: error
|
16
|
+
flowtype/object-type-delimiter: [error, semicolon]
|
17
|
+
flowtype/space-after-type-colon: error
|
18
|
+
flowtype/space-before-generic-bracket: [error, never]
|
19
|
+
flowtype/space-before-type-colon: [error, never]
|
20
|
+
flowtype/union-intersection-spacing: error
|
21
|
+
flowtype/use-flow-type: error
|
22
|
+
react/jsx-uses-react: error
|
23
|
+
react/jsx-uses-vars: error
|
@@ -0,0 +1,44 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {BrowserRouter, Route, Switch} from "react-router-dom";
|
4
|
+
|
5
|
+
import EditPage from "./resource-page/edit";
|
6
|
+
import Header from "./components/header";
|
7
|
+
import IndexPage from "./resource-page/index";
|
8
|
+
import NewPage from "./resource-page/new";
|
9
|
+
import ShowPage from "./resource-page/show";
|
10
|
+
import Sidebar from "./components/sidebar";
|
11
|
+
import type {ResourceManager, SidebarItem} from "./decls";
|
12
|
+
|
13
|
+
export default class App extends React.Component {
|
14
|
+
props: {
|
15
|
+
sidebar: SidebarItem[];
|
16
|
+
resourceManagers: ResourceManager[];
|
17
|
+
children?: any;
|
18
|
+
siteName: string;
|
19
|
+
}
|
20
|
+
|
21
|
+
render() {
|
22
|
+
return (
|
23
|
+
<BrowserRouter>
|
24
|
+
<div className="app">
|
25
|
+
<Header siteName={this.props.siteName} />
|
26
|
+
<div className="app-body">
|
27
|
+
<Sidebar items={this.props.sidebar} />
|
28
|
+
<Switch>
|
29
|
+
{this.props.children}
|
30
|
+
{this.props.resourceManagers.map((resourceManager, i) => [
|
31
|
+
<Route exact path={resourceManager.indexPath} component={IndexPage(resourceManager)} key={`index-${i}`} />,
|
32
|
+
resourceManager.newPath && <Route exact path={resourceManager.newPath} component={NewPage(resourceManager)} key={`new-${i}`} />,
|
33
|
+
<Route exact path={resourceManager.showPath} component={ShowPage(resourceManager)} key={`show-${i}`} />,
|
34
|
+
resourceManager.editPath && <Route exact path={resourceManager.editPath} component={EditPage(resourceManager)} key={`edit-${i}`} />,
|
35
|
+
].filter(route => route))}
|
36
|
+
</Switch>
|
37
|
+
</div>
|
38
|
+
</div>
|
39
|
+
</BrowserRouter>
|
40
|
+
);
|
41
|
+
}
|
42
|
+
}
|
43
|
+
|
44
|
+
App.displayName = "AdminCore";
|
@@ -0,0 +1,18 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {Link} from "react-router-dom";
|
4
|
+
|
5
|
+
const Breadcrumb = ({ links, current }: { links: string[][]; current: ?string; }) => (
|
6
|
+
<ol className="breadcrumb">
|
7
|
+
{links.map(([text, path], i) =>
|
8
|
+
<span className="breadcrumb-item" key={i}>
|
9
|
+
<Link to={path}>{text}</Link>
|
10
|
+
</span>
|
11
|
+
)}
|
12
|
+
<span className="breadcrumb-item">
|
13
|
+
{current || "..."}
|
14
|
+
</span>
|
15
|
+
</ol>
|
16
|
+
);
|
17
|
+
|
18
|
+
export default Breadcrumb;
|
@@ -0,0 +1,45 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {Link} from "react-router-dom";
|
4
|
+
|
5
|
+
export default class Header extends React.Component {
|
6
|
+
props: {
|
7
|
+
siteName: string;
|
8
|
+
}
|
9
|
+
|
10
|
+
sidebarToggle(e: SyntheticEvent) {
|
11
|
+
e.preventDefault();
|
12
|
+
this.toggle("sidebar-hidden");
|
13
|
+
}
|
14
|
+
|
15
|
+
mobileSidebarToggle(e: SyntheticEvent) {
|
16
|
+
e.preventDefault();
|
17
|
+
this.toggle("sidebar-mobile-show");
|
18
|
+
}
|
19
|
+
|
20
|
+
asideToggle(e: SyntheticEvent) {
|
21
|
+
e.preventDefault();
|
22
|
+
this.toggle("aside-menu-hidden");
|
23
|
+
}
|
24
|
+
|
25
|
+
toggle(name: string) {
|
26
|
+
const body = document.body;
|
27
|
+
if (body) {
|
28
|
+
body.classList.toggle(name);
|
29
|
+
}
|
30
|
+
}
|
31
|
+
|
32
|
+
render() {
|
33
|
+
return (
|
34
|
+
<header className="app-header navbar">
|
35
|
+
<button className="navbar-toggler mobile-sidebar-toggler hidden-lg-up" onClick={this.mobileSidebarToggle.bind(this)} type="button">☰</button>
|
36
|
+
<Link to="/" className="navbar-brand">{this.props.siteName}</Link>
|
37
|
+
<ul className="nav navbar-nav hidden-md-down">
|
38
|
+
<li className="nav-item">
|
39
|
+
<a className="nav-link navbar-toggler sidebar-toggler" onClick={this.sidebarToggle.bind(this)} href="#">☰</a>
|
40
|
+
</li>
|
41
|
+
</ul>
|
42
|
+
</header>
|
43
|
+
);
|
44
|
+
}
|
45
|
+
}
|
@@ -0,0 +1,72 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {Link} from "react-router-dom";
|
4
|
+
|
5
|
+
export default class Pagination extends React.Component {
|
6
|
+
props: {
|
7
|
+
total: number;
|
8
|
+
current: number;
|
9
|
+
location: {
|
10
|
+
pathname: string;
|
11
|
+
search: string;
|
12
|
+
hash: string;
|
13
|
+
};
|
14
|
+
}
|
15
|
+
|
16
|
+
link(page: number) {
|
17
|
+
const queries = this.props.location.search.substring(1).split("&")
|
18
|
+
.filter(query => query && !query.startsWith("page="));
|
19
|
+
return {
|
20
|
+
pathname: this.props.location.pathname,
|
21
|
+
search: `?${queries.concat([`page=${page}`]).join("&")}`,
|
22
|
+
hash: this.props.location.hash,
|
23
|
+
};
|
24
|
+
}
|
25
|
+
|
26
|
+
renderPrev() {
|
27
|
+
return (
|
28
|
+
this.props.current === 1 ?
|
29
|
+
<li className="page-item disabled">
|
30
|
+
<a className="page-link" href="#">‹</a>
|
31
|
+
</li>
|
32
|
+
:
|
33
|
+
<li className="page-item">
|
34
|
+
<Link to={this.link(this.props.current - 1)} className="page-link">‹</Link>
|
35
|
+
</li>
|
36
|
+
);
|
37
|
+
}
|
38
|
+
|
39
|
+
renderNext() {
|
40
|
+
return (
|
41
|
+
this.props.current === this.props.total ?
|
42
|
+
<li className="page-item disabled">
|
43
|
+
<a className="page-link" href="#">›</a>
|
44
|
+
</li>
|
45
|
+
:
|
46
|
+
<li className="page-item">
|
47
|
+
<Link to={this.link(this.props.total)} className="page-link">›</Link>
|
48
|
+
</li>
|
49
|
+
);
|
50
|
+
}
|
51
|
+
|
52
|
+
render() {
|
53
|
+
const pages = [];
|
54
|
+
for (let i = 1; i <= this.props.total; i++) { pages.push(i); }
|
55
|
+
return (
|
56
|
+
<ul className="pagination">
|
57
|
+
{this.renderPrev()}
|
58
|
+
{ pages.map(page =>
|
59
|
+
page === this.props.current ?
|
60
|
+
<li className="page-item active" key={page}>
|
61
|
+
<a className="page-link" href="#">{page}</a>
|
62
|
+
</li>
|
63
|
+
:
|
64
|
+
<li className="page-item" key={page}>
|
65
|
+
<Link to={this.link(page)} className="page-link">{page}</Link>
|
66
|
+
</li>
|
67
|
+
)}
|
68
|
+
{this.renderNext()}
|
69
|
+
</ul>
|
70
|
+
);
|
71
|
+
}
|
72
|
+
}
|
@@ -0,0 +1,87 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {Link} from "react-router-dom";
|
4
|
+
import type {Location} from "react-router-dom";
|
5
|
+
import {Card, CardBlock, CardHeader, CardFooter} from "reactstrap";
|
6
|
+
|
7
|
+
import type {ResourceFilter} from "../decls";
|
8
|
+
import {renderFilter} from "../resource-filter-renderer";
|
9
|
+
|
10
|
+
const REGEXP = /\Afilter\[[^]*]\]=/;
|
11
|
+
|
12
|
+
export default class ResourceFilters extends React.Component {
|
13
|
+
props: {
|
14
|
+
filters: ResourceFilter[];
|
15
|
+
location: Location;
|
16
|
+
};
|
17
|
+
|
18
|
+
state: {
|
19
|
+
[string]: { operator: string; value: string; };
|
20
|
+
};
|
21
|
+
|
22
|
+
constructor(props: any) {
|
23
|
+
super(props);
|
24
|
+
const state = {};
|
25
|
+
this.props.filters.forEach(filter => {
|
26
|
+
state[filter.name] = {
|
27
|
+
operator: filter.query.operator,
|
28
|
+
value: filter.query.value,
|
29
|
+
};
|
30
|
+
});
|
31
|
+
this.state = state;
|
32
|
+
}
|
33
|
+
|
34
|
+
handleChange(name: string, operator: string, value: string) {
|
35
|
+
const state = {};
|
36
|
+
state[name] = { operator, value };
|
37
|
+
this.setState(state);
|
38
|
+
}
|
39
|
+
|
40
|
+
renderApply() {
|
41
|
+
const queries = this.props.location.search.substring(1).split("&").filter(kv => REGEXP.test(kv));
|
42
|
+
Object.keys(this.state).forEach(name => {
|
43
|
+
if (this.state[name].value) {
|
44
|
+
queries.push(`filter[${name}:${this.state[name].operator}]=${this.state[name].value}`);
|
45
|
+
}
|
46
|
+
});
|
47
|
+
const search = queries.join("&");
|
48
|
+
return (
|
49
|
+
<Link
|
50
|
+
to={{
|
51
|
+
pathname: this.props.location.pathname,
|
52
|
+
search: search ? `?${search}` : "",
|
53
|
+
hash: this.props.location.hash,
|
54
|
+
}}
|
55
|
+
className="btn btn-primary"
|
56
|
+
>
|
57
|
+
Apply
|
58
|
+
</Link>
|
59
|
+
);
|
60
|
+
}
|
61
|
+
|
62
|
+
render() {
|
63
|
+
return (
|
64
|
+
<Card>
|
65
|
+
<CardHeader>
|
66
|
+
<i className="fa fa-filter"/>
|
67
|
+
Filters
|
68
|
+
</CardHeader>
|
69
|
+
<CardBlock>
|
70
|
+
{this.props.filters.map((filter, i) =>
|
71
|
+
<div className="row" key={i}>
|
72
|
+
<div className="col-md-3 col-lg-12">
|
73
|
+
{filter.displayName}
|
74
|
+
</div>
|
75
|
+
<div className="col-md-9 col-lg-12">
|
76
|
+
{renderFilter(filter, this.handleChange.bind(this))}
|
77
|
+
</div>
|
78
|
+
</div>
|
79
|
+
)}
|
80
|
+
</CardBlock>
|
81
|
+
<CardFooter>
|
82
|
+
{this.renderApply()}
|
83
|
+
</CardFooter>
|
84
|
+
</Card>
|
85
|
+
);
|
86
|
+
}
|
87
|
+
}
|
@@ -0,0 +1,103 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import classNames from "classnames";
|
4
|
+
import {Redirect} from "react-router-dom";
|
5
|
+
import type {RouterHistory} from "react-router-dom";
|
6
|
+
|
7
|
+
import httpClient from "../http-client";
|
8
|
+
import type {Resource, ResourceField} from "../decls";
|
9
|
+
import {getValue, renderNew, renderEdit} from "../resource-field-renderer";
|
10
|
+
|
11
|
+
export default class ResourceForm extends React.Component {
|
12
|
+
props: {
|
13
|
+
action: string;
|
14
|
+
resource: Resource;
|
15
|
+
history: RouterHistory;
|
16
|
+
};
|
17
|
+
|
18
|
+
state: {
|
19
|
+
redirectTo?: string;
|
20
|
+
errors: {
|
21
|
+
[string]: string[];
|
22
|
+
};
|
23
|
+
values: {
|
24
|
+
[string]: any;
|
25
|
+
};
|
26
|
+
}
|
27
|
+
|
28
|
+
constructor(props: any) {
|
29
|
+
super(props);
|
30
|
+
this.state = {
|
31
|
+
errors: {},
|
32
|
+
values: {},
|
33
|
+
};
|
34
|
+
props.resource.fields.forEach(field => {
|
35
|
+
this.state.values[field.name] = getValue(field);
|
36
|
+
});
|
37
|
+
}
|
38
|
+
|
39
|
+
request(data: any) {
|
40
|
+
return this.props.resource.showPath ? httpClient.put(this.props.action, data) : httpClient.post(this.props.action, data);
|
41
|
+
}
|
42
|
+
|
43
|
+
handleSubmit(e: SyntheticEvent) {
|
44
|
+
e.preventDefault();
|
45
|
+
const data = {};
|
46
|
+
data[this.props.resource.name] = this.state.values;
|
47
|
+
this.request(data)
|
48
|
+
.then(r => this.setState({ redirectTo: r.data.redirectTo }))
|
49
|
+
.catch(e => this.setState({ errors: e.response.data.errors }));
|
50
|
+
}
|
51
|
+
|
52
|
+
handleChange(name: string, value: any) {
|
53
|
+
const values = this.state.values;
|
54
|
+
values[name] = value;
|
55
|
+
this.setState({ values });
|
56
|
+
}
|
57
|
+
|
58
|
+
handleClickCancel() {
|
59
|
+
this.props.history.goBack();
|
60
|
+
}
|
61
|
+
|
62
|
+
shouldComponentUpdate(_: any, prevState: $PropertyType<ResourceForm, "state">) {
|
63
|
+
return this.state.errors !== prevState.errors ||
|
64
|
+
this.state.redirectTo !== prevState.redirectTo;
|
65
|
+
}
|
66
|
+
|
67
|
+
renderField(field: ResourceField) {
|
68
|
+
const renderer = this.props.resource.showPath ? renderEdit : renderNew;
|
69
|
+
return renderer(field, this.handleChange.bind(this));
|
70
|
+
}
|
71
|
+
|
72
|
+
render() {
|
73
|
+
if (this.state.redirectTo) {
|
74
|
+
return <Redirect to={this.state.redirectTo} />;
|
75
|
+
}
|
76
|
+
return (
|
77
|
+
<form onSubmit={this.handleSubmit.bind(this)} className="form-horizontal">
|
78
|
+
{this.props.resource.fields.map((field, i) =>
|
79
|
+
<div
|
80
|
+
className={classNames("form-group", "row", { "has-danger": this.state.errors[field.name] })}
|
81
|
+
key={i}
|
82
|
+
>
|
83
|
+
<label className="col-md-3 form-control-label" htmlFor={field.name}>
|
84
|
+
{field.displayName}
|
85
|
+
</label>
|
86
|
+
<div className="col-md-9">
|
87
|
+
{this.renderField(field)}
|
88
|
+
{this.state.errors[field.name] && this.state.errors[field.name].map((error, i) =>
|
89
|
+
<div className="form-control-feedback" key={i}>
|
90
|
+
{error}
|
91
|
+
</div>
|
92
|
+
)}
|
93
|
+
</div>
|
94
|
+
</div>
|
95
|
+
)}
|
96
|
+
<button type="submit" className="btn btn-primary">Submit</button>
|
97
|
+
<span className="btn btn-secondary" onClick={this.handleClickCancel.bind(this)}>
|
98
|
+
Cancel
|
99
|
+
</span>
|
100
|
+
</form>
|
101
|
+
);
|
102
|
+
}
|
103
|
+
}
|
@@ -0,0 +1,41 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {Link} from "react-router-dom";
|
4
|
+
|
5
|
+
import type {Resource} from "../decls";
|
6
|
+
import {renderIndex} from "../resource-field-renderer";
|
7
|
+
|
8
|
+
export default class ResourcesCollection extends React.Component {
|
9
|
+
props: {
|
10
|
+
attributes: string[];
|
11
|
+
resources: Resource[];
|
12
|
+
}
|
13
|
+
|
14
|
+
render() {
|
15
|
+
return (
|
16
|
+
<table className="table table-bordered table-hover table-sm">
|
17
|
+
<thead>
|
18
|
+
<tr>
|
19
|
+
{this.props.attributes.map((attribute, i) =>
|
20
|
+
<th key={i}>{attribute}</th>
|
21
|
+
)}
|
22
|
+
</tr>
|
23
|
+
</thead>
|
24
|
+
<tbody>
|
25
|
+
{this.props.resources.map((resource, i) =>
|
26
|
+
<tr key={i}>
|
27
|
+
{resource.fields.map((field, j) => {
|
28
|
+
const showPath = resource.showPath;
|
29
|
+
if (j === 0 && showPath) {
|
30
|
+
return <td key={j}><Link to={showPath}>{renderIndex(field)}</Link></td>;
|
31
|
+
} else {
|
32
|
+
return <td key={j}>{renderIndex(field)}</td>;
|
33
|
+
}
|
34
|
+
})}
|
35
|
+
</tr>
|
36
|
+
)}
|
37
|
+
</tbody>
|
38
|
+
</table>
|
39
|
+
);
|
40
|
+
}
|
41
|
+
}
|
@@ -0,0 +1,90 @@
|
|
1
|
+
// @flow
|
2
|
+
import React from "react";
|
3
|
+
import {NavLink} from "react-router-dom";
|
4
|
+
|
5
|
+
import type {
|
6
|
+
SidebarDropdown,
|
7
|
+
SidebarItem,
|
8
|
+
SidebarLink,
|
9
|
+
SidebarTitle,
|
10
|
+
} from "../decls";
|
11
|
+
|
12
|
+
export default class Sidebar extends React.Component {
|
13
|
+
props: {
|
14
|
+
items: SidebarItem[];
|
15
|
+
}
|
16
|
+
|
17
|
+
handleClick(e: SyntheticEvent) {
|
18
|
+
e.preventDefault();
|
19
|
+
const target = e.target;
|
20
|
+
if (target instanceof HTMLElement) {
|
21
|
+
const parent = target.parentElement;
|
22
|
+
if (parent) {
|
23
|
+
parent.classList.toggle("open");
|
24
|
+
}
|
25
|
+
}
|
26
|
+
}
|
27
|
+
|
28
|
+
renderSidebarItem(item: SidebarItem, key: number) {
|
29
|
+
if (item.type === 'title') {
|
30
|
+
return this.renderSidebarTitle(item, key);
|
31
|
+
} else if (item.type === 'link') {
|
32
|
+
return this.renderSidebarLink(item, key);
|
33
|
+
} else if (item.type === 'dropdown') {
|
34
|
+
return this.renderSidebarDropdown(item, key);
|
35
|
+
}
|
36
|
+
}
|
37
|
+
|
38
|
+
renderSidebarTitle(sidebarTitle: SidebarTitle, key: number) {
|
39
|
+
return (
|
40
|
+
<li className="nav-title" key={key}>
|
41
|
+
{sidebarTitle.displayName}
|
42
|
+
</li>
|
43
|
+
);
|
44
|
+
}
|
45
|
+
|
46
|
+
renderSidebarDropdown(sidebarDropdown: SidebarDropdown, key: number) {
|
47
|
+
return (
|
48
|
+
<li className="nav-item nav-dropdown" key={key}>
|
49
|
+
<a className="nav-link nav-dropdown-toggle" href="#" onClick={this.handleClick}>
|
50
|
+
{sidebarDropdown.displayName}
|
51
|
+
</a>
|
52
|
+
<ul className="nav-dropdown-items">
|
53
|
+
{sidebarDropdown.links.map((link, i) =>
|
54
|
+
this.renderSidebarLink(link, i)
|
55
|
+
)}
|
56
|
+
</ul>
|
57
|
+
</li>
|
58
|
+
)
|
59
|
+
}
|
60
|
+
|
61
|
+
renderSidebarLink(link: SidebarLink, key: number) {
|
62
|
+
return (
|
63
|
+
<li className="nav-item" key={key}>
|
64
|
+
{link.external ?
|
65
|
+
<a href={link.link} className="nav-link">
|
66
|
+
{link.displayName}
|
67
|
+
</a>
|
68
|
+
:
|
69
|
+
<NavLink to={link.link} className="nav-link" activeClassName="active">
|
70
|
+
{link.displayName}
|
71
|
+
</NavLink>
|
72
|
+
}
|
73
|
+
</li>
|
74
|
+
);
|
75
|
+
}
|
76
|
+
|
77
|
+
render() {
|
78
|
+
return (
|
79
|
+
<div className="sidebar">
|
80
|
+
<nav className="sidebar-nav">
|
81
|
+
<ul className="nav">
|
82
|
+
{this.props.items.map((item, i) =>
|
83
|
+
this.renderSidebarItem(item, i)
|
84
|
+
)}
|
85
|
+
</ul>
|
86
|
+
</nav>
|
87
|
+
</div>
|
88
|
+
);
|
89
|
+
}
|
90
|
+
}
|