12/08/2018, 17:53

Tạo admin page với admin-on-rest

Việc xây dựng trang admin có nhiều framework khác nhau có thể hỗ trợ, việc sử dụng framawork sẽ tiết kiệm được công sức và thời gian. Admin-on-rest là một framwork được xây dựng trên nguồn cảm hứng của ng-admin Admin-on-Rest cho phép xây dựng trang admin trên REST service sử dụng ES6, React và ...

Việc xây dựng trang admin có nhiều framework khác nhau có thể hỗ trợ, việc sử dụng framawork sẽ tiết kiệm được công sức và thời gian.

Admin-on-rest là một framwork được xây dựng trên nguồn cảm hứng của ng-admin

Admin-on-Rest cho phép xây dựng trang admin trên REST service sử dụng ES6, React và giao diện material design

  • Đáp ứng được với mọi REST backend
  • Tối ưu hoá việc render
  • Hỗ trợ quan hệ (many to one, one to many)
  • Internationalization (i18n)
  • Full-featured Datagrid (sort, pagination, filters)
  • Connect tới nhiều backend
  • Dễ dàng custom
  • …..

Ta sẽ sử dụng create-react-app của facebook để tạo ra một react app và cài đặt admin-on-rest

Tôi sẽ viết chung phần back-end và front này trong cùng project

Đầu tiên tạo ứng dụng rails

rails new vk_shop
npm install -g create-react-app
create-react-app vk_shop_client
cd vk_shop_client
yarn add admin-on-rest
yarn start

Với cấu trúc thư mục như sau

Chú ý là khi chạy yarn start thì client của bạn sẽ chạy cổng 3000 nên khi chạy server bạn phải chọn port khác, và tôi chọn 3001

rails s -p 3001

Chú ý vì đây là gọi cors domain nên phía server phải cài đặt thêm gem

gem 'rack-cors', require: 'rack/cors'
#config/application.rb
        expose: [
          'X-Total-Count',
        ]

Vì response header của admin-on-rest phải xử dụng ‘X-Total-Count’ để cho việc phân trang, nên dữ liệu trả về từ list page phải response header này

<Admin> component tạo ra một ứng dụng với state, routing và controller login Trong một <Admin> component cần phải có prop restClient và ít nhất một <Resorce> component bên trong

restClient

Đây lầ một prop bắt buộc phải có của admin component, nó phải là một function trả về promise

const restClient = (type, resource, params) => new Promise();

Bạn cũng có thể sửa header, authentication từ restClient

theme

Bạn có thể tuỳ biến sử dụng fonts, colors space, cho theme của mình. Và sử dụng theme bằng props theme

import darkBaseTheme from 'material-ui/styles/baseThemes/darkBaseTheme';
import getMuiTheme from 'material-ui/styles/getMuiTheme';

const App = () => (
    <Admin theme={getMuiTheme(darkBaseTheme)} restClient={simpleRestClient('http://path.to.my.api')}>
        // ...
    </Admin>
);

Ngoài ra thì còn rất nhiều các props khác của admin component:

  • title
  • dashboard
  • catchAll
  • menu
  • theme
  • appLayout
  • customReducers
  • customSagas
  • customRoutes
  • authClient
  • loginPage
  • logoutButton
  • locale
  • messages
  • initialState
  • history

https://marmelab.com/admin-on-rest/Admin.html

<Resouce> component sẽ map tương ứng với api CRUD

Ví dụ resouce name là orders http://localhost:3001/orders

// in src/App.js
import React from 'react';
import { jsonServerRestClient, Admin, Resource } from 'admin-on-rest';
import { OrderList } from './orders.js;

const App = () => (
    <Admin restClient={jsonServerRestClient('http://localhost:3001')}>
        <Resource name="orders" list={OrderList} />
    </Admin>
);

<Resource> cho phép định nghĩa một component cho mỗi CRUD, thông qua prop names

  • list
  • create
  • edit
  • show
  • remove
<Resource name="orders" list={OrderList} create={OrderCreate}  edit={OrderEdit}/>

Name

admin-on-rest sử dụng name với 2 mục đích là để xác định api endpoint và để hình thành url cho resource

Ví dụ

        <Resource name="orders" list={OrderList} create={OrderCreate}  edit={OrderEdit}/>

Resource này sẽ lấy từ api http://localhost:3001/orders với các url tương ứng

/orders/ maps to OrderList /orders/create maps to OrderCreate /orders/:id maps to OrderEdit /orders/:id/show maps to OrderShow /orders/:id/delete maps to OrderRemove Nếu bạn muốn sử dụng một endpoint nào đó mà không muốn sửa url của routes thì chỉ cần mapping resource name orders tới api endpoind bằng restClient

Example

//App.js
import React from 'react';
import { jsonServerRestClient, Admin, Resource , Delete} from 'admin-on-rest';
import { StaffList, StaffCreate, StaffEdit } from './staffs';
import { OrderList, OrderCreate, OrderEdit } from './orders';
import authClient from './authClient';

import darkBaseTheme from 'material-ui/styles/baseThemes/darkBaseTheme';
import getMuiTheme from 'material-ui/styles/getMuiTheme';



const App = () => (
    <Admin theme={getMuiTheme(darkBaseTheme)}  restClient={jsonServerRestClient('http://localhost:3001')} authClient={authClient}>
        <Resource name="orders" list={OrderList} create={OrderCreate}  edit={OrderEdit}/>
        <Resource name="staffs" list={StaffList} create={StaffCreate}   />
    </Admin>
);

export default App;

//orders.js
// in src/staffs.js
import React from 'react';
import {NumberInput,DateInput, NumberField, DateField, List, Edit, Create, Datagrid, ReferenceField, TextField, EditButton, DisabledInput, LongTextInput, ReferenceInput, required, SelectInput, SimpleForm, TextInput, EmailInput } from 'admin-on-rest';


export const OrderList = (props) => (
    <List title="All Orders" {...props}>
        <Datagrid>
            <TextField source="name" />
            <TextField source="fb_url" />
            <DateField source="order_time" showTime/>
            <TextField source="phone" />
            <TextField source="address" />
            <TextField source="size" />
            <TextField source="product_id" />
            <NumberField source="original_price" />
            <NumberField source="shipping_fee" />
            <NumberField source="total_price" />
            <TextField source="shipping_code" />
            <TextField source="shipping_method" />
            <TextField source="order_status" />
            <TextField source="user_order_info" />
            <TextField source="note" />
            <TextField source="staff_id" />
            <EditButton />
        </Datagrid>
    </List>
);

export const OrderCreate = (props) => (
    <Create {...props}>
        <SimpleForm>
          <TextInput source="name" label="Name" validate={required} />
          <TextInput source="fb_url" validate={required} />
          <DateInput source="order_time" options={{

          }}/>
          <TextInput source="phone" validate={required} />
          <TextInput source="address" validate={required} />
          <TextInput source="size" />
          <TextInput source="product_id" />
          <NumberInput source="original_price" />
          <NumberInput source="shipping_fee" />
          <NumberInput source="total_price" />
          <TextInput source="shipping_code" />
          <TextInput source="shipping_method" />
          <TextInput source="order_status" />
          <TextInput source="user_order_info" />
          <LongTextInput source="note" />
          <TextInput source="staff_id" />
        </SimpleForm>
    </Create>
);

const OrderTitle = ({ record }) => {
    return <span>Order {record ? `"${record.name}"` : '}</span>;
};

export const OrderEdit = (props) => (
  <Edit title={<OrderTitle />}  {...props}>
    <SimpleForm>
      <DisabledInput label="Id" source="id" />
      <TextInput source="name" label="Name" validate={required} />
      <TextInput source="fb_url" validate={required} />
      <DateInput source="order_time" options={{

      }}/>
      <TextInput source="phone" validate={required} />
      <TextInput source="address" validate={required} />
      <TextInput source="size" />
      <TextInput source="product_id" />
      <NumberInput source="original_price" />
      <NumberInput source="shipping_fee" />
      <NumberInput source="total_price" />
      <TextInput source="shipping_code" />
      <TextInput source="shipping_method" />
      <TextInput source="order_status" />
      <TextInput source="user_order_info" />
      <LongTextInput source="note" />
      <TextInput source="staff_id" />
    </SimpleForm>
  </Edit>
);

Một số chú ý về phần server-side trong action index response header phải trả về X-Total-count

  def index
    response.headers["X-Total-Count"] = Order.all.size
    render json: Order.all
    # data trả về phải là mảng
  end

Detail: https://github.com/khanhhd/vk_shop

Nguồn

Bài viết xây dựng admin SPAs dựa trên nguồn https://marmelab.com/admin-on-rest/

0