Install bit secara global sebagai component sharing tool
https://docs.bit.dev/docs/installation
npm install bit-bin --global
or
yarn global add bit-bin
File dengan extensi jsx > layout File dengan nama index.js > logic Entah kenapa key nya harus field, ngga bisa label T_T
Crud application
- Basic CRUD - ALL working properly
- Update - see the known bugs at the end of this readme
- Design a better and comprehensive system
- Using NestJs / Or at this point, probably GraphQL at the backend
- Better component composition and make use of reducer more
- Exporting to html / excel
- No testing
- Clone this repo
- CD into the folder just created, and run yarn
- For the development, run yarn dev
- For the API Server, CD into api-server folder, and run yarn
- Run yarn server
- Configure database as defined in my-sql.js in config folder
- A sql file is available for testing
- To Login as a Administrator, use username: eko.andri@icloud.com, password: 111
- To Login as a Manager, use username: adi.sopian@gmail.com, password: 111
- To Login as a User, use username: eko.andri@gmail.com, password: 111
- After creating a new user by registering, notice that you can not use it immediately, a adminstrator account must be use to activate it.
===============================================================
- ReactJs
- NextJs: React Framework for file routing and server side rendering
- Ant Design: for styling (CSS)
- Express: Node Framework for serving Restful API
I am using some features of the NextJs Framework:
- Built-in File Name Routing: tldr; the file name is the routing as you type and see in the address bar, for instance, index.js is a homepage (a root domain, "/"), while about.js is translated to "/about" and contact.js to "/contact"
- Exporting to a static html, so it's partially SSR
- Working Hour per Day: 8
- Not set to any user group, must contact admin to do so
- Not activated, must contact admin to be activated
There are 3 Context used in this App
- UserContext: as a global user data after a user logged in
- MenuContext: a a global dispatcher to change Menu and the content of the Dashboard. The Content of the Dasboard is called "Modules" and they are placed inside a modules folder
- LoginContext: a local context just to pass down a single "logout" props
I will start explain the structure of this application from a root folder of this application, and just for the importance files and folders.
Files in root folder:
- next.config.js: next configuration for defining styling, folder aliasing and files exporting to static html
- .babelrc: babel configuration for defining Ant Design
Folder "pages"
- _app : is called first to define the initial state, which one of it is login state, which is false at first
- index.js : usually, this file is called first in the Next File System, but since we now have _app, this file is rendered second as a home page ("/"). Homepage will be used as Landing Page
- contact.js: Will be used for Contact Page
- about.js: Will be used for About Page
- login.js: Will be used for Login Page
- register.js: Will be used for Register Page
- dashboard.js: Will be used for Content Page. This is the page that will be loaded when the user successfully logged in. And since the state of the user can not be exposed, I decided to render the content page as a "React Component" instead of "file-routing", so there's no way that a user can access the content of the page by typing "/content" or "/user-profile" and etc in the address bar
For API Server Added index in table login for field: user_name Added index in table users for field: user_name
Added Protection in the backend to prevent unactivated user log in "Your Profile is not activated, please contact Admin!"
- Custom Modal
import { useReducer } from "react";
import { Modal, useModal } from "components";
function YourComponent() {
const [modal, dispatchModal] = useModal();
/**
* When you call modal and want to send some attributs
* this is inside a modal reducer
* const { status, message } = results;
* this result is coming from the backend
*
* isModalVisible: true,
* modalTitle: status, // This could be Warning, Success, whatever
* modalMessage: message // Message is the body of the modal
*
*/
dispatchModal({ type: "success", results });
/**
* If you only want to display Modal, don't use the "success" type
* as it requires results
*/
dispatchModal({ type: "modal-show" });
/**
* Call this when you want to close the modal
* Currently, pressing ok doesn't do additional function
*/
dispatchModal({ type: "modal-ok" });
dispatchModal({ type: "modal-cancel" });
/**
* Calling Modal to respon from a server, use it like this
*/
<Modal modal={modal} dispatchModal={dispatchModal} />;
/**
* If you want to render other component, do it like this
*/
<Modal modal={modal} dispatchModal={dispatchModal}>
<YourOtherComponent />
</Modal>;
}
-
Custom Sider Menu. Menu is now stateless, which now only received state from modules. Modules are what I called "pages" that rendered in the dashboard. Example of these pages are: User List, Report Table, User Profile, User Preference, Change Password and so on. This makes the App highly flexible, where you can just add or remove "pages" easily from an array. Example is below:
Pastikan jika memiliki 2 pages default, biasanya homepage (sebelum login) dan dashboard (sesudah login), keduanya memiliki 'key' yang sama, agar pada saat load pertama kali, baik itu homepage atau dashboard dapat langsung diload tanpa perlu melakukan klik pada menu nya
const subMenu = [
{
key: "sub-menu-1",
title: (
<span>
<Icon type="user" />
Your Dashboard
</span>
),
children: [
{
key: "default",
title: "Main Dashboard",
component: <Default />
},
{
key: "user-list",
title: "User List"
}
]
},
{
key: "sub-menu-2",
title: (
<span>
<Icon type="laptop" />
Tambahan
</span>
),
children: [
{
key: "20",
title: "Main Dashboard"
}
]
},
{
key: "sub-menu-3",
title: (
<span>
<Icon type="notification" />
User Profile
</span>
),
children: [
{
key: "profile",
title: "Profile",
component: <Profile />
},
{
key: "preference",
title: "Preference",
component: <Preference />
},
{
key: "change-password",
title: "Change Password"
}
]
}
];
- To use Ant Design Editable Table, make sure you have unique key in your data. Since I don't have key column, and I can't alias a column table name to "key", MySQL won't let me, instead I alias it with indx. And, you pass it through a rowKey props as indx. Find item.key or record.key and replace it with item.indx or record.indx
- Updating task do not update the table immediately, have to refresh or login/logout