mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-18 05:40:31 +00:00
Compare commits
314 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cfc625edf9 | ||
|
|
e1b6f0d879 | ||
|
|
ecda9296b8 | ||
|
|
aba732724b | ||
|
|
40bc7d4e99 | ||
|
|
7f1844aa6b | ||
|
|
7ab7456d08 | ||
|
|
de53a24d58 | ||
|
|
e0c565388a | ||
|
|
3c73540b4e | ||
|
|
a4a2d0c888 | ||
|
|
e9797fd9a0 | ||
|
|
40ae1aeb52 | ||
|
|
b2c08d5645 | ||
|
|
ee6f4028eb | ||
|
|
cdb8ea6721 | ||
|
|
0173630e80 | ||
|
|
4403bf2b06 | ||
|
|
f8e44c4f4e | ||
|
|
0bd11419bb | ||
|
|
c1e229814f | ||
|
|
871dcb62c4 | ||
|
|
5bf591d8c9 | ||
|
|
75b1cce380 | ||
|
|
119e6fa216 | ||
|
|
6b98e1f1c2 | ||
|
|
fa23aa67d3 | ||
|
|
9474a4fcd1 | ||
|
|
81c81bd09f | ||
|
|
1492c24cfb | ||
|
|
b4e9ee7ff3 | ||
|
|
e97bf56adb | ||
|
|
11d44c128b | ||
|
|
e615f62ef5 | ||
|
|
c69d29fb22 | ||
|
|
1e472d278a | ||
|
|
1688f49bc1 | ||
|
|
9182e13c07 | ||
|
|
f2150a982a | ||
|
|
d8efb17651 | ||
|
|
1e361dac66 | ||
|
|
a6cd483eb9 | ||
|
|
765f00d203 | ||
|
|
088d7d4b1d | ||
|
|
92c9001c52 | ||
|
|
93bd2af1d7 | ||
|
|
f261555b86 | ||
|
|
62ab31efd2 | ||
|
|
96cb3177aa | ||
|
|
d5f7cdf131 | ||
|
|
6ad3065f1f | ||
|
|
0860db87fb | ||
|
|
2707415319 | ||
|
|
99716e99a9 | ||
|
|
d1a67ca2ed | ||
|
|
a1ed9bf4da | ||
|
|
99df627633 | ||
|
|
e1646e92a8 | ||
|
|
d4138fdf4e | ||
|
|
49b6deed6d | ||
|
|
0dddf3d1d0 | ||
|
|
d5acb3696e | ||
|
|
2a9c1a9833 | ||
|
|
80d86b01de | ||
|
|
4a42fa4edc | ||
|
|
8ea92513fe | ||
|
|
a075b998d3 | ||
|
|
93aa60124b | ||
|
|
f9a7306e47 | ||
|
|
10f0d47b54 | ||
|
|
1d505db7bf | ||
|
|
6975ebb9e7 | ||
|
|
5defb5a279 | ||
|
|
1497a27e7d | ||
|
|
af9f4c45fc | ||
|
|
d719ac60bd | ||
|
|
789c8db693 | ||
|
|
7eea202f9d | ||
|
|
5b581b86f4 | ||
|
|
1ad2117c37 | ||
|
|
b1afa88c16 | ||
|
|
4b7af1c634 | ||
|
|
29a4ab7590 | ||
|
|
7150f12a01 | ||
|
|
934b0abbe6 | ||
|
|
b36468ff7f | ||
|
|
10c421dea0 | ||
|
|
661cbaf11e | ||
|
|
1e5b394575 | ||
|
|
dd7b44eb29 | ||
|
|
a456b9d942 | ||
|
|
5429643db5 | ||
|
|
13310b1aac | ||
|
|
7987f68aa3 | ||
|
|
1d159c2757 | ||
|
|
588995e759 | ||
|
|
e2349f1951 | ||
|
|
e80901c7a6 | ||
|
|
c87a98a5e4 | ||
|
|
da3564d315 | ||
|
|
0d924464ea | ||
|
|
961082c50b | ||
|
|
828c33bee1 | ||
|
|
1540a20a6e | ||
|
|
2a37497648 | ||
|
|
dbe067a27c | ||
|
|
3511348d04 | ||
|
|
ace3fdc569 | ||
|
|
a0098382e7 | ||
|
|
a953236bff | ||
|
|
4dda2a37aa | ||
|
|
0e79dceae8 | ||
|
|
0ca0a2ee0b | ||
|
|
7095903653 | ||
|
|
30841d57dc | ||
|
|
d0f889850c | ||
|
|
f9fe3506c5 | ||
|
|
8390e5ea6b | ||
|
|
fcd1a8849d | ||
|
|
9bd047917d | ||
|
|
2108666f3a | ||
|
|
0f10b71604 | ||
|
|
09e739a5a9 | ||
|
|
a7a94ba201 | ||
|
|
3466828b65 | ||
|
|
cfed874182 | ||
|
|
c688b8700e | ||
|
|
7a608d2ee3 | ||
|
|
33b998cbda | ||
|
|
d7da0ad24e | ||
|
|
92cff82184 | ||
|
|
7cd2b1c533 | ||
|
|
29a34f826a | ||
|
|
bd97a73c65 | ||
|
|
eebe4c46fa | ||
|
|
6aba694518 | ||
|
|
ae7d37a0e0 | ||
|
|
897e674912 | ||
|
|
a52f00eeb2 | ||
|
|
0ee9b54a9b | ||
|
|
dd8bb9cb18 | ||
|
|
86762d2dc8 | ||
|
|
c124002605 | ||
|
|
8e18d4a0dc | ||
|
|
21e075b479 | ||
|
|
51dfee699f | ||
|
|
a6a10ef6b8 | ||
|
|
fba46e2479 | ||
|
|
c4650f5d31 | ||
|
|
a441f78e61 | ||
|
|
280d991567 | ||
|
|
62e5db86a5 | ||
|
|
4fe4178090 | ||
|
|
dfc0fbbb71 | ||
|
|
295fff4746 | ||
|
|
c671fcd011 | ||
|
|
a6141d9f73 | ||
|
|
91d1dddfed | ||
|
|
ca746df86d | ||
|
|
02330c84b1 | ||
|
|
c89b8b131d | ||
|
|
3690c88c04 | ||
|
|
6573f19ff8 | ||
|
|
a948dd4236 | ||
|
|
56b1c36c9d | ||
|
|
bc7ec834c0 | ||
|
|
5963d14fdd | ||
|
|
80b8083059 | ||
|
|
ecaf23d269 | ||
|
|
14f33c667b | ||
|
|
78c42acb17 | ||
|
|
192bcdc696 | ||
|
|
ac99a6ca75 | ||
|
|
2a48d9be51 | ||
|
|
ab48e6092a | ||
|
|
bf99bda616 | ||
|
|
e1a3f471cb | ||
|
|
07e52bef7a | ||
|
|
d3c5131020 | ||
|
|
58a0ed4ecd | ||
|
|
26cf1d20f3 | ||
|
|
0fedd2479d | ||
|
|
ba585271b0 | ||
|
|
0c4259a05b | ||
|
|
31e0bbc187 | ||
|
|
43b91503d2 | ||
|
|
432e2d202a | ||
|
|
eeb16f4362 | ||
|
|
15fa1729a2 | ||
|
|
119d0b2839 | ||
|
|
0a9798e7a7 | ||
|
|
008faaece6 | ||
|
|
27916585a5 | ||
|
|
346696f673 | ||
|
|
68227b81e8 | ||
|
|
0c806366cd | ||
|
|
8b28d6894f | ||
|
|
3db00f6f70 | ||
|
|
56ab0a68e2 | ||
|
|
4ac32b3aad | ||
|
|
f98b429fcc | ||
|
|
012f204c5c | ||
|
|
7c62466c5b | ||
|
|
111ade2ac8 | ||
|
|
408e3cbb0b | ||
|
|
9cc770f168 | ||
|
|
383a9aad3b | ||
|
|
1be30fd142 | ||
|
|
63cb3f9fef | ||
|
|
ca3ff3fd8f | ||
|
|
313d0f3d0f | ||
|
|
64bf223458 | ||
|
|
97c421e2f1 | ||
|
|
ccad55dd4a | ||
|
|
a21d70a59d | ||
|
|
c2ccb7f879 | ||
|
|
fe9ca215ab | ||
|
|
c14b35356b | ||
|
|
6fd8a24802 | ||
|
|
80531b7fdb | ||
|
|
600a835dad | ||
|
|
3dff8763d4 | ||
|
|
e197d66d9f | ||
|
|
5dfb592ecc | ||
|
|
2630e0235d | ||
|
|
8b4dfe4ded | ||
|
|
9ceee6d02e | ||
|
|
553334f063 | ||
|
|
aef8eb7907 | ||
|
|
cc1f4cc26b | ||
|
|
719302b241 | ||
|
|
3db52e9c63 | ||
|
|
7393d68b7a | ||
|
|
6ec86d3cf7 | ||
|
|
1cba4b5f18 | ||
|
|
3a8e1f5238 | ||
|
|
371e374dc5 | ||
|
|
c2650c76e8 | ||
|
|
fc74346695 | ||
|
|
fca4dedeac | ||
|
|
e5d02043ad | ||
|
|
51fde0cc31 | ||
|
|
afee2e90e0 | ||
|
|
802f7cc442 | ||
|
|
d45005d8c2 | ||
|
|
15e7f34879 | ||
|
|
a54ddf27c7 | ||
|
|
955ae97c19 | ||
|
|
ddbadb67c8 | ||
|
|
b853eb1e75 | ||
|
|
c139e129bf | ||
|
|
3d3827b683 | ||
|
|
cf6d8d6038 | ||
|
|
d12b965bac | ||
|
|
5f0700b5e5 | ||
|
|
48348da663 | ||
|
|
fe8f41f200 | ||
|
|
0ad5a9ed03 | ||
|
|
bd282acae4 | ||
|
|
377fb07c70 | ||
|
|
b32abc0417 | ||
|
|
11d7029568 | ||
|
|
1990ce7562 | ||
|
|
b6f0f6c2d5 | ||
|
|
4c58e49169 | ||
|
|
376a16fd65 | ||
|
|
918cd4aef3 | ||
|
|
ec844637c3 | ||
|
|
5803760c61 | ||
|
|
2e34df5d63 | ||
|
|
35d755e417 | ||
|
|
66641ca56e | ||
|
|
307aaf0aa4 | ||
|
|
eb5a82d413 | ||
|
|
ce9169b24d | ||
|
|
22069f4795 | ||
|
|
567b4da7e9 | ||
|
|
06345a5615 | ||
|
|
6b8178f643 | ||
|
|
449ff724e1 | ||
|
|
95e75f0e8f | ||
|
|
1a63ac69d8 | ||
|
|
da67217d74 | ||
|
|
e0c03141f0 | ||
|
|
4d563e3ddd | ||
|
|
8bad78b0d3 | ||
|
|
56fdf245d3 | ||
|
|
5a8c61396f | ||
|
|
5fcf32dcaa | ||
|
|
acf457c0a0 | ||
|
|
e205c0b9a3 | ||
|
|
85f1c5584b | ||
|
|
9e5fddf294 | ||
|
|
3039e43767 | ||
|
|
7371557482 | ||
|
|
4b5e06f50c | ||
|
|
8daefb6946 | ||
|
|
6bf605f9ea | ||
|
|
48221a7af1 | ||
|
|
7a1c9caa70 | ||
|
|
8c2d138976 | ||
|
|
5b09d8279e | ||
|
|
92d8096f3a | ||
|
|
adc6b336e0 | ||
|
|
6d67d6163d | ||
|
|
4d89f1e0e0 | ||
|
|
7706d2992c | ||
|
|
6dcb98a438 | ||
|
|
834d365a97 | ||
|
|
d26ef01afc | ||
|
|
2bd4c5f724 | ||
|
|
2c71d07512 | ||
|
|
17a4744e58 | ||
|
|
46f6380fe6 |
34
CHANGELOG.md
34
CHANGELOG.md
@@ -2,13 +2,43 @@
|
|||||||
|
|
||||||
All notable changes to Bigcapital server-side will be in this file.
|
All notable changes to Bigcapital server-side will be in this file.
|
||||||
|
|
||||||
|
## [1.5.0] - 20-12-2021
|
||||||
|
### Added
|
||||||
|
- Add credit note on sales module.
|
||||||
|
- Add vendor credit on purchases module.
|
||||||
|
- Optimize landed costs on purchase invoices.
|
||||||
|
- Display associated payment transactions on sale invoice drawer.
|
||||||
|
- Display associated pamyment transactions on purchase invoice drawer.
|
||||||
|
- Display item associate invoice, bill, estimate and receipt transactions.
|
||||||
|
- Transactions locking on all transactions or individual modules.
|
||||||
|
- Roles and permissions access control module.
|
||||||
|
- Optimize readonly details style of invoice, receipt, estimate, payment receive,
|
||||||
|
purchase invoice, expense, manual journal, inventory adjustment and cashflow transaction.
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- Dashboard meta boot and authenticated user request query.
|
||||||
|
|
||||||
|
## [1.4.0] - 11-09-2021
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Add SMS notification on sale invoice, receipt, customers payments modules.
|
||||||
|
- Customer quick create in customers list.
|
||||||
|
- Item quick create in items list.
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
change: BIG-171 alerts in global scope and lazy loading.
|
||||||
|
### Fixed
|
||||||
|
fix: BIG-140 - Reordering sell, cost and inventory account on item details.
|
||||||
|
fix: BIG-144 - Typo adjustment dialog success message.
|
||||||
|
fix: BIG-148 - Items entries ordered by index.
|
||||||
|
fix: BIG-132 AR/AP aging summary report filter by none transactions/zero contacts.
|
||||||
|
|
||||||
## [1.2.0-RC] - 03-09-2021
|
## [1.2.0-RC] - 03-09-2021
|
||||||
|
|
||||||
Here we write upgrading notes for brands. It's a team effort to make them as
|
Here we write upgrading notes for brands. It's a team effort to make them as
|
||||||
straightforward as possible.
|
straightforward as possible.
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Add slidable sub-sidebar to improve user experience instead of sub-menu.
|
- Add slidable sub-sidebar to improve user experience instead of sub-menu.
|
||||||
- Add Subscription guard to ensure the organization's subscription is active or
|
- Add Subscription guard to ensure the organization's subscription is active or
|
||||||
redirect all routes to subscription billing page.
|
redirect all routes to subscription billing page.
|
||||||
@@ -35,13 +65,11 @@ straightforward as possible.
|
|||||||
- Add clickable datatable rows to display each row details.
|
- Add clickable datatable rows to display each row details.
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Optimize style of datatable selection checkbox.
|
- Optimize style of datatable selection checkbox.
|
||||||
- Disable animation in dashboard views tabs.
|
- Disable animation in dashboard views tabs.
|
||||||
- Optimize Arabic localization.
|
- Optimize Arabic localization.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- fix: disable submit buttons in pereferences pages.
|
- fix: disable submit buttons in pereferences pages.
|
||||||
- fix: inventory adjustment cost field max/min range to avoid out of range error.
|
- fix: inventory adjustment cost field max/min range to avoid out of range error.
|
||||||
- fix: transactions by customers/vendors report localization.
|
- fix: transactions by customers/vendors report localization.
|
||||||
|
|||||||
@@ -10,6 +10,8 @@
|
|||||||
"@blueprintjs/select": "^3.11.2",
|
"@blueprintjs/select": "^3.11.2",
|
||||||
"@blueprintjs/table": "^3.8.3",
|
"@blueprintjs/table": "^3.8.3",
|
||||||
"@blueprintjs/timezone": "^3.6.2",
|
"@blueprintjs/timezone": "^3.6.2",
|
||||||
|
"@casl/ability": "^5.4.3",
|
||||||
|
"@casl/react": "^2.3.0",
|
||||||
"@reduxjs/toolkit": "^1.2.5",
|
"@reduxjs/toolkit": "^1.2.5",
|
||||||
"@sentry/react": "^6.13.2",
|
"@sentry/react": "^6.13.2",
|
||||||
"@sentry/tracing": "^6.13.2",
|
"@sentry/tracing": "^6.13.2",
|
||||||
@@ -34,6 +36,7 @@
|
|||||||
"cross-env": "^7.0.2",
|
"cross-env": "^7.0.2",
|
||||||
"css-loader": "3.4.2",
|
"css-loader": "3.4.2",
|
||||||
"deep-map-keys": "^2.0.1",
|
"deep-map-keys": "^2.0.1",
|
||||||
|
"dependency-graph": "^0.11.0",
|
||||||
"dotenv": "8.2.0",
|
"dotenv": "8.2.0",
|
||||||
"dotenv-expand": "5.1.0",
|
"dotenv-expand": "5.1.0",
|
||||||
"eslint": "^6.6.0",
|
"eslint": "^6.6.0",
|
||||||
@@ -44,6 +47,7 @@
|
|||||||
"eslint-plugin-jsx-a11y": "6.2.3",
|
"eslint-plugin-jsx-a11y": "6.2.3",
|
||||||
"eslint-plugin-react": "7.18.0",
|
"eslint-plugin-react": "7.18.0",
|
||||||
"eslint-plugin-react-hooks": "^1.6.1",
|
"eslint-plugin-react-hooks": "^1.6.1",
|
||||||
|
"fast-deep-equal": "^3.1.3",
|
||||||
"file-loader": "4.3.0",
|
"file-loader": "4.3.0",
|
||||||
"flow-bin": "^0.123.0",
|
"flow-bin": "^0.123.0",
|
||||||
"formik": "^2.2.5",
|
"formik": "^2.2.5",
|
||||||
@@ -104,6 +108,7 @@
|
|||||||
"semver": "6.3.0",
|
"semver": "6.3.0",
|
||||||
"style-loader": "0.23.1",
|
"style-loader": "0.23.1",
|
||||||
"styled-components": "^5.3.1",
|
"styled-components": "^5.3.1",
|
||||||
|
"stylis-rtlcss": "^2.1.1",
|
||||||
"terser-webpack-plugin": "2.3.4",
|
"terser-webpack-plugin": "2.3.4",
|
||||||
"ts-pnp": "1.1.5",
|
"ts-pnp": "1.1.5",
|
||||||
"url-loader": "2.3.0",
|
"url-loader": "2.3.0",
|
||||||
|
|||||||
6
src/common/TableStyle.js
Normal file
6
src/common/TableStyle.js
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
|
||||||
|
|
||||||
|
export const TableStyle = {
|
||||||
|
Constrant: 'constrant',
|
||||||
|
Regular: 'regular'
|
||||||
|
}
|
||||||
178
src/common/abilityOption.js
Normal file
178
src/common/abilityOption.js
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
export const AbilitySubject = {
|
||||||
|
Item: 'Item',
|
||||||
|
InventoryAdjustment: 'InventoryAdjustment',
|
||||||
|
Estimate: 'SaleEstimate',
|
||||||
|
Invoice: 'SaleInvoice',
|
||||||
|
Receipt: 'SaleReceipt',
|
||||||
|
PaymentReceive: 'PaymentReceive',
|
||||||
|
Bill: 'Bill',
|
||||||
|
PaymentMade: 'PaymentMade',
|
||||||
|
Customer: 'Customer',
|
||||||
|
Vendor: 'Vendor',
|
||||||
|
Account: 'Account',
|
||||||
|
ManualJournal: 'ManualJournal',
|
||||||
|
Expense: 'Expense',
|
||||||
|
Cashflow: 'Cashflow',
|
||||||
|
Report: 'Report',
|
||||||
|
Preferences: 'Preferences',
|
||||||
|
ExchangeRate: 'ExchangeRate',
|
||||||
|
SubscriptionBilling: 'SubscriptionBilling',
|
||||||
|
CreditNote: 'CreditNote',
|
||||||
|
VendorCredit: 'VendorCredit',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ItemAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const InventoryAdjustmentAction = {
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
View: 'View',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SaleEstimateAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
NotifyBySms: 'NotifyBySms',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SaleInvoiceAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
Writeoff: 'bad-debt',
|
||||||
|
NotifyBySms: 'NotifyBySms',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SaleReceiptAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
NotifyBySms: 'NotifyBySms',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PaymentReceiveAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
NotifyBySms: 'NotifyBySms',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CreditNoteAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
Refund: 'Refund'
|
||||||
|
};
|
||||||
|
|
||||||
|
export const VendorCreditAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
Refund: 'Refund'
|
||||||
|
};
|
||||||
|
export const BillAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
NotifyBySms: 'NotifyBySms',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PaymentMadeAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CustomerAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const VendorAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const AccountAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
TransactionsLocking: 'TransactionsLocking',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ManualJournalAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
TransactionLocking: 'TransactionLocking',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ExpenseAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Edit: 'Edit',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CashflowAction = {
|
||||||
|
View: 'View',
|
||||||
|
Create: 'Create',
|
||||||
|
Delete: 'Delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ReportsAction = {
|
||||||
|
ALL: 'all',
|
||||||
|
READ_BALANCE_SHEET: 'read-balance-sheet',
|
||||||
|
READ_TRIAL_BALANCE_SHEET: 'read-trial-balance-sheet',
|
||||||
|
READ_PROFIT_LOSS: 'read-profit-loss',
|
||||||
|
READ_JOURNAL: 'read-journal',
|
||||||
|
READ_GENERAL_LEDGET: 'read-general-ledger',
|
||||||
|
READ_CASHFLOW: 'read-cashflow',
|
||||||
|
READ_AR_AGING_SUMMARY: 'read-ar-aging-summary',
|
||||||
|
READ_AP_AGING_SUMMARY: 'read-ap-aging-summary',
|
||||||
|
READ_PURCHASES_BY_ITEMS: 'read-purchases-by-items',
|
||||||
|
READ_SALES_BY_ITEMS: 'read-sales-by-items',
|
||||||
|
READ_CUSTOMERS_TRANSACTIONS: 'read-customers-transactions',
|
||||||
|
READ_VENDORS_TRANSACTIONS: 'read-vendors-transactions',
|
||||||
|
READ_CUSTOMERS_SUMMARY_BALANCE: 'read-customers-summary-balance',
|
||||||
|
READ_VENDORS_SUMMARY_BALANCE: 'read-vendors-summary-balance',
|
||||||
|
READ_INVENTORY_VALUATION_SUMMARY: 'read-inventory-valuation-summary',
|
||||||
|
READ_INVENTORY_ITEM_DETAILS: 'read-inventory-item-details',
|
||||||
|
READ_CASHFLOW_ACCOUNT_TRANSACTION: 'read-cashflow-account-transactions',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PreferencesAbility = {
|
||||||
|
Mutate: 'Mutate',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ExchangeRateAbility = {
|
||||||
|
View: 'view',
|
||||||
|
Create: 'create',
|
||||||
|
Delete: 'delete',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SubscriptionBillingAbility = {
|
||||||
|
View: 'view',
|
||||||
|
Payment: 'payment',
|
||||||
|
};
|
||||||
@@ -39,6 +39,8 @@ const CLASSES = {
|
|||||||
PAGE_FORM_ITEM: 'page-form--item',
|
PAGE_FORM_ITEM: 'page-form--item',
|
||||||
PAGE_FORM_MAKE_JOURNAL: 'page-form--make-journal-entries',
|
PAGE_FORM_MAKE_JOURNAL: 'page-form--make-journal-entries',
|
||||||
PAGE_FORM_EXPENSE: 'page-form--expense',
|
PAGE_FORM_EXPENSE: 'page-form--expense',
|
||||||
|
PAGE_FORM_CREDIT_NOTE:'page-form--credit-note',
|
||||||
|
PAGE_FORM_VENDOR_CREDIT_NOTE:'page-form--vendor-credit-note',
|
||||||
|
|
||||||
FORM_GROUP_LIST_SELECT: 'form-group--select-list',
|
FORM_GROUP_LIST_SELECT: 'form-group--select-list',
|
||||||
|
|
||||||
@@ -66,6 +68,8 @@ const CLASSES = {
|
|||||||
PREFERENCES_PAGE_INSIDE_CONTENT_USERS: 'preferences-page__inside-content--users',
|
PREFERENCES_PAGE_INSIDE_CONTENT_USERS: 'preferences-page__inside-content--users',
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_CURRENCIES: 'preferences-page__inside-content--currencies',
|
PREFERENCES_PAGE_INSIDE_CONTENT_CURRENCIES: 'preferences-page__inside-content--currencies',
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_ACCOUNTANT: 'preferences-page__inside-content--accountant',
|
PREFERENCES_PAGE_INSIDE_CONTENT_ACCOUNTANT: 'preferences-page__inside-content--accountant',
|
||||||
|
PREFERENCES_PAGE_INSIDE_CONTENT_SMS_INTEGRATION: 'preferences-page__inside-content--sms-integration',
|
||||||
|
PREFERENCES_PAGE_INSIDE_CONTENT_ROLES_FORM: 'preferences-page__inside-content--roles-form',
|
||||||
|
|
||||||
FINANCIAL_REPORT_INSIDER: 'dashboard__insider--financial-report',
|
FINANCIAL_REPORT_INSIDER: 'dashboard__insider--financial-report',
|
||||||
|
|
||||||
|
|||||||
@@ -10,4 +10,12 @@ export const DRAWERS = {
|
|||||||
BILL_DRAWER: 'bill-drawer',
|
BILL_DRAWER: 'bill-drawer',
|
||||||
INVENTORY_ADJUSTMENT_DRAWER: 'inventory-adjustment-drawer',
|
INVENTORY_ADJUSTMENT_DRAWER: 'inventory-adjustment-drawer',
|
||||||
CASHFLOW_TRNASACTION_DRAWER: 'cashflow-transaction-drawer',
|
CASHFLOW_TRNASACTION_DRAWER: 'cashflow-transaction-drawer',
|
||||||
|
|
||||||
|
QUICK_WRITE_VENDOR: 'quick-write-vendor',
|
||||||
|
QUICK_CREATE_CUSTOMER: 'quick-create-customer',
|
||||||
|
QUICK_CREATE_ITEM: 'quick-create-item',
|
||||||
|
CREDIT_NOTE_DETAIL_DRAWER: 'credit-note-detail-drawer',
|
||||||
|
VENDOR_CREDIT_DETAIL_DRAWER: 'vendor-credit-detail-drawer',
|
||||||
|
REFUND_CREDIT_NOTE_DETAIL_DRAWER:'refund-credit-detail-drawer',
|
||||||
|
REFUND_VENDOR_CREDIT_DETAIL_DRAWER:'refund-vendor-detail-drawer'
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +1,21 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { FormattedMessage as T } from 'components';
|
import { FormattedMessage as T } from 'components';
|
||||||
|
import {
|
||||||
|
SaleInvoiceAction,
|
||||||
|
SaleEstimateAction,
|
||||||
|
AbilitySubject,
|
||||||
|
SaleReceiptAction,
|
||||||
|
CustomerAction,
|
||||||
|
PaymentReceiveAction,
|
||||||
|
BillAction,
|
||||||
|
VendorAction,
|
||||||
|
PaymentMadeAction,
|
||||||
|
AccountAction,
|
||||||
|
ManualJournalAction,
|
||||||
|
ExpenseAction,
|
||||||
|
ItemAction,
|
||||||
|
ReportsAction,
|
||||||
|
} from '../common/abilityOption';
|
||||||
|
|
||||||
export const accountsReceivable = [
|
export const accountsReceivable = [
|
||||||
{
|
{
|
||||||
@@ -9,21 +25,29 @@ export const accountsReceivable = [
|
|||||||
title: <T id={'sales_invoices'} />,
|
title: <T id={'sales_invoices'} />,
|
||||||
description: <T id={'tracking_sales_invoices_with_your_customers'} />,
|
description: <T id={'tracking_sales_invoices_with_your_customers'} />,
|
||||||
link: '/invoices',
|
link: '/invoices',
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'sales_estimates'} />,
|
title: <T id={'sales_estimates'} />,
|
||||||
description: <T id={'manage_your_sales_estimates_to_create_quotes'} />,
|
description: <T id={'manage_your_sales_estimates_to_create_quotes'} />,
|
||||||
link: '/estimates',
|
link: '/estimates',
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'sales_receipts'} />,
|
title: <T id={'sales_receipts'} />,
|
||||||
description: <T id={'manage_sales_receipts_for_sales_that_get_paid'} />,
|
description: <T id={'manage_sales_receipts_for_sales_that_get_paid'} />,
|
||||||
link: '/receipts',
|
link: '/receipts',
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'customers'} />,
|
title: <T id={'customers'} />,
|
||||||
description: <T id={'manage_the_customers_relations_with_customer'} />,
|
description: <T id={'manage_the_customers_relations_with_customer'} />,
|
||||||
link: '/customers',
|
link: '/customers',
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
ability: CustomerAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'customers_payments'} />,
|
title: <T id={'customers_payments'} />,
|
||||||
@@ -31,6 +55,8 @@ export const accountsReceivable = [
|
|||||||
<T id={'manage_payment_transactions_from_your_customers'} />
|
<T id={'manage_payment_transactions_from_your_customers'} />
|
||||||
),
|
),
|
||||||
link: '/payment-receives',
|
link: '/payment-receives',
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
ability: PaymentReceiveAction.View,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -46,6 +72,8 @@ export const accountsPayable = [
|
|||||||
<T id={'manage_the_purchase_invoices_with_your_vendors'} />
|
<T id={'manage_the_purchase_invoices_with_your_vendors'} />
|
||||||
),
|
),
|
||||||
link: '/bills',
|
link: '/bills',
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'vendors'} />,
|
title: <T id={'vendors'} />,
|
||||||
@@ -53,11 +81,15 @@ export const accountsPayable = [
|
|||||||
<T id={'manage_the_vendors_relations_with_vendor_relations'} />
|
<T id={'manage_the_vendors_relations_with_vendor_relations'} />
|
||||||
),
|
),
|
||||||
link: '/vendors',
|
link: '/vendors',
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
ability: VendorAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'vendors_payments'} />,
|
title: <T id={'vendors_payments'} />,
|
||||||
description: <T id={'manage_payments_transactions_to_your_vendors'} />,
|
description: <T id={'manage_payments_transactions_to_your_vendors'} />,
|
||||||
link: '/payment-mades',
|
link: '/payment-mades',
|
||||||
|
subject: AbilitySubject.PaymentMade,
|
||||||
|
ability: PaymentMadeAction.View,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -77,21 +109,35 @@ export const financialAccounting = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/accounts',
|
link: '/accounts',
|
||||||
|
subject: AbilitySubject.Account,
|
||||||
|
ability: AccountAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'manual_journal'}/>,
|
title: <T id={'manual_journal'} />,
|
||||||
description:<T id={'manage_manual_journal_transactions_on_accounts'}/>,
|
description: (
|
||||||
|
<T id={'manage_manual_journal_transactions_on_accounts'} />
|
||||||
|
),
|
||||||
link: '/manual-journals',
|
link: '/manual-journals',
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'expenses'}/>,
|
title: <T id={'expenses'} />,
|
||||||
description:<T id={'track_your_indirect_expenses_under_specific_categories'}/>,
|
description: (
|
||||||
|
<T id={'track_your_indirect_expenses_under_specific_categories'} />
|
||||||
|
),
|
||||||
link: '/expenses',
|
link: '/expenses',
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
ability: ExpenseAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'financial_statements'}/>,
|
title: <T id={'financial_statements'} />,
|
||||||
description:<T id={'show_financial_reports_about_your_organization'}/>,
|
description: (
|
||||||
|
<T id={'show_financial_reports_about_your_organization'} />
|
||||||
|
),
|
||||||
link: '/financial-reports',
|
link: '/financial-reports',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.ALL,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -102,19 +148,27 @@ export const productsServices = [
|
|||||||
sectionTitle: <T id={'products_services_inventory'} />,
|
sectionTitle: <T id={'products_services_inventory'} />,
|
||||||
shortcuts: [
|
shortcuts: [
|
||||||
{
|
{
|
||||||
title: <T id={'products_services'}/>,
|
title: <T id={'products_services'} />,
|
||||||
description:<T id={'manage_your_products_inventory_or_non_inventory'}/>,
|
description: (
|
||||||
|
<T id={'manage_your_products_inventory_or_non_inventory'} />
|
||||||
|
),
|
||||||
link: '/items',
|
link: '/items',
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.View,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'products_services_categories'}/>,
|
title: <T id={'products_services_categories'} />,
|
||||||
description:<T id={'group_your_products_and_service'}/>,
|
description: <T id={'group_your_products_and_service'} />,
|
||||||
link: 'items/categories',
|
link: 'items/categories',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'inventory_adjustments'}/>,
|
title: <T id={'inventory_adjustments'} />,
|
||||||
description: <T id={'manage_your_inventory_adjustment_of_inventory_items'}/>,
|
description: (
|
||||||
|
<T id={'manage_your_inventory_adjustment_of_inventory_items'} />
|
||||||
|
),
|
||||||
link: '/inventory-adjustments',
|
link: '/inventory-adjustments',
|
||||||
|
subject: AbilitySubject.InventoryAdjustment,
|
||||||
|
ability: SaleInvoiceAction.View,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|||||||
4
src/common/index.js
Normal file
4
src/common/index.js
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export * from './TableStyle';
|
||||||
51
src/common/itemPaymentTranactionsOption.js
Normal file
51
src/common/itemPaymentTranactionsOption.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import {
|
||||||
|
AbilitySubject,
|
||||||
|
SaleEstimateAction,
|
||||||
|
SaleReceiptAction,
|
||||||
|
SaleInvoiceAction,
|
||||||
|
BillAction,
|
||||||
|
} from '../common/abilityOption';
|
||||||
|
import { useAbilitiesFilter } from '../hooks';
|
||||||
|
|
||||||
|
export const getItemPaymentTransactions = () => [
|
||||||
|
{
|
||||||
|
name: 'invoices',
|
||||||
|
label: intl.get('invoices'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'estimates',
|
||||||
|
label: intl.get('estimates'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'receipts',
|
||||||
|
label: intl.get('receipts'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'bills',
|
||||||
|
label: intl.get('bills'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const useGetItemPaymentTransactionsMenu = () => {
|
||||||
|
const itemTransactionMenu = getItemPaymentTransactions();
|
||||||
|
const abilitiesFilter = useAbilitiesFilter();
|
||||||
|
|
||||||
|
return abilitiesFilter(itemTransactionMenu);
|
||||||
|
};
|
||||||
@@ -1,110 +1,228 @@
|
|||||||
import React from 'react';
|
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
|
import {
|
||||||
|
AbilitySubject,
|
||||||
|
AccountAction,
|
||||||
|
BillAction,
|
||||||
|
CashflowAction,
|
||||||
|
CustomerAction,
|
||||||
|
ExpenseAction,
|
||||||
|
ItemAction,
|
||||||
|
ManualJournalAction,
|
||||||
|
ReportsAction,
|
||||||
|
SaleEstimateAction,
|
||||||
|
SaleInvoiceAction,
|
||||||
|
SaleReceiptAction,
|
||||||
|
VendorAction,
|
||||||
|
} from './abilityOption';
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + I',
|
shortcut_key: 'Shift + I',
|
||||||
description: intl.get('jump_to_the_invoices'),
|
description: intl.get('jump_to_the_invoices'),
|
||||||
|
permission: {
|
||||||
|
ability: SaleInvoiceAction.View,
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + E',
|
shortcut_key: 'Shift + E',
|
||||||
description: intl.get('jump_to_the_estimates'),
|
description: intl.get('jump_to_the_estimates'),
|
||||||
|
permission: {
|
||||||
|
ability: SaleEstimateAction.View,
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + R',
|
shortcut_key: 'Shift + R',
|
||||||
description: intl.get('jump_to_the_receipts'),
|
description: intl.get('jump_to_the_receipts'),
|
||||||
|
permission: {
|
||||||
|
ability: SaleReceiptAction.View,
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + X',
|
shortcut_key: 'Shift + X',
|
||||||
description: intl.get('jump_to_the_expenses'),
|
description: intl.get('jump_to_the_expenses'),
|
||||||
|
permission: {
|
||||||
|
ability: ExpenseAction.View,
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + C',
|
shortcut_key: 'Shift + C',
|
||||||
description: intl.get('jump_to_the_customers'),
|
description: intl.get('jump_to_the_customers'),
|
||||||
|
permission: {
|
||||||
|
ability: CustomerAction.View,
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + V',
|
shortcut_key: 'Shift + V',
|
||||||
description: intl.get('jump_to_the_vendors'),
|
description: intl.get('jump_to_the_vendors'),
|
||||||
|
permission: {
|
||||||
|
ability: VendorAction.View,
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + A',
|
shortcut_key: 'Shift + A',
|
||||||
description: intl.get('jump_to_the_chart_of_accounts'),
|
description: intl.get('jump_to_the_chart_of_accounts'),
|
||||||
|
permission: {
|
||||||
|
ability: AccountAction.View,
|
||||||
|
subject: AbilitySubject.Account,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + B',
|
shortcut_key: 'Shift + B',
|
||||||
description: intl.get('jump_to_the_bills'),
|
description: intl.get('jump_to_the_bills'),
|
||||||
|
permission: {
|
||||||
|
ability: BillAction.View,
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + M',
|
shortcut_key: 'Shift + M',
|
||||||
description: intl.get('jump_to_the_manual_journals'),
|
description: intl.get('jump_to_the_manual_journals'),
|
||||||
|
permission: {
|
||||||
|
ability: ManualJournalAction.View,
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + W',
|
shortcut_key: 'Shift + W',
|
||||||
description: intl.get('jump_to_the_items'),
|
description: intl.get('jump_to_the_items'),
|
||||||
|
permission: {
|
||||||
|
ability: ItemAction.View,
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + D',
|
shortcut_key: 'Shift + D',
|
||||||
description: intl.get('jump_to_the_add_money_in'),
|
description: intl.get('jump_to_the_add_money_in'),
|
||||||
|
permission: {
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + Q',
|
shortcut_key: 'Shift + Q',
|
||||||
description: intl.get('jump_to_the_add_money_out'),
|
description: intl.get('jump_to_the_add_money_out'),
|
||||||
|
permission: {
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + 1',
|
shortcut_key: 'Shift + 1',
|
||||||
description: intl.get('jump_to_the_balance_sheet'),
|
description: intl.get('jump_to_the_balance_sheet'),
|
||||||
|
permission: {
|
||||||
|
ability: ReportsAction.READ_BALANCE_SHEET,
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + 2',
|
shortcut_key: 'Shift + 2',
|
||||||
description: intl.get('jump_to_the_profit_loss_sheet'),
|
description: intl.get('jump_to_the_profit_loss_sheet'),
|
||||||
|
permission: {
|
||||||
|
ability: ReportsAction.READ_PROFIT_LOSS,
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + 3',
|
shortcut_key: 'Shift + 3',
|
||||||
description: intl.get('jump_to_the_journal_sheet'),
|
description: intl.get('jump_to_the_journal_sheet'),
|
||||||
|
permission: {
|
||||||
|
ability: ReportsAction.READ_JOURNAL,
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + 4',
|
shortcut_key: 'Shift + 4',
|
||||||
description: intl.get('jump_to_the_general_ledger_sheet'),
|
description: intl.get('jump_to_the_general_ledger_sheet'),
|
||||||
|
permission: {
|
||||||
|
ability: ReportsAction.READ_GENERAL_LEDGET,
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Shift + 5',
|
shortcut_key: 'Shift + 5',
|
||||||
description: intl.get('jump_to_the_trial_balance_sheet'),
|
description: intl.get('jump_to_the_trial_balance_sheet'),
|
||||||
|
permission: {
|
||||||
|
ability: ReportsAction.READ_TRIAL_BALANCE_SHEET,
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + I ',
|
shortcut_key: 'Ctrl + Shift + I ',
|
||||||
description: intl.get('create_a_new_invoice'),
|
description: intl.get('create_a_new_invoice'),
|
||||||
|
permission: {
|
||||||
|
ability: SaleInvoiceAction.Create,
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + E ',
|
shortcut_key: 'Ctrl + Shift + E ',
|
||||||
description: intl.get('create_a_new_estimate'),
|
description: intl.get('create_a_new_estimate'),
|
||||||
|
permission: {
|
||||||
|
ability: SaleEstimateAction.Create,
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + R ',
|
shortcut_key: 'Ctrl + Shift + R ',
|
||||||
description: intl.get('create_a_new_receipt'),
|
description: intl.get('create_a_new_receipt'),
|
||||||
|
permission: {
|
||||||
|
ability: SaleReceiptAction.Create,
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + X ',
|
shortcut_key: 'Ctrl + Shift + X ',
|
||||||
description: intl.get('create_a_new_expense'),
|
description: intl.get('create_a_new_expense'),
|
||||||
|
permission: {
|
||||||
|
ability: ExpenseAction.Create,
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + C ',
|
shortcut_key: 'Ctrl + Shift + C ',
|
||||||
description: intl.get('create_a_new_customer'),
|
description: intl.get('create_a_new_customer'),
|
||||||
|
permission: {
|
||||||
|
ability: CustomerAction.Create,
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + V ',
|
shortcut_key: 'Ctrl + Shift + V ',
|
||||||
description: intl.get('create_a_new_vendor'),
|
description: intl.get('create_a_new_vendor'),
|
||||||
|
permission: {
|
||||||
|
ability: VendorAction.Create,
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + B ',
|
shortcut_key: 'Ctrl + Shift + B ',
|
||||||
description: intl.get('create_a_new_bill'),
|
description: intl.get('create_a_new_bill'),
|
||||||
|
permission: {
|
||||||
|
ability: BillAction.Create,
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + M ',
|
shortcut_key: 'Ctrl + Shift + M ',
|
||||||
description: intl.get('create_a_new_journal'),
|
description: intl.get('create_a_new_journal'),
|
||||||
|
permission: {
|
||||||
|
ability: ManualJournalAction.Create,
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + Shift + W ',
|
shortcut_key: 'Ctrl + Shift + W ',
|
||||||
description: intl.get('create_a_new_item'),
|
description: intl.get('create_a_new_item'),
|
||||||
|
permission: {
|
||||||
|
ability: ItemAction.Create,
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
shortcut_key: 'Ctrl + / ',
|
shortcut_key: 'Ctrl + / ',
|
||||||
|
|||||||
659
src/common/permissionsSchema.js
Normal file
659
src/common/permissionsSchema.js
Normal file
@@ -0,0 +1,659 @@
|
|||||||
|
import { chain } from 'lodash';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import {
|
||||||
|
AbilitySubject,
|
||||||
|
AccountAction,
|
||||||
|
BillAction,
|
||||||
|
CreditNoteAction,
|
||||||
|
CustomerAction,
|
||||||
|
ExpenseAction,
|
||||||
|
ItemAction,
|
||||||
|
ManualJournalAction,
|
||||||
|
PaymentMadeAction,
|
||||||
|
PaymentReceiveAction,
|
||||||
|
ReportsAction,
|
||||||
|
SaleEstimateAction,
|
||||||
|
SaleInvoiceAction,
|
||||||
|
SaleReceiptAction,
|
||||||
|
VendorAction,
|
||||||
|
VendorCreditAction,
|
||||||
|
} from './abilityOption';
|
||||||
|
|
||||||
|
export const ModulePermissionsStyle = {
|
||||||
|
Columns: 'columns',
|
||||||
|
Vertical: 'vertical',
|
||||||
|
};
|
||||||
|
|
||||||
|
const PermissionColumn = {
|
||||||
|
View: 'view',
|
||||||
|
Create: 'create',
|
||||||
|
Delete: 'delete',
|
||||||
|
Edit: 'edit',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getPermissionsSchema = () => [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.items_inventory'),
|
||||||
|
type: ModulePermissionsStyle.Columns,
|
||||||
|
serviceFullAccess: true,
|
||||||
|
columns: [
|
||||||
|
{ label: intl.get('permissions.column.view'), key: 'view' },
|
||||||
|
{ label: intl.get('permissions.column.create'), key: 'create' },
|
||||||
|
{ label: intl.get('permissions.column.edit'), key: 'edit' },
|
||||||
|
{ label: intl.get('permissions.column.delete'), key: 'delete' },
|
||||||
|
],
|
||||||
|
services: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.items'),
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: ItemAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: ItemAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: ItemAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: ItemAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: ItemAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: ItemAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: ItemAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.inventory_adjustment'),
|
||||||
|
subject: AbilitySubject.InventoryAdjustment,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: 'View',
|
||||||
|
key: ItemAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Create',
|
||||||
|
key: ItemAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: ItemAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Edit',
|
||||||
|
key: ItemAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: ItemAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Delete',
|
||||||
|
key: ItemAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: ItemAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.contacts'),
|
||||||
|
type: ModulePermissionsStyle.Columns,
|
||||||
|
serviceFullAccess: true,
|
||||||
|
moduleFullAccess: true,
|
||||||
|
columns: [
|
||||||
|
{ label: intl.get('permissions.column.view'), key: 'view' },
|
||||||
|
{ label: intl.get('permissions.column.create'), key: 'create' },
|
||||||
|
{ label: intl.get('permissions.column.edit'), key: 'edit' },
|
||||||
|
{ label: intl.get('permissions.column.delete'), key: 'delete' },
|
||||||
|
],
|
||||||
|
services: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.customers'),
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: CustomerAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: CustomerAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: CustomerAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: CustomerAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: CustomerAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: CustomerAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: CustomerAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.vendors'),
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: VendorAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: VendorAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: VendorAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: VendorAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: VendorAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: VendorAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: VendorAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.sales'),
|
||||||
|
type: ModulePermissionsStyle.Columns,
|
||||||
|
serviceFullAccess: true,
|
||||||
|
moduleFullAccess: true,
|
||||||
|
columns: [
|
||||||
|
{ label: intl.get('permissions.column.view'), key: 'view' },
|
||||||
|
{ label: intl.get('permissions.column.create'), key: 'create' },
|
||||||
|
{ label: intl.get('permissions.column.edit'), key: 'edit' },
|
||||||
|
{ label: intl.get('permissions.column.delete'), key: 'delete' },
|
||||||
|
],
|
||||||
|
services: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.sale_invoice'),
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: SaleInvoiceAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: SaleInvoiceAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: SaleInvoiceAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: SaleInvoiceAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: SaleInvoiceAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: SaleInvoiceAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: SaleInvoiceAction.Edit }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.written_off_invoice'),
|
||||||
|
key: SaleInvoiceAction.Writeoff,
|
||||||
|
depend: [{ key: SaleInvoiceAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.sale_estimate'),
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: SaleEstimateAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: SaleEstimateAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: SaleEstimateAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: SaleEstimateAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: SaleEstimateAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: SaleEstimateAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: SaleEstimateAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.sale_receipt'),
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: SaleReceiptAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: SaleReceiptAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: SaleReceiptAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: SaleReceiptAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: SaleReceiptAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: SaleReceiptAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: SaleReceiptAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.credit_note'),
|
||||||
|
subject: AbilitySubject.CreditNote,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: CreditNoteAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: CreditNoteAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: CreditNoteAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: CreditNoteAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: CreditNoteAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: CreditNoteAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: CreditNoteAction.Edit }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.refund_credit_note'),
|
||||||
|
key: CreditNoteAction.Refund,
|
||||||
|
depend: [{ key: CreditNoteAction.View }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.payment_receive'),
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: PaymentReceiveAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: PaymentReceiveAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: PaymentReceiveAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: PaymentReceiveAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: PaymentReceiveAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: PaymentReceiveAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: PaymentReceiveAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.purchases'),
|
||||||
|
type: ModulePermissionsStyle.Columns,
|
||||||
|
serviceFullAccess: true,
|
||||||
|
moduleFullAccess: true,
|
||||||
|
columns: [
|
||||||
|
{ label: intl.get('permissions.column.view'), key: 'view' },
|
||||||
|
{ label: intl.get('permissions.column.create'), key: 'create' },
|
||||||
|
{ label: intl.get('permissions.column.edit'), key: 'edit' },
|
||||||
|
{ label: intl.get('permissions.column.delete'), key: 'delete' },
|
||||||
|
],
|
||||||
|
services: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.bills'),
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: BillAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: BillAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: BillAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: BillAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: BillAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: BillAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: BillAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.vendor_credits'),
|
||||||
|
subject: AbilitySubject.VendorCredit,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: VendorCreditAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: VendorCreditAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: VendorCreditAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: VendorCreditAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: VendorCreditAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: VendorCreditAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: VendorCreditAction.Edit }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.refund_vendor_credit'),
|
||||||
|
key: VendorCreditAction.Refund,
|
||||||
|
depend: [{ key: VendorCreditAction.View }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.payment_made'),
|
||||||
|
subject: AbilitySubject.PaymentMade,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: PaymentMadeAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: PaymentMadeAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: PaymentMadeAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: PaymentMadeAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: PaymentMadeAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: PaymentMadeAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: PaymentMadeAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.financial_accounting'),
|
||||||
|
type: ModulePermissionsStyle.Columns,
|
||||||
|
serviceFullAccess: true,
|
||||||
|
moduleFullAccess: true,
|
||||||
|
columns: [
|
||||||
|
{ label: intl.get('permissions.column.view'), key: 'view' },
|
||||||
|
{ label: intl.get('permissions.column.create'), key: 'create' },
|
||||||
|
{ label: intl.get('permissions.column.edit'), key: 'edit' },
|
||||||
|
{ label: intl.get('permissions.column.delete'), key: 'delete' },
|
||||||
|
],
|
||||||
|
services: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.manual_journals'),
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: ManualJournalAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: ManualJournalAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: ManualJournalAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: ManualJournalAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: ManualJournalAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: ManualJournalAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: ManualJournalAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.chart_of_accounts'),
|
||||||
|
subject: AbilitySubject.Account,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: AccountAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: AccountAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: AccountAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: AccountAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: AccountAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: AccountAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: AccountAction.Edit }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.transactions_locking'),
|
||||||
|
key: AccountAction.TransactionsLocking,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.expenses'),
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.view'),
|
||||||
|
key: ExpenseAction.View,
|
||||||
|
relatedColumn: PermissionColumn.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.create'),
|
||||||
|
key: ExpenseAction.Create,
|
||||||
|
relatedColumn: PermissionColumn.Create,
|
||||||
|
depend: [{ key: ExpenseAction.View }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.edit'),
|
||||||
|
key: ExpenseAction.Edit,
|
||||||
|
relatedColumn: PermissionColumn.Edit,
|
||||||
|
depend: [{ key: ExpenseAction.Create }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.column.delete'),
|
||||||
|
key: ExpenseAction.Delete,
|
||||||
|
relatedColumn: PermissionColumn.Delete,
|
||||||
|
depend: [{ key: ExpenseAction.Edit }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.reports'),
|
||||||
|
type: ModulePermissionsStyle.Vertical,
|
||||||
|
serviceFullAccess: true,
|
||||||
|
moduleFullAccess: true,
|
||||||
|
services: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.financial_reports'),
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
permissions: [
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.balance_sheet'),
|
||||||
|
key: ReportsAction.READ_BALANCE_SHEET,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.trial_balance_sheet'),
|
||||||
|
key: ReportsAction.READ_TRIAL_BALANCE_SHEET,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.profit_loss_sheet'),
|
||||||
|
key: ReportsAction.READ_PROFIT_LOSS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.cash_flow_sheet'),
|
||||||
|
key: ReportsAction.READ_CASHFLOW,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.journal_sheet'),
|
||||||
|
key: ReportsAction.READ_JOURNAL,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.general_ledger'),
|
||||||
|
key: ReportsAction.READ_GENERAL_LEDGET,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.a_r_aging_summary_report'),
|
||||||
|
key: ReportsAction.READ_AR_AGING_SUMMARY,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.a_r_aging_summary_report'),
|
||||||
|
key: ReportsAction.READ_AP_AGING_SUMMARY,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.purchases_by_items'),
|
||||||
|
key: ReportsAction.READ_PURCHASES_BY_ITEMS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.sales_by_items'),
|
||||||
|
key: ReportsAction.READ_SALES_BY_ITEMS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.customers_transactions'),
|
||||||
|
key: ReportsAction.READ_CUSTOMERS_TRANSACTIONS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.vendors_transactions'),
|
||||||
|
key: ReportsAction.READ_VENDORS_TRANSACTIONS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.customers_summary_balance'),
|
||||||
|
key: ReportsAction.READ_CUSTOMERS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.vendors_summary_balance'),
|
||||||
|
key: ReportsAction.READ_VENDORS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.inventory_valuation_summary'),
|
||||||
|
key: ReportsAction.READ_INVENTORY_VALUATION_SUMMARY,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.inventory_items_details'),
|
||||||
|
key: ReportsAction.READ_INVENTORY_ITEM_DETAILS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: intl.get('permissions.cashflow_account_transactions'),
|
||||||
|
key: ReportsAction.READ_CASHFLOW_ACCOUNT_TRANSACTION,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export function getPermissionsSchemaService(subject) {
|
||||||
|
const permissions = getPermissionsSchema();
|
||||||
|
|
||||||
|
return chain(permissions)
|
||||||
|
.map((perm) => perm.services)
|
||||||
|
.flatten()
|
||||||
|
.find((service) => service.subject === subject)
|
||||||
|
.value();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPermissionsSchemaServices() {
|
||||||
|
const permissions = getPermissionsSchema();
|
||||||
|
|
||||||
|
return chain(permissions)
|
||||||
|
.map((module) => module.services)
|
||||||
|
.flatten()
|
||||||
|
.value();
|
||||||
|
}
|
||||||
@@ -1,10 +1,71 @@
|
|||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
|
import {
|
||||||
|
AbilitySubject,
|
||||||
|
SaleInvoiceAction,
|
||||||
|
CustomerAction,
|
||||||
|
VendorAction,
|
||||||
|
ManualJournalAction,
|
||||||
|
ExpenseAction,
|
||||||
|
} from '../common/abilityOption';
|
||||||
|
import { useAbilitiesFilter } from '../hooks';
|
||||||
|
|
||||||
export const getQuickNewActions = () => [
|
export const getQuickNewActions = () => [
|
||||||
{ path: 'invoices/new', name: intl.get('sale_invoice') },
|
{
|
||||||
{ path: 'bills/new', name: intl.get('purchase_invoice') },
|
path: 'invoices/new',
|
||||||
{ path: 'make-journal-entry', name: intl.get('manual_journal') },
|
name: intl.get('sale_invoice'),
|
||||||
{ path: 'expenses/new', name: intl.get('expense') },
|
permission: {
|
||||||
{ path: 'customers/new', name: intl.get('customer') },
|
subject: AbilitySubject.Invoice,
|
||||||
{ path: 'vendors/new', name: intl.get('vendor') },
|
ability: SaleInvoiceAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'bills/new',
|
||||||
|
name: intl.get('purchase_invoice'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'make-journal-entry',
|
||||||
|
name: intl.get('manual_journal'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'expenses/new',
|
||||||
|
name: intl.get('expense'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
ability: ExpenseAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'customers/new',
|
||||||
|
name: intl.get('customer'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
ability: CustomerAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'vendors/new',
|
||||||
|
name: intl.get('vendor'),
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
ability: VendorAction.Vendor,
|
||||||
|
},
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve the dashboard quick new menu items.
|
||||||
|
*/
|
||||||
|
export const useGetQuickNewMenu = () => {
|
||||||
|
const quickNewMenu = getQuickNewActions();
|
||||||
|
const abilitiesFilter = useAbilitiesFilter();
|
||||||
|
|
||||||
|
return abilitiesFilter(quickNewMenu);
|
||||||
|
};
|
||||||
|
|||||||
@@ -11,4 +11,6 @@ export const RESOURCES_TYPES = {
|
|||||||
EXPENSE: 'expense',
|
EXPENSE: 'expense',
|
||||||
MANUAL_JOURNAL: 'manual_journal',
|
MANUAL_JOURNAL: 'manual_journal',
|
||||||
ACCOUNT: 'account',
|
ACCOUNT: 'account',
|
||||||
|
CREDIT_NOTE: 'credit_note',
|
||||||
|
VENDOR_CREDIT:'vendor_credit'
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -14,6 +14,8 @@ export const TABLES = {
|
|||||||
EXPENSES: 'expenses',
|
EXPENSES: 'expenses',
|
||||||
CASHFLOW_ACCOUNTS: 'cashflow_accounts',
|
CASHFLOW_ACCOUNTS: 'cashflow_accounts',
|
||||||
CASHFLOW_Transactions: 'cashflow_transactions',
|
CASHFLOW_Transactions: 'cashflow_transactions',
|
||||||
|
CREDIT_NOTES: 'credit_notes',
|
||||||
|
VENDOR_CREDITS: 'vendor_credits',
|
||||||
};
|
};
|
||||||
|
|
||||||
export const TABLE_SIZE = {
|
export const TABLE_SIZE = {
|
||||||
|
|||||||
@@ -1,13 +1,56 @@
|
|||||||
import React, { useCallback, useState, useEffect, useMemo } from 'react';
|
import React, { useCallback, useState, useEffect, useMemo } from 'react';
|
||||||
import { MenuItem, Button } from '@blueprintjs/core';
|
import { MenuItem, Button } from '@blueprintjs/core';
|
||||||
import { Select } from '@blueprintjs/select';
|
import { Select } from '@blueprintjs/select';
|
||||||
import { MenuItemNestedText, FormattedMessage as T } from 'components';
|
import * as R from 'ramda';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
import intl from 'react-intl-universal'
|
||||||
|
|
||||||
|
import { MenuItemNestedText, FormattedMessage as T } from 'components';
|
||||||
import { filterAccountsByQuery } from './utils';
|
import { filterAccountsByQuery } from './utils';
|
||||||
import { nestedArrayToflatten } from 'utils';
|
import { nestedArrayToflatten } from 'utils';
|
||||||
import { CLASSES } from 'common/classes';
|
import { CLASSES } from 'common/classes';
|
||||||
|
|
||||||
export default function AccountsSelectList({
|
import withDialogActions from 'containers/Dialog/withDialogActions';
|
||||||
|
|
||||||
|
// Create new account renderer.
|
||||||
|
const createNewItemRenderer = (query, active, handleClick) => {
|
||||||
|
return (
|
||||||
|
<MenuItem
|
||||||
|
icon="add"
|
||||||
|
text={intl.get('list.create', { value: `"${query}"` })}
|
||||||
|
active={active}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Create new item from the given query string.
|
||||||
|
const createNewItemFromQuery = (name) => {
|
||||||
|
return {
|
||||||
|
name,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// Filters accounts items.
|
||||||
|
const filterAccountsPredicater = (query, account, _index, exactMatch) => {
|
||||||
|
const normalizedTitle = account.name.toLowerCase();
|
||||||
|
const normalizedQuery = query.toLowerCase();
|
||||||
|
|
||||||
|
if (exactMatch) {
|
||||||
|
return normalizedTitle === normalizedQuery;
|
||||||
|
} else {
|
||||||
|
return `${account.code} ${normalizedTitle}`.indexOf(normalizedQuery) >= 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Accounts select list.
|
||||||
|
*/
|
||||||
|
function AccountsSelectList({
|
||||||
|
// #withDialogActions
|
||||||
|
openDialog,
|
||||||
|
|
||||||
|
// #ownProps
|
||||||
accounts,
|
accounts,
|
||||||
initialAccountId,
|
initialAccountId,
|
||||||
selectedAccountId,
|
selectedAccountId,
|
||||||
@@ -21,6 +64,8 @@ export default function AccountsSelectList({
|
|||||||
filterByNormal,
|
filterByNormal,
|
||||||
filterByRootTypes,
|
filterByRootTypes,
|
||||||
|
|
||||||
|
allowCreate,
|
||||||
|
|
||||||
buttonProps = {},
|
buttonProps = {},
|
||||||
}) {
|
}) {
|
||||||
const flattenAccounts = useMemo(
|
const flattenAccounts = useMemo(
|
||||||
@@ -51,6 +96,7 @@ export default function AccountsSelectList({
|
|||||||
[initialAccountId, filteredAccounts],
|
[initialAccountId, filteredAccounts],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Select account item.
|
||||||
const [selectedAccount, setSelectedAccount] = useState(
|
const [selectedAccount, setSelectedAccount] = useState(
|
||||||
initialAccount || null,
|
initialAccount || null,
|
||||||
);
|
);
|
||||||
@@ -76,31 +122,25 @@ export default function AccountsSelectList({
|
|||||||
);
|
);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onAccountSelect = useCallback(
|
// Handle the account item select.
|
||||||
|
const handleAccountSelect = useCallback(
|
||||||
(account) => {
|
(account) => {
|
||||||
setSelectedAccount({ ...account });
|
if (account.id) {
|
||||||
onAccountSelected && onAccountSelected(account);
|
setSelectedAccount({ ...account });
|
||||||
},
|
onAccountSelected && onAccountSelected(account);
|
||||||
[setSelectedAccount, onAccountSelected],
|
|
||||||
);
|
|
||||||
|
|
||||||
// Filters accounts items.
|
|
||||||
const filterAccountsPredicater = useCallback(
|
|
||||||
(query, account, _index, exactMatch) => {
|
|
||||||
const normalizedTitle = account.name.toLowerCase();
|
|
||||||
const normalizedQuery = query.toLowerCase();
|
|
||||||
|
|
||||||
if (exactMatch) {
|
|
||||||
return normalizedTitle === normalizedQuery;
|
|
||||||
} else {
|
} else {
|
||||||
return (
|
openDialog('account-form');
|
||||||
`${account.code} ${normalizedTitle}`.indexOf(normalizedQuery) >= 0
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
[],
|
[setSelectedAccount, onAccountSelected, openDialog],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Maybe inject new item props to select component.
|
||||||
|
const maybeCreateNewItemRenderer = allowCreate ? createNewItemRenderer : null;
|
||||||
|
const maybeCreateNewItemFromQuery = allowCreate
|
||||||
|
? createNewItemFromQuery
|
||||||
|
: null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Select
|
<Select
|
||||||
items={filteredAccounts}
|
items={filteredAccounts}
|
||||||
@@ -113,11 +153,13 @@ export default function AccountsSelectList({
|
|||||||
inline: popoverFill,
|
inline: popoverFill,
|
||||||
}}
|
}}
|
||||||
filterable={true}
|
filterable={true}
|
||||||
onItemSelect={onAccountSelect}
|
onItemSelect={handleAccountSelect}
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
className={classNames('form-group--select-list', {
|
className={classNames('form-group--select-list', {
|
||||||
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
})}
|
})}
|
||||||
|
createNewItemRenderer={maybeCreateNewItemRenderer}
|
||||||
|
createNewItemFromQuery={maybeCreateNewItemFromQuery}
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
@@ -127,3 +169,5 @@ export default function AccountsSelectList({
|
|||||||
</Select>
|
</Select>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export default R.compose(withDialogActions)(AccountsSelectList);
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import React, { useState, useCallback, useEffect, useMemo } from 'react';
|
|||||||
import { MenuItem } from '@blueprintjs/core';
|
import { MenuItem } from '@blueprintjs/core';
|
||||||
import { Suggest } from '@blueprintjs/select';
|
import { Suggest } from '@blueprintjs/select';
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { CLASSES } from 'common/classes';
|
import { CLASSES } from 'common/classes';
|
||||||
@@ -10,10 +11,55 @@ import { MenuItemNestedText, FormattedMessage as T } from 'components';
|
|||||||
import { filterAccountsByQuery } from './utils';
|
import { filterAccountsByQuery } from './utils';
|
||||||
import { nestedArrayToflatten } from 'utils';
|
import { nestedArrayToflatten } from 'utils';
|
||||||
|
|
||||||
|
import withDialogActions from 'containers/Dialog/withDialogActions';
|
||||||
|
|
||||||
|
// Create new account renderer.
|
||||||
|
const createNewItemRenderer = (query, active, handleClick) => {
|
||||||
|
return (
|
||||||
|
<MenuItem
|
||||||
|
icon="add"
|
||||||
|
text={intl.get('list.create', { value: `"${query}"` })}
|
||||||
|
active={active}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Create new item from the given query string.
|
||||||
|
const createNewItemFromQuery = (name) => {
|
||||||
|
return {
|
||||||
|
name,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle input value renderer.
|
||||||
|
const handleInputValueRenderer = (inputValue) => {
|
||||||
|
if (inputValue) {
|
||||||
|
return inputValue.name.toString();
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
};
|
||||||
|
|
||||||
|
// Filters accounts items.
|
||||||
|
const filterAccountsPredicater = (query, account, _index, exactMatch) => {
|
||||||
|
const normalizedTitle = account.name.toLowerCase();
|
||||||
|
const normalizedQuery = query.toLowerCase();
|
||||||
|
|
||||||
|
if (exactMatch) {
|
||||||
|
return normalizedTitle === normalizedQuery;
|
||||||
|
} else {
|
||||||
|
return `${account.code} ${normalizedTitle}`.indexOf(normalizedQuery) >= 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accounts suggest field.
|
* Accounts suggest field.
|
||||||
*/
|
*/
|
||||||
export default function AccountsSuggestField({
|
function AccountsSuggestField({
|
||||||
|
// #withDialogActions
|
||||||
|
openDialog,
|
||||||
|
|
||||||
|
// #ownProps
|
||||||
accounts,
|
accounts,
|
||||||
initialAccountId,
|
initialAccountId,
|
||||||
selectedAccountId,
|
selectedAccountId,
|
||||||
@@ -26,6 +72,8 @@ export default function AccountsSuggestField({
|
|||||||
filterByNormal,
|
filterByNormal,
|
||||||
filterByRootTypes = [],
|
filterByRootTypes = [],
|
||||||
|
|
||||||
|
allowCreate,
|
||||||
|
|
||||||
...suggestProps
|
...suggestProps
|
||||||
}) {
|
}) {
|
||||||
const flattenAccounts = useMemo(
|
const flattenAccounts = useMemo(
|
||||||
@@ -69,23 +117,6 @@ export default function AccountsSuggestField({
|
|||||||
}
|
}
|
||||||
}, [selectedAccountId, filteredAccounts, setSelectedAccount]);
|
}, [selectedAccountId, filteredAccounts, setSelectedAccount]);
|
||||||
|
|
||||||
// Filters accounts items.
|
|
||||||
const filterAccountsPredicater = useCallback(
|
|
||||||
(query, account, _index, exactMatch) => {
|
|
||||||
const normalizedTitle = account.name.toLowerCase();
|
|
||||||
const normalizedQuery = query.toLowerCase();
|
|
||||||
|
|
||||||
if (exactMatch) {
|
|
||||||
return normalizedTitle === normalizedQuery;
|
|
||||||
} else {
|
|
||||||
return (
|
|
||||||
`${account.code} ${normalizedTitle}`.indexOf(normalizedQuery) >= 0
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[],
|
|
||||||
);
|
|
||||||
|
|
||||||
// Account item of select accounts field.
|
// Account item of select accounts field.
|
||||||
const accountItem = useCallback((item, { handleClick, modifiers, query }) => {
|
const accountItem = useCallback((item, { handleClick, modifiers, query }) => {
|
||||||
return (
|
return (
|
||||||
@@ -98,28 +129,31 @@ export default function AccountsSuggestField({
|
|||||||
);
|
);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleInputValueRenderer = (inputValue) => {
|
const handleAccountSelect = useCallback(
|
||||||
if (inputValue) {
|
|
||||||
return inputValue.name.toString();
|
|
||||||
}
|
|
||||||
return '';
|
|
||||||
};
|
|
||||||
|
|
||||||
const onAccountSelect = useCallback(
|
|
||||||
(account) => {
|
(account) => {
|
||||||
setSelectedAccount({ ...account });
|
if (account.id) {
|
||||||
onAccountSelected && onAccountSelected(account);
|
setSelectedAccount({ ...account });
|
||||||
|
onAccountSelected && onAccountSelected(account);
|
||||||
|
} else {
|
||||||
|
openDialog('account-form');
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[setSelectedAccount, onAccountSelected],
|
[setSelectedAccount, onAccountSelected, openDialog],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Maybe inject new item props to select component.
|
||||||
|
const maybeCreateNewItemRenderer = allowCreate ? createNewItemRenderer : null;
|
||||||
|
const maybeCreateNewItemFromQuery = allowCreate
|
||||||
|
? createNewItemFromQuery
|
||||||
|
: null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Suggest
|
<Suggest
|
||||||
items={filteredAccounts}
|
items={filteredAccounts}
|
||||||
noResults={<MenuItem disabled={true} text={<T id={'no_accounts'} />} />}
|
noResults={<MenuItem disabled={true} text={<T id={'no_accounts'} />} />}
|
||||||
itemRenderer={accountItem}
|
itemRenderer={accountItem}
|
||||||
itemPredicate={filterAccountsPredicater}
|
itemPredicate={filterAccountsPredicater}
|
||||||
onItemSelect={onAccountSelect}
|
onItemSelect={handleAccountSelect}
|
||||||
selectedItem={selectedAccount}
|
selectedItem={selectedAccount}
|
||||||
inputProps={{ placeholder: defaultSelectText }}
|
inputProps={{ placeholder: defaultSelectText }}
|
||||||
resetOnClose={true}
|
resetOnClose={true}
|
||||||
@@ -129,7 +163,11 @@ export default function AccountsSuggestField({
|
|||||||
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
||||||
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
})}
|
})}
|
||||||
|
createNewItemRenderer={maybeCreateNewItemRenderer}
|
||||||
|
createNewItemFromQuery={maybeCreateNewItemFromQuery}
|
||||||
{...suggestProps}
|
{...suggestProps}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export default R.compose(withDialogActions)(AccountsSuggestField);
|
||||||
|
|||||||
@@ -1,17 +1,60 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import clsx from 'classnames';
|
import clsx from 'classnames';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
import Style from './style.module.scss';
|
export function Alert({ title, description, children, intent, className }) {
|
||||||
|
|
||||||
export function Alert({ title, description, intent }) {
|
|
||||||
return (
|
return (
|
||||||
<div
|
<AlertRoot className={clsx(className)} intent={intent}>
|
||||||
className={clsx(Style.root, {
|
{title && <AlertTitle>{title}</AlertTitle>}
|
||||||
[`${Style['root_' + intent]}`]: intent,
|
{description && <AlertDesc>{description}</AlertDesc>}
|
||||||
})}
|
{children && <AlertDesc>{children}</AlertDesc>}
|
||||||
>
|
</AlertRoot>
|
||||||
{title && <h3 className={clsx(Style.title)}>{title}</h3>}
|
|
||||||
{description && <p class={clsx(Style.description)}>{description}</p>}
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const AlertRoot = styled.div`
|
||||||
|
border: 1px solid rgb(223, 227, 230);
|
||||||
|
padding: 12px;
|
||||||
|
border-radius: 6px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.intent === 'danger' &&
|
||||||
|
`
|
||||||
|
border-color: rgb(249, 198, 198);
|
||||||
|
background: rgb(255, 248, 248);
|
||||||
|
|
||||||
|
${AlertDesc} {
|
||||||
|
color: #d95759;
|
||||||
|
}
|
||||||
|
${AlertTitle} {
|
||||||
|
color: rgb(205, 43, 49);
|
||||||
|
}
|
||||||
|
`}
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.intent === 'primary' &&
|
||||||
|
`
|
||||||
|
background: #fff;
|
||||||
|
border-color: #98a8ee;
|
||||||
|
|
||||||
|
${AlertTitle} {
|
||||||
|
color: #1a3bd4;
|
||||||
|
}
|
||||||
|
${AlertDesc} {
|
||||||
|
color: #455883;
|
||||||
|
}
|
||||||
|
`}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const AlertTitle = styled.h3`
|
||||||
|
color: rgb(17, 24, 28);
|
||||||
|
margin-bottom: 4px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 600;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const AlertDesc = styled.p`
|
||||||
|
color: rgb(104, 112, 118);
|
||||||
|
margin: 0;
|
||||||
|
`;
|
||||||
|
|||||||
@@ -1,32 +0,0 @@
|
|||||||
.root {
|
|
||||||
border: 1px solid rgb(223, 227, 230);
|
|
||||||
padding: 12px;
|
|
||||||
border-radius: 6px;
|
|
||||||
margin-bottom: 20px;
|
|
||||||
|
|
||||||
&_danger {
|
|
||||||
border-color: rgb(249, 198, 198);
|
|
||||||
background: rgb(255, 248, 248);
|
|
||||||
|
|
||||||
.description {
|
|
||||||
color: #d95759;
|
|
||||||
}
|
|
||||||
|
|
||||||
.title {
|
|
||||||
color: rgb(205, 43, 49);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.title {
|
|
||||||
color: rgb(17, 24, 28);
|
|
||||||
margin-bottom: 4px;
|
|
||||||
font-size: 14px;
|
|
||||||
font-weight: 600;
|
|
||||||
}
|
|
||||||
|
|
||||||
.description {
|
|
||||||
color: rgb(104, 112, 118);
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
@@ -4,7 +4,6 @@ import styled from 'styled-components';
|
|||||||
import { Classes } from '@blueprintjs/core';
|
import { Classes } from '@blueprintjs/core';
|
||||||
import clsx from 'classnames';
|
import clsx from 'classnames';
|
||||||
import Icon from '../Icon';
|
import Icon from '../Icon';
|
||||||
import { whenRtl, whenLtr } from 'utils/styled-components';
|
|
||||||
|
|
||||||
const ACCOUNT_TYPE = {
|
const ACCOUNT_TYPE = {
|
||||||
CASH: 'cash',
|
CASH: 'cash',
|
||||||
@@ -185,9 +184,7 @@ const MetaLineValue = styled.div`
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
color: rgb(23, 43, 77);
|
color: rgb(23, 43, 77);
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
|
margin-left: auto;
|
||||||
${whenLtr(`margin-left: auto;`)}
|
|
||||||
${whenRtl(`margin-right: auto;`)}
|
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const BankAccountMeta = styled.div`
|
const BankAccountMeta = styled.div`
|
||||||
@@ -204,7 +201,5 @@ const AccountIconWrap = styled.div`
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
top: 14px;
|
top: 14px;
|
||||||
color: #abb3bb;
|
color: #abb3bb;
|
||||||
|
right: 12px;
|
||||||
${whenLtr(`right: 12px;`)}
|
|
||||||
${whenRtl(`left: 12px;`)}
|
|
||||||
`;
|
`;
|
||||||
|
|||||||
@@ -1,23 +1,14 @@
|
|||||||
import styled from 'styled-components';
|
import styled from 'styled-components';
|
||||||
import { Button } from '@blueprintjs/core';
|
|
||||||
|
|
||||||
export const ButtonLink = styled(Button)`
|
export const ButtonLink = styled.button`
|
||||||
line-height: inherit;
|
color: #0052cc;
|
||||||
|
border: 0;
|
||||||
|
background: transparent;
|
||||||
|
cursor: pointer;
|
||||||
|
text-align: inherit;
|
||||||
|
|
||||||
&.bp3-small {
|
&:hover,
|
||||||
min-height: auto;
|
&:active {
|
||||||
min-width: auto;
|
text-decoration: underline;
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
&:not([class*='bp3-intent-']) {
|
|
||||||
&,
|
|
||||||
&:hover {
|
|
||||||
color: #0052cc;
|
|
||||||
background: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
text-decoration: underline;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import classNames from 'classnames';
|
|
||||||
|
|
||||||
export default function Card({ className, children }) {
|
|
||||||
return <div className={classNames('card', className)}>{children}</div>;
|
|
||||||
}
|
|
||||||
27
src/components/Card/index.js
Normal file
27
src/components/Card/index.js
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
export function Card({ className, children }) {
|
||||||
|
return <CardRoot className={className}>{children}</CardRoot>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const CardRoot = styled.div`
|
||||||
|
padding: 15px;
|
||||||
|
margin: 15px;
|
||||||
|
background: #fff;
|
||||||
|
border: 1px solid #d2dce2;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const CardFooterActions = styled.div`
|
||||||
|
padding-top: 16px;
|
||||||
|
border-top: 1px solid #e0e7ea;
|
||||||
|
margin-top: 30px;
|
||||||
|
|
||||||
|
.bp3-button {
|
||||||
|
min-width: 70px;
|
||||||
|
|
||||||
|
+ .bp3-button {
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
25
src/components/CommercialDoc/index.js
Normal file
25
src/components/CommercialDoc/index.js
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import styled from 'styled-components';
|
||||||
|
import { Card } from '../Card';
|
||||||
|
import DataTable from '../DataTable';
|
||||||
|
|
||||||
|
export const CommercialDocBox = styled(Card)`
|
||||||
|
padding: 22px 20px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const CommercialDocHeader = styled.div`
|
||||||
|
margin-bottom: 25px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const CommercialDocTopHeader = styled.div`
|
||||||
|
margin-bottom: 25px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const CommercialDocEntriesTable = styled(DataTable)`
|
||||||
|
.tbody .tr:last-child .td {
|
||||||
|
border-bottom: 1px solid #d2dce2;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const CommercialDocFooter = styled.div`
|
||||||
|
margin-top: 25px;
|
||||||
|
`;
|
||||||
@@ -11,12 +11,14 @@ export default function ContactSelecetList({
|
|||||||
contactsList,
|
contactsList,
|
||||||
initialContactId,
|
initialContactId,
|
||||||
selectedContactId,
|
selectedContactId,
|
||||||
selectedContactType,
|
createNewItemFrom,
|
||||||
defaultSelectText = <T id={'select_contact'} />,
|
defaultSelectText = <T id={'select_contact'} />,
|
||||||
onContactSelected,
|
onContactSelected,
|
||||||
popoverFill = false,
|
popoverFill = false,
|
||||||
disabled = false,
|
disabled = false,
|
||||||
buttonProps,
|
buttonProps,
|
||||||
|
|
||||||
|
...restProps
|
||||||
}) {
|
}) {
|
||||||
const contacts = useMemo(
|
const contacts = useMemo(
|
||||||
() =>
|
() =>
|
||||||
@@ -65,7 +67,7 @@ export default function ContactSelecetList({
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Filter Contact List
|
// Filter Contact List
|
||||||
const filterContacts = (query, contact, index, exactMatch) => {
|
const itemPredicate = (query, contact, index, exactMatch) => {
|
||||||
const normalizedTitle = contact.display_name.toLowerCase();
|
const normalizedTitle = contact.display_name.toLowerCase();
|
||||||
const normalizedQuery = query.toLowerCase();
|
const normalizedQuery = query.toLowerCase();
|
||||||
if (exactMatch) {
|
if (exactMatch) {
|
||||||
@@ -83,7 +85,7 @@ export default function ContactSelecetList({
|
|||||||
items={contacts}
|
items={contacts}
|
||||||
noResults={<MenuItem disabled={true} text={<T id={'no_results'} />} />}
|
noResults={<MenuItem disabled={true} text={<T id={'no_results'} />} />}
|
||||||
itemRenderer={handleContactRenderer}
|
itemRenderer={handleContactRenderer}
|
||||||
itemPredicate={filterContacts}
|
itemPredicate={itemPredicate}
|
||||||
filterable={true}
|
filterable={true}
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
onItemSelect={onContactSelect}
|
onItemSelect={onContactSelect}
|
||||||
@@ -92,8 +94,9 @@ export default function ContactSelecetList({
|
|||||||
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
})}
|
})}
|
||||||
inputProps={{
|
inputProps={{
|
||||||
placeholder: intl.get('filter_')
|
placeholder: intl.get('filter_'),
|
||||||
}}
|
}}
|
||||||
|
{...restProps}
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
|
|||||||
86
src/components/Contacts/ContactSelectField.js
Normal file
86
src/components/Contacts/ContactSelectField.js
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
import React, { useCallback, useState, useEffect, useMemo } from 'react';
|
||||||
|
import { FormattedMessage as T } from 'components';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
|
||||||
|
import { MenuItem, Button } from '@blueprintjs/core';
|
||||||
|
import { Select } from '@blueprintjs/select';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { CLASSES } from 'common/classes';
|
||||||
|
|
||||||
|
import { itemPredicate, handleContactRenderer } from './utils';
|
||||||
|
|
||||||
|
export default function ContactSelectField({
|
||||||
|
contacts,
|
||||||
|
initialContactId,
|
||||||
|
selectedContactId,
|
||||||
|
defaultSelectText = <T id={'select_contact'} />,
|
||||||
|
onContactSelected,
|
||||||
|
popoverFill = false,
|
||||||
|
disabled = false,
|
||||||
|
buttonProps,
|
||||||
|
|
||||||
|
...restProps
|
||||||
|
}) {
|
||||||
|
const localContacts = useMemo(
|
||||||
|
() =>
|
||||||
|
contacts.map((contact) => ({
|
||||||
|
...contact,
|
||||||
|
_id: `${contact.id}_${contact.contact_type}`,
|
||||||
|
})),
|
||||||
|
[contacts],
|
||||||
|
);
|
||||||
|
|
||||||
|
const initialContact = useMemo(
|
||||||
|
() => contacts.find((a) => a.id === initialContactId),
|
||||||
|
[initialContactId, contacts],
|
||||||
|
);
|
||||||
|
|
||||||
|
const [selecetedContact, setSelectedContact] = useState(
|
||||||
|
initialContact || null,
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (typeof selectedContactId !== 'undefined') {
|
||||||
|
const account = selectedContactId
|
||||||
|
? contacts.find((a) => a.id === selectedContactId)
|
||||||
|
: null;
|
||||||
|
setSelectedContact(account);
|
||||||
|
}
|
||||||
|
}, [selectedContactId, contacts, setSelectedContact]);
|
||||||
|
|
||||||
|
const handleContactSelect = useCallback(
|
||||||
|
(contact) => {
|
||||||
|
setSelectedContact({ ...contact });
|
||||||
|
onContactSelected && onContactSelected(contact);
|
||||||
|
},
|
||||||
|
[setSelectedContact, onContactSelected],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Select
|
||||||
|
items={localContacts}
|
||||||
|
noResults={<MenuItem disabled={true} text={<T id={'no_results'} />} />}
|
||||||
|
itemRenderer={handleContactRenderer}
|
||||||
|
itemPredicate={itemPredicate}
|
||||||
|
filterable={true}
|
||||||
|
disabled={disabled}
|
||||||
|
onItemSelect={handleContactSelect}
|
||||||
|
popoverProps={{ minimal: true, usePortal: !popoverFill }}
|
||||||
|
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
||||||
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
|
})}
|
||||||
|
inputProps={{
|
||||||
|
placeholder: intl.get('filter_'),
|
||||||
|
}}
|
||||||
|
{...restProps}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
disabled={disabled}
|
||||||
|
text={
|
||||||
|
selecetedContact ? selecetedContact.display_name : defaultSelectText
|
||||||
|
}
|
||||||
|
{...buttonProps}
|
||||||
|
/>
|
||||||
|
</Select>
|
||||||
|
);
|
||||||
|
}
|
||||||
116
src/components/Contacts/CustomerSelectField.js
Normal file
116
src/components/Contacts/CustomerSelectField.js
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
import React, { useCallback, useState, useEffect, useMemo } from 'react';
|
||||||
|
import { FormattedMessage as T } from 'components';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
|
import { MenuItem, Button } from '@blueprintjs/core';
|
||||||
|
import { Select } from '@blueprintjs/select';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { CLASSES } from 'common/classes';
|
||||||
|
|
||||||
|
import {
|
||||||
|
itemPredicate,
|
||||||
|
handleContactRenderer,
|
||||||
|
createNewItemRenderer,
|
||||||
|
createNewItemFromQuery,
|
||||||
|
} from './utils';
|
||||||
|
|
||||||
|
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
||||||
|
|
||||||
|
import { DRAWERS } from 'common/drawers';
|
||||||
|
|
||||||
|
function CustomerSelectField({
|
||||||
|
// #withDrawerActions
|
||||||
|
openDrawer,
|
||||||
|
|
||||||
|
// #ownProps
|
||||||
|
contacts,
|
||||||
|
initialContactId,
|
||||||
|
selectedContactId,
|
||||||
|
defaultSelectText = <T id={'select_contact'} />,
|
||||||
|
onContactSelected,
|
||||||
|
popoverFill = false,
|
||||||
|
disabled = false,
|
||||||
|
allowCreate,
|
||||||
|
buttonProps,
|
||||||
|
|
||||||
|
...restProps
|
||||||
|
}) {
|
||||||
|
const localContacts = useMemo(
|
||||||
|
() =>
|
||||||
|
contacts.map((contact) => ({
|
||||||
|
...contact,
|
||||||
|
_id: `${contact.id}_${contact.contact_type}`,
|
||||||
|
})),
|
||||||
|
[contacts],
|
||||||
|
);
|
||||||
|
|
||||||
|
const initialContact = useMemo(
|
||||||
|
() => contacts.find((a) => a.id === initialContactId),
|
||||||
|
[initialContactId, contacts],
|
||||||
|
);
|
||||||
|
|
||||||
|
const [selecetedContact, setSelectedContact] = useState(
|
||||||
|
initialContact || null,
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (typeof selectedContactId !== 'undefined') {
|
||||||
|
const account = selectedContactId
|
||||||
|
? contacts.find((a) => a.id === selectedContactId)
|
||||||
|
: null;
|
||||||
|
setSelectedContact(account);
|
||||||
|
}
|
||||||
|
}, [selectedContactId, contacts, setSelectedContact]);
|
||||||
|
|
||||||
|
const handleContactSelect = useCallback(
|
||||||
|
(contact) => {
|
||||||
|
if (contact.id) {
|
||||||
|
setSelectedContact({ ...contact });
|
||||||
|
onContactSelected && onContactSelected(contact);
|
||||||
|
} else {
|
||||||
|
openDrawer(DRAWERS.QUICK_CREATE_CUSTOMER);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[setSelectedContact, onContactSelected, openDrawer],
|
||||||
|
);
|
||||||
|
|
||||||
|
// Maybe inject create new item props to suggest component.
|
||||||
|
const maybeCreateNewItemRenderer = allowCreate ? createNewItemRenderer : null;
|
||||||
|
const maybeCreateNewItemFromQuery = allowCreate
|
||||||
|
? createNewItemFromQuery
|
||||||
|
: null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Select
|
||||||
|
items={localContacts}
|
||||||
|
noResults={<MenuItem disabled={true} text={<T id={'no_results'} />} />}
|
||||||
|
itemRenderer={handleContactRenderer}
|
||||||
|
itemPredicate={itemPredicate}
|
||||||
|
filterable={true}
|
||||||
|
disabled={disabled}
|
||||||
|
onItemSelect={handleContactSelect}
|
||||||
|
popoverProps={{ minimal: true, usePortal: !popoverFill }}
|
||||||
|
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
||||||
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
|
})}
|
||||||
|
inputProps={{
|
||||||
|
placeholder: intl.get('filter_'),
|
||||||
|
}}
|
||||||
|
createNewItemRenderer={maybeCreateNewItemRenderer}
|
||||||
|
createNewItemFromQuery={maybeCreateNewItemFromQuery}
|
||||||
|
createNewItemPosition={'top'}
|
||||||
|
{...restProps}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
disabled={disabled}
|
||||||
|
text={
|
||||||
|
selecetedContact ? selecetedContact.display_name : defaultSelectText
|
||||||
|
}
|
||||||
|
{...buttonProps}
|
||||||
|
/>
|
||||||
|
</Select>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default R.compose(withDrawerActions)(CustomerSelectField);
|
||||||
115
src/components/Contacts/VendorSelectField.js
Normal file
115
src/components/Contacts/VendorSelectField.js
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
import React, { useCallback, useState, useEffect, useMemo } from 'react';
|
||||||
|
import { FormattedMessage as T } from 'components';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
|
import { MenuItem, Button } from '@blueprintjs/core';
|
||||||
|
import { Select } from '@blueprintjs/select';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { CLASSES } from 'common/classes';
|
||||||
|
|
||||||
|
import {
|
||||||
|
itemPredicate,
|
||||||
|
handleContactRenderer,
|
||||||
|
createNewItemFromQuery,
|
||||||
|
createNewItemRenderer,
|
||||||
|
} from './utils';
|
||||||
|
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
||||||
|
|
||||||
|
import { DRAWERS } from 'common/drawers';
|
||||||
|
|
||||||
|
function VendorSelectField({
|
||||||
|
// #withDrawerActions
|
||||||
|
openDrawer,
|
||||||
|
|
||||||
|
// #ownProps
|
||||||
|
contacts,
|
||||||
|
initialContactId,
|
||||||
|
selectedContactId,
|
||||||
|
defaultSelectText = <T id={'select_contact'} />,
|
||||||
|
onContactSelected,
|
||||||
|
popoverFill = false,
|
||||||
|
disabled = false,
|
||||||
|
allowCreate,
|
||||||
|
buttonProps,
|
||||||
|
|
||||||
|
...restProps
|
||||||
|
}) {
|
||||||
|
const localContacts = useMemo(
|
||||||
|
() =>
|
||||||
|
contacts.map((contact) => ({
|
||||||
|
...contact,
|
||||||
|
_id: `${contact.id}_${contact.contact_type}`,
|
||||||
|
})),
|
||||||
|
[contacts],
|
||||||
|
);
|
||||||
|
|
||||||
|
const initialContact = useMemo(
|
||||||
|
() => contacts.find((a) => a.id === initialContactId),
|
||||||
|
[initialContactId, contacts],
|
||||||
|
);
|
||||||
|
|
||||||
|
const [selecetedContact, setSelectedContact] = useState(
|
||||||
|
initialContact || null,
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (typeof selectedContactId !== 'undefined') {
|
||||||
|
const account = selectedContactId
|
||||||
|
? contacts.find((a) => a.id === selectedContactId)
|
||||||
|
: null;
|
||||||
|
setSelectedContact(account);
|
||||||
|
}
|
||||||
|
}, [selectedContactId, contacts, setSelectedContact]);
|
||||||
|
|
||||||
|
const handleContactSelect = useCallback(
|
||||||
|
(contact) => {
|
||||||
|
if (contact.id) {
|
||||||
|
setSelectedContact({ ...contact });
|
||||||
|
onContactSelected && onContactSelected(contact);
|
||||||
|
} else {
|
||||||
|
openDrawer(DRAWERS.QUICK_WRITE_VENDOR);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[setSelectedContact, onContactSelected, openDrawer],
|
||||||
|
);
|
||||||
|
|
||||||
|
// Maybe inject create new item props to suggest component.
|
||||||
|
const maybeCreateNewItemRenderer = allowCreate ? createNewItemRenderer : null;
|
||||||
|
const maybeCreateNewItemFromQuery = allowCreate
|
||||||
|
? createNewItemFromQuery
|
||||||
|
: null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Select
|
||||||
|
items={localContacts}
|
||||||
|
noResults={<MenuItem disabled={true} text={<T id={'no_results'} />} />}
|
||||||
|
itemRenderer={handleContactRenderer}
|
||||||
|
itemPredicate={itemPredicate}
|
||||||
|
filterable={true}
|
||||||
|
disabled={disabled}
|
||||||
|
onItemSelect={handleContactSelect}
|
||||||
|
popoverProps={{ minimal: true, usePortal: !popoverFill }}
|
||||||
|
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
||||||
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
|
})}
|
||||||
|
inputProps={{
|
||||||
|
placeholder: intl.get('filter_'),
|
||||||
|
}}
|
||||||
|
createNewItemRenderer={maybeCreateNewItemRenderer}
|
||||||
|
createNewItemFromQuery={maybeCreateNewItemFromQuery}
|
||||||
|
createNewItemPosition={'top'}
|
||||||
|
{...restProps}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
disabled={disabled}
|
||||||
|
text={
|
||||||
|
selecetedContact ? selecetedContact.display_name : defaultSelectText
|
||||||
|
}
|
||||||
|
{...buttonProps}
|
||||||
|
/>
|
||||||
|
</Select>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default R.compose(withDrawerActions)(VendorSelectField);
|
||||||
5
src/components/Contacts/index.js
Normal file
5
src/components/Contacts/index.js
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import ContactSelectField from './ContactSelectField';
|
||||||
|
import CustomerSelectField from './CustomerSelectField';
|
||||||
|
import VendorSelectField from './VendorSelectField';
|
||||||
|
|
||||||
|
export { ContactSelectField, CustomerSelectField, VendorSelectField };
|
||||||
44
src/components/Contacts/utils.js
Normal file
44
src/components/Contacts/utils.js
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import { MenuItem } from '@blueprintjs/core';
|
||||||
|
|
||||||
|
// Filter Contact List
|
||||||
|
export const itemPredicate = (query, contact, index, exactMatch) => {
|
||||||
|
const normalizedTitle = contact.display_name.toLowerCase();
|
||||||
|
const normalizedQuery = query.toLowerCase();
|
||||||
|
if (exactMatch) {
|
||||||
|
return normalizedTitle === normalizedQuery;
|
||||||
|
} else {
|
||||||
|
return (
|
||||||
|
`${contact.display_name} ${normalizedTitle}`.indexOf(normalizedQuery) >= 0
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const handleContactRenderer = (contact, { handleClick }) => (
|
||||||
|
<MenuItem
|
||||||
|
key={contact.id}
|
||||||
|
text={contact.display_name}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
// Creates a new item from query.
|
||||||
|
export const createNewItemFromQuery = (name) => {
|
||||||
|
return {
|
||||||
|
name,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle quick create new customer.
|
||||||
|
export const createNewItemRenderer = (query, active, handleClick) => {
|
||||||
|
return (
|
||||||
|
<MenuItem
|
||||||
|
icon="add"
|
||||||
|
text={intl.get('list.create', { value: `"${query}"` })}
|
||||||
|
active={active}
|
||||||
|
shouldDismissPopover={false}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
25
src/components/Customers/CustomerDrawerLink.js
Normal file
25
src/components/Customers/CustomerDrawerLink.js
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
|
import { ButtonLink } from 'components';
|
||||||
|
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
||||||
|
|
||||||
|
function CustomerDrawerLinkComponent({
|
||||||
|
// #ownProps
|
||||||
|
children,
|
||||||
|
customerId,
|
||||||
|
|
||||||
|
// #withDrawerActions
|
||||||
|
openDrawer,
|
||||||
|
}) {
|
||||||
|
// Handle view customer drawer.
|
||||||
|
const handleCustomerDrawer = () => {
|
||||||
|
openDrawer('customer-details-drawer', { customerId });
|
||||||
|
};
|
||||||
|
|
||||||
|
return <ButtonLink onClick={handleCustomerDrawer}>{children}</ButtonLink>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const CustomerDrawerLink = R.compose(withDrawerActions)(
|
||||||
|
CustomerDrawerLinkComponent,
|
||||||
|
);
|
||||||
1
src/components/Customers/index.js
Normal file
1
src/components/Customers/index.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './CustomerDrawerLink';
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import { useUser } from 'hooks/query';
|
|
||||||
import withAuthentication from '../../containers/Authentication/withAuthentication';
|
|
||||||
|
|
||||||
const AuthenticatedUserContext = React.createContext();
|
|
||||||
|
|
||||||
function AuthenticatedUserComponent({ authenticatedUserId, children }) {
|
|
||||||
const { data: user, ...restProps } = useUser(authenticatedUserId);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<AuthenticatedUserContext.Provider
|
|
||||||
value={{
|
|
||||||
user,
|
|
||||||
...restProps,
|
|
||||||
}}
|
|
||||||
children={children}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export const AuthenticatedUser = withAuthentication(
|
|
||||||
({ authenticatedUserId }) => ({
|
|
||||||
authenticatedUserId,
|
|
||||||
}),
|
|
||||||
)(AuthenticatedUserComponent);
|
|
||||||
|
|
||||||
export const useAuthenticatedUser = () =>
|
|
||||||
React.useContext(AuthenticatedUserContext);
|
|
||||||
25
src/components/Dashboard/DashboardAbilityProvider.js
Normal file
25
src/components/Dashboard/DashboardAbilityProvider.js
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Ability } from '@casl/ability';
|
||||||
|
import { createContextualCan } from '@casl/react';
|
||||||
|
import { useDashboardMeta } from '../../hooks/query';
|
||||||
|
|
||||||
|
export const AbilityContext = React.createContext();
|
||||||
|
export const Can = createContextualCan(AbilityContext.Consumer);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard ability provider.
|
||||||
|
*/
|
||||||
|
export function DashboardAbilityProvider({ children }) {
|
||||||
|
const {
|
||||||
|
data: { abilities },
|
||||||
|
} = useDashboardMeta();
|
||||||
|
|
||||||
|
// Ability instance.
|
||||||
|
const ability = new Ability(abilities);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<AbilityContext.Provider value={ability}>
|
||||||
|
{children}
|
||||||
|
</AbilityContext.Provider>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,16 +1,19 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import classnames from 'classnames';
|
import clsx from 'classnames';
|
||||||
import { Navbar } from '@blueprintjs/core';
|
import { Navbar } from '@blueprintjs/core';
|
||||||
|
|
||||||
export default function DashboardActionsBar({ children, name }) {
|
export default function DashboardActionsBar({ className, children, name }) {
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={classnames({
|
className={clsx(
|
||||||
'dashboard__actions-bar': true,
|
{
|
||||||
[`dashboard__actions-bar--${name}`]: !!name
|
'dashboard__actions-bar': true,
|
||||||
})}
|
[`dashboard__actions-bar--${name}`]: !!name,
|
||||||
|
},
|
||||||
|
className,
|
||||||
|
)}
|
||||||
>
|
>
|
||||||
<Navbar className='navbar--dashboard-actions-bar'>{children}</Navbar>
|
<Navbar className="navbar--dashboard-actions-bar">{children}</Navbar>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +1,53 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import * as R from 'ramda';
|
import {
|
||||||
|
useAuthenticatedAccount,
|
||||||
import { useUser, useCurrentOrganization } from '../../hooks/query';
|
useCurrentOrganization,
|
||||||
|
useDashboardMeta,
|
||||||
|
} from '../../hooks/query';
|
||||||
import { useSplashLoading } from '../../hooks/state';
|
import { useSplashLoading } from '../../hooks/state';
|
||||||
import { useWatch, useWatchImmediate, useWhen } from '../../hooks';
|
import { useWatch, useWatchImmediate, useWhen } from '../../hooks';
|
||||||
|
|
||||||
import withAuthentication from '../../containers/Authentication/withAuthentication';
|
|
||||||
|
|
||||||
import { setCookie, getCookie } from '../../utils';
|
import { setCookie, getCookie } from '../../utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dashboard async booting.
|
* Dashboard meta async booting.
|
||||||
*/
|
*/
|
||||||
function DashboardBootJSX({ authenticatedUserId }) {
|
export function useDashboardMetaBoot() {
|
||||||
|
const {
|
||||||
|
data: dashboardMeta,
|
||||||
|
isLoading: isDashboardMetaLoading,
|
||||||
|
isSuccess: isDashboardMetaSuccess,
|
||||||
|
} = useDashboardMeta({
|
||||||
|
keepPreviousData: true,
|
||||||
|
});
|
||||||
|
const [startLoading, stopLoading] = useSplashLoading();
|
||||||
|
|
||||||
|
useWatchImmediate((value) => {
|
||||||
|
value && startLoading();
|
||||||
|
}, isDashboardMetaLoading);
|
||||||
|
|
||||||
|
useWatchImmediate(() => {
|
||||||
|
isDashboardMetaSuccess && stopLoading();
|
||||||
|
}, isDashboardMetaSuccess);
|
||||||
|
|
||||||
|
return {
|
||||||
|
isLoading: isDashboardMetaLoading,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard async booting.
|
||||||
|
* @returns {{ isLoading: boolean }}
|
||||||
|
*/
|
||||||
|
export function useDashboardBoot() {
|
||||||
|
const { isLoading } = useDashboardMetaBoot();
|
||||||
|
|
||||||
|
return { isLoading };
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Application async booting.
|
||||||
|
*/
|
||||||
|
export function useApplicationBoot() {
|
||||||
// Fetches the current user's organization.
|
// Fetches the current user's organization.
|
||||||
const {
|
const {
|
||||||
isSuccess: isCurrentOrganizationSuccess,
|
isSuccess: isCurrentOrganizationSuccess,
|
||||||
@@ -22,7 +57,7 @@ function DashboardBootJSX({ authenticatedUserId }) {
|
|||||||
|
|
||||||
// Authenticated user.
|
// Authenticated user.
|
||||||
const { isSuccess: isAuthUserSuccess, isLoading: isAuthUserLoading } =
|
const { isSuccess: isAuthUserSuccess, isLoading: isAuthUserLoading } =
|
||||||
useUser(authenticatedUserId);
|
useAuthenticatedAccount();
|
||||||
|
|
||||||
// Initial locale cookie value.
|
// Initial locale cookie value.
|
||||||
const localeCookie = getCookie('locale');
|
const localeCookie = getCookie('locale');
|
||||||
@@ -86,11 +121,8 @@ function DashboardBootJSX({ authenticatedUserId }) {
|
|||||||
isBooted.current = true;
|
isBooted.current = true;
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const DashboardBoot = R.compose(
|
return {
|
||||||
withAuthentication(({ authenticatedUserId }) => ({
|
isLoading: isOrgLoading || isAuthUserLoading,
|
||||||
authenticatedUserId,
|
};
|
||||||
})),
|
}
|
||||||
)(DashboardBootJSX);
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import React from 'react';
|
|||||||
import { ErrorBoundary } from 'react-error-boundary';
|
import { ErrorBoundary } from 'react-error-boundary';
|
||||||
import DashboardTopbar from 'components/Dashboard/DashboardTopbar';
|
import DashboardTopbar from 'components/Dashboard/DashboardTopbar';
|
||||||
import DashboardContentRoutes from 'components/Dashboard/DashboardContentRoute';
|
import DashboardContentRoutes from 'components/Dashboard/DashboardContentRoute';
|
||||||
import DashboardFooter from 'components/Dashboard/DashboardFooter';
|
|
||||||
import DashboardErrorBoundary from './DashboardErrorBoundary';
|
import DashboardErrorBoundary from './DashboardErrorBoundary';
|
||||||
|
|
||||||
export default React.forwardRef(({}, ref) => {
|
export default React.forwardRef(({}, ref) => {
|
||||||
@@ -11,7 +10,6 @@ export default React.forwardRef(({}, ref) => {
|
|||||||
<div className="dashboard-content" id="dashboard" ref={ref}>
|
<div className="dashboard-content" id="dashboard" ref={ref}>
|
||||||
<DashboardTopbar />
|
<DashboardTopbar />
|
||||||
<DashboardContentRoutes />
|
<DashboardContentRoutes />
|
||||||
<DashboardFooter />
|
|
||||||
</div>
|
</div>
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,8 +1,16 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { DashboardAbilityProvider } from '../../components';
|
||||||
|
import { useDashboardBoot } from './DashboardBoot';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dashboard provider.
|
* Dashboard provider.
|
||||||
*/
|
*/
|
||||||
export default function DashboardProvider({ children }) {
|
export default function DashboardProvider({ children }) {
|
||||||
return children;
|
const { isLoading } = useDashboardBoot();
|
||||||
|
|
||||||
|
// Avoid display any dashboard component before complete booting.
|
||||||
|
if (isLoading) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return <DashboardAbilityProvider>{children}</DashboardAbilityProvider>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,16 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { ThemeProvider } from 'styled-components';
|
import { ThemeProvider, StyleSheetManager } from 'styled-components';
|
||||||
|
import rtlcss from 'stylis-rtlcss';
|
||||||
import { useAppIntlContext } from '../AppIntlProvider';
|
import { useAppIntlContext } from '../AppIntlProvider';
|
||||||
|
|
||||||
export function DashboardThemeProvider({ children }) {
|
export function DashboardThemeProvider({ children }) {
|
||||||
const { direction } = useAppIntlContext();
|
const { direction } = useAppIntlContext();
|
||||||
|
|
||||||
return <ThemeProvider theme={{ dir: direction }}>{children}</ThemeProvider>;
|
return (
|
||||||
|
<StyleSheetManager
|
||||||
|
{...(direction === 'rtl' ? { stylisPlugins: [rtlcss] } : {})}
|
||||||
|
>
|
||||||
|
<ThemeProvider theme={{ dir: direction }}>{children}</ThemeProvider>
|
||||||
|
</StyleSheetManager>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ import withDashboard from 'containers/Dashboard/withDashboard';
|
|||||||
import QuickNewDropdown from 'containers/QuickNewDropdown/QuickNewDropdown';
|
import QuickNewDropdown from 'containers/QuickNewDropdown/QuickNewDropdown';
|
||||||
import { compose } from 'utils';
|
import { compose } from 'utils';
|
||||||
import withSubscriptions from '../../containers/Subscriptions/withSubscriptions';
|
import withSubscriptions from '../../containers/Subscriptions/withSubscriptions';
|
||||||
|
import { useGetUniversalSearchTypeOptions } from '../../containers/UniversalSearch/utils';
|
||||||
|
|
||||||
function DashboardTopbarSubscriptionMessage() {
|
function DashboardTopbarSubscriptionMessage() {
|
||||||
return (
|
return (
|
||||||
@@ -142,11 +143,8 @@ function DashboardTopbar({
|
|||||||
<Navbar class="dashboard__topbar-navbar">
|
<Navbar class="dashboard__topbar-navbar">
|
||||||
<NavbarGroup>
|
<NavbarGroup>
|
||||||
<If condition={isSubscriptionActive}>
|
<If condition={isSubscriptionActive}>
|
||||||
<Button
|
<DashboardQuickSearchButton
|
||||||
onClick={() => openGlobalSearch(true)}
|
onClick={() => openGlobalSearch(true)}
|
||||||
className={Classes.MINIMAL}
|
|
||||||
icon={<Icon icon={'search-24'} iconSize={20} />}
|
|
||||||
text={<T id={'quick_find'} />}
|
|
||||||
/>
|
/>
|
||||||
<QuickNewDropdown />
|
<QuickNewDropdown />
|
||||||
|
|
||||||
@@ -195,3 +193,23 @@ export default compose(
|
|||||||
'main',
|
'main',
|
||||||
),
|
),
|
||||||
)(DashboardTopbar);
|
)(DashboardTopbar);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard quick search button.
|
||||||
|
*/
|
||||||
|
function DashboardQuickSearchButton({ ...rest }) {
|
||||||
|
const searchTypeOptions = useGetUniversalSearchTypeOptions();
|
||||||
|
|
||||||
|
// Can't continue if there is no any search type option.
|
||||||
|
if (searchTypeOptions.length <= 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
className={Classes.MINIMAL}
|
||||||
|
icon={<Icon icon={'search-24'} iconSize={20} />}
|
||||||
|
text={<T id={'quick_find'} />}
|
||||||
|
{...rest}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,31 +1,15 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import * as R from 'ramda';
|
|
||||||
|
|
||||||
import { AuthenticatedUser } from './AuthenticatedUser';
|
import { useApplicationBoot } from '../../components';
|
||||||
import { DashboardBoot } from '../../components';
|
|
||||||
|
|
||||||
import withDashboard from '../../containers/Dashboard/withDashboard';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Private pages provider.
|
* Private pages provider.
|
||||||
*/
|
*/
|
||||||
function PrivatePagesProviderComponent({
|
export function PrivatePagesProvider({
|
||||||
splashScreenCompleted,
|
|
||||||
|
|
||||||
// #ownProps
|
// #ownProps
|
||||||
children,
|
children,
|
||||||
}) {
|
}) {
|
||||||
return (
|
const { isLoading } = useApplicationBoot();
|
||||||
<AuthenticatedUser>
|
|
||||||
<DashboardBoot />
|
|
||||||
|
|
||||||
{splashScreenCompleted ? children : null}
|
return <React.Fragment>{!isLoading ? children : null}</React.Fragment>;
|
||||||
</AuthenticatedUser>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const PrivatePagesProvider = R.compose(
|
|
||||||
withDashboard(({ splashScreenCompleted }) => ({
|
|
||||||
splashScreenCompleted,
|
|
||||||
})),
|
|
||||||
)(PrivatePagesProviderComponent);
|
|
||||||
|
|||||||
@@ -14,10 +14,14 @@ import { firstLettersArgs } from 'utils';
|
|||||||
import { useAuthActions } from 'hooks/state';
|
import { useAuthActions } from 'hooks/state';
|
||||||
|
|
||||||
import withDialogActions from 'containers/Dialog/withDialogActions';
|
import withDialogActions from 'containers/Dialog/withDialogActions';
|
||||||
import { compose } from 'utils';
|
|
||||||
import withSubscriptions from '../../containers/Subscriptions/withSubscriptions';
|
import withSubscriptions from '../../containers/Subscriptions/withSubscriptions';
|
||||||
import { useAuthenticatedUser } from './AuthenticatedUser';
|
|
||||||
|
|
||||||
|
import { useAuthenticatedAccount } from 'hooks/query'
|
||||||
|
import { compose } from 'utils';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard topbar user.
|
||||||
|
*/
|
||||||
function DashboardTopbarUser({
|
function DashboardTopbarUser({
|
||||||
openDialog,
|
openDialog,
|
||||||
|
|
||||||
@@ -28,7 +32,7 @@ function DashboardTopbarUser({
|
|||||||
const { setLogout } = useAuthActions();
|
const { setLogout } = useAuthActions();
|
||||||
|
|
||||||
// Retrieve authenticated user information.
|
// Retrieve authenticated user information.
|
||||||
const { user } = useAuthenticatedUser();
|
const { data: user } = useAuthenticatedAccount();
|
||||||
|
|
||||||
const onClickLogout = () => {
|
const onClickLogout = () => {
|
||||||
setLogout();
|
setLogout();
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
export * from './SplashScreen';
|
export * from './SplashScreen';
|
||||||
export * from './DashboardBoot';
|
export * from './DashboardBoot';
|
||||||
export * from './DashboardThemeProvider';
|
export * from './DashboardThemeProvider';
|
||||||
|
export * from './DashboardAbilityProvider';
|
||||||
@@ -17,6 +17,8 @@ export default function AccountCellRenderer({
|
|||||||
accountsDataProp,
|
accountsDataProp,
|
||||||
filterAccountsByRootTypes,
|
filterAccountsByRootTypes,
|
||||||
filterAccountsByTypes,
|
filterAccountsByTypes,
|
||||||
|
fieldProps,
|
||||||
|
formGroupProps,
|
||||||
},
|
},
|
||||||
row: { index, original },
|
row: { index, original },
|
||||||
cell: { value: initialValue },
|
cell: { value: initialValue },
|
||||||
@@ -53,6 +55,7 @@ export default function AccountCellRenderer({
|
|||||||
'form-group--account',
|
'form-group--account',
|
||||||
Classes.FILL,
|
Classes.FILL,
|
||||||
)}
|
)}
|
||||||
|
{...formGroupProps}
|
||||||
>
|
>
|
||||||
<AccountsSuggestField
|
<AccountsSuggestField
|
||||||
accounts={accounts}
|
accounts={accounts}
|
||||||
@@ -66,6 +69,7 @@ export default function AccountCellRenderer({
|
|||||||
}}
|
}}
|
||||||
openOnKeyDown={true}
|
openOnKeyDown={true}
|
||||||
blurOnSelectClose={false}
|
blurOnSelectClose={false}
|
||||||
|
{...fieldProps}
|
||||||
/>
|
/>
|
||||||
</FormGroup>
|
</FormGroup>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,15 +1,17 @@
|
|||||||
import React, { useCallback, useRef } from 'react';
|
import React, { useCallback, useRef } from 'react';
|
||||||
// import ItemsListField from 'components/ItemsListField';
|
|
||||||
import ItemsSuggestField from 'components/ItemsSuggestField';
|
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
|
||||||
import { FormGroup, Classes, Intent } from '@blueprintjs/core';
|
import { FormGroup, Classes, Intent } from '@blueprintjs/core';
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
|
|
||||||
|
import ItemsSuggestField from 'components/ItemsSuggestField';
|
||||||
|
|
||||||
import { useCellAutoFocus } from 'hooks';
|
import { useCellAutoFocus } from 'hooks';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Items list cell.
|
||||||
|
*/
|
||||||
export default function ItemsListCell({
|
export default function ItemsListCell({
|
||||||
column: { id, filterSellable, filterPurchasable },
|
column: { id, filterSellable, filterPurchasable, fieldProps, formGroupProps },
|
||||||
row: { index },
|
row: { index },
|
||||||
cell: { value: initialValue },
|
cell: { value: initialValue },
|
||||||
payload: { items, updateData, errors, autoFocus },
|
payload: { items, updateData, errors, autoFocus },
|
||||||
@@ -19,6 +21,7 @@ export default function ItemsListCell({
|
|||||||
// Auto-focus the items list input field.
|
// Auto-focus the items list input field.
|
||||||
useCellAutoFocus(fieldRef, autoFocus, id, index);
|
useCellAutoFocus(fieldRef, autoFocus, id, index);
|
||||||
|
|
||||||
|
// Handle the item selected.
|
||||||
const handleItemSelected = useCallback(
|
const handleItemSelected = useCallback(
|
||||||
(item) => {
|
(item) => {
|
||||||
updateData(index, id, item.id);
|
updateData(index, id, item.id);
|
||||||
@@ -32,6 +35,7 @@ export default function ItemsListCell({
|
|||||||
<FormGroup
|
<FormGroup
|
||||||
intent={error ? Intent.DANGER : null}
|
intent={error ? Intent.DANGER : null}
|
||||||
className={classNames('form-group--select-list', Classes.FILL)}
|
className={classNames('form-group--select-list', Classes.FILL)}
|
||||||
|
{...formGroupProps}
|
||||||
>
|
>
|
||||||
<ItemsSuggestField
|
<ItemsSuggestField
|
||||||
items={items}
|
items={items}
|
||||||
@@ -45,6 +49,7 @@ export default function ItemsListCell({
|
|||||||
}}
|
}}
|
||||||
openOnKeyDown={true}
|
openOnKeyDown={true}
|
||||||
blurOnSelectClose={false}
|
blurOnSelectClose={false}
|
||||||
|
{...fieldProps}
|
||||||
/>
|
/>
|
||||||
</FormGroup>
|
</FormGroup>
|
||||||
);
|
);
|
||||||
|
|||||||
51
src/components/DataTableCells/SwitchFieldCell.js
Normal file
51
src/components/DataTableCells/SwitchFieldCell.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { Classes, Switch, FormGroup, Intent } from '@blueprintjs/core';
|
||||||
|
|
||||||
|
import { safeInvoke } from 'utils';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Switch editable cell.
|
||||||
|
*/
|
||||||
|
const SwitchEditableCell = ({
|
||||||
|
row: { index, original },
|
||||||
|
column: { id, switchProps, onSwitchChange },
|
||||||
|
cell: { value: initialValue },
|
||||||
|
payload,
|
||||||
|
}) => {
|
||||||
|
const [value, setValue] = React.useState(initialValue);
|
||||||
|
|
||||||
|
// Handle the switch change.
|
||||||
|
const onChange = (e) => {
|
||||||
|
const newValue = e.target.checked;
|
||||||
|
|
||||||
|
setValue(newValue);
|
||||||
|
|
||||||
|
safeInvoke(payload.updateData, index, id, newValue);
|
||||||
|
safeInvoke(onSwitchChange, e, newValue, original);
|
||||||
|
};
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
setValue(initialValue);
|
||||||
|
}, [initialValue]);
|
||||||
|
|
||||||
|
const error = payload.errors?.[index]?.[id];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<FormGroup
|
||||||
|
intent={error ? Intent.DANGER : null}
|
||||||
|
className={classNames(Classes.FILL)}
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
checked={initialValue}
|
||||||
|
minimal={true}
|
||||||
|
className="ml2"
|
||||||
|
{...switchProps}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SwitchEditableCell;
|
||||||
42
src/components/DataTableCells/TextAreaCell.js
Normal file
42
src/components/DataTableCells/TextAreaCell.js
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
import React, { useState, useEffect } from 'react';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { Classes, TextArea, FormGroup, Intent } from '@blueprintjs/core';
|
||||||
|
|
||||||
|
const TextAreaEditableCell = ({
|
||||||
|
row: { index },
|
||||||
|
column: { id },
|
||||||
|
cell: { value: initialValue },
|
||||||
|
payload,
|
||||||
|
}) => {
|
||||||
|
const [value, setValue] = useState(initialValue);
|
||||||
|
|
||||||
|
const onChange = (e) => {
|
||||||
|
setValue(e.target.value);
|
||||||
|
};
|
||||||
|
const onBlur = () => {
|
||||||
|
payload.updateData(index, id, value);
|
||||||
|
};
|
||||||
|
useEffect(() => {
|
||||||
|
setValue(initialValue);
|
||||||
|
}, [initialValue]);
|
||||||
|
|
||||||
|
const error = payload.errors?.[index]?.[id];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<FormGroup
|
||||||
|
intent={error ? Intent.DANGER : null}
|
||||||
|
className={classNames(Classes.FILL)}
|
||||||
|
>
|
||||||
|
<TextArea
|
||||||
|
growVertically={true}
|
||||||
|
large={true}
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
onBlur={onBlur}
|
||||||
|
fill={true}
|
||||||
|
/>
|
||||||
|
</FormGroup>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default TextAreaEditableCell;
|
||||||
@@ -6,7 +6,9 @@ import ItemsListCell from './ItemsListCell';
|
|||||||
import PercentFieldCell from './PercentFieldCell';
|
import PercentFieldCell from './PercentFieldCell';
|
||||||
import { DivFieldCell, EmptyDiv } from './DivFieldCell';
|
import { DivFieldCell, EmptyDiv } from './DivFieldCell';
|
||||||
import NumericInputCell from './NumericInputCell';
|
import NumericInputCell from './NumericInputCell';
|
||||||
import CheckBoxFieldCell from './CheckBoxFieldCell'
|
import CheckBoxFieldCell from './CheckBoxFieldCell';
|
||||||
|
import SwitchFieldCell from './SwitchFieldCell';
|
||||||
|
import TextAreaCell from './TextAreaCell';
|
||||||
|
|
||||||
export {
|
export {
|
||||||
AccountsListFieldCell,
|
AccountsListFieldCell,
|
||||||
@@ -18,5 +20,7 @@ export {
|
|||||||
DivFieldCell,
|
DivFieldCell,
|
||||||
EmptyDiv,
|
EmptyDiv,
|
||||||
NumericInputCell,
|
NumericInputCell,
|
||||||
CheckBoxFieldCell
|
CheckBoxFieldCell,
|
||||||
|
SwitchFieldCell,
|
||||||
|
TextAreaCell,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ export default function TableWrapper({ children }) {
|
|||||||
expandable,
|
expandable,
|
||||||
virtualizedRows,
|
virtualizedRows,
|
||||||
className,
|
className,
|
||||||
|
styleName,
|
||||||
size,
|
size,
|
||||||
},
|
},
|
||||||
} = useContext(TableContext);
|
} = useContext(TableContext);
|
||||||
@@ -28,6 +29,7 @@ export default function TableWrapper({ children }) {
|
|||||||
'is-expandable': expandable,
|
'is-expandable': expandable,
|
||||||
'is-loading': loading,
|
'is-loading': loading,
|
||||||
'has-virtualized-rows': virtualizedRows,
|
'has-virtualized-rows': virtualizedRows,
|
||||||
|
[`table--${styleName}`]: styleName,
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<ScrollSync>
|
<ScrollSync>
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ const useDetailsMenuContext = () => React.useContext(DetailsMenuContext);
|
|||||||
export function DetailsMenu({
|
export function DetailsMenu({
|
||||||
children,
|
children,
|
||||||
direction = DIRECTION.VERTICAL,
|
direction = DIRECTION.VERTICAL,
|
||||||
|
textAlign,
|
||||||
minLabelSize,
|
minLabelSize,
|
||||||
className,
|
className,
|
||||||
}) {
|
}) {
|
||||||
@@ -27,6 +28,7 @@ export function DetailsMenu({
|
|||||||
{
|
{
|
||||||
'details-menu--vertical': direction === DIRECTION.VERTICAL,
|
'details-menu--vertical': direction === DIRECTION.VERTICAL,
|
||||||
'details-menu--horizantal': direction === DIRECTION.HORIZANTAL,
|
'details-menu--horizantal': direction === DIRECTION.HORIZANTAL,
|
||||||
|
[`align-${textAlign}`]: textAlign,
|
||||||
},
|
},
|
||||||
className,
|
className,
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -9,16 +9,16 @@ function DialogComponent(props) {
|
|||||||
const { name, children, closeDialog, onClose } = props;
|
const { name, children, closeDialog, onClose } = props;
|
||||||
|
|
||||||
const handleClose = (event) => {
|
const handleClose = (event) => {
|
||||||
closeDialog(name)
|
closeDialog(name);
|
||||||
onClose && onClose(event);
|
onClose && onClose(event);
|
||||||
};
|
};
|
||||||
return (
|
return (
|
||||||
<Dialog {...props} onClose={handleClose}>
|
<Dialog {...props} onClose={handleClose}>
|
||||||
{ children }
|
{children}
|
||||||
</Dialog>
|
</Dialog>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default compose(
|
const DialogRoot = compose(withDialogActions)(DialogComponent);
|
||||||
withDialogActions,
|
|
||||||
)(DialogComponent);
|
export { DialogRoot as Dialog };
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import React from 'react';
|
|||||||
import { Spinner, Classes } from '@blueprintjs/core';
|
import { Spinner, Classes } from '@blueprintjs/core';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
|
||||||
export default function DialogContent(props) {
|
export function DialogContent(props) {
|
||||||
const { isLoading, children } = props;
|
const { isLoading, children } = props;
|
||||||
|
|
||||||
const loadingContent = (
|
const loadingContent = (
|
||||||
|
|||||||
15
src/components/Dialog/DialogFooter.js
Normal file
15
src/components/Dialog/DialogFooter.js
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
import { Classes } from '@blueprintjs/core';
|
||||||
|
|
||||||
|
export function DialogFooter({ children }) {
|
||||||
|
return (
|
||||||
|
<DialogFooterRoot className={Classes.DIALOG_FOOTER}>
|
||||||
|
{children}
|
||||||
|
</DialogFooterRoot>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogFooterRoot = styled.div`
|
||||||
|
display: flex;
|
||||||
|
`;
|
||||||
41
src/components/Dialog/DialogFooterActions.js
Normal file
41
src/components/Dialog/DialogFooterActions.js
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
import { Classes } from '@blueprintjs/core';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dialog footer actions.
|
||||||
|
* @returns {React.JSX}
|
||||||
|
*/
|
||||||
|
export function DialogFooterActions({ alignment = 'right', children }) {
|
||||||
|
return (
|
||||||
|
<DialogFooterActionsRoot
|
||||||
|
className={Classes.DIALOG_FOOTER_ACTIONS}
|
||||||
|
alignment={alignment}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</DialogFooterActionsRoot>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dialog footer.
|
||||||
|
* @returns {React.JSX}
|
||||||
|
*/
|
||||||
|
export function DialogFooter({ ...props }) {
|
||||||
|
return <DialogFooterRoot {...props} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogFooterRoot = styled.div`
|
||||||
|
flex: 0 0 auto;
|
||||||
|
margin: 0 20px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const DialogFooterActionsRoot = styled.div`
|
||||||
|
${(props) =>
|
||||||
|
props.alignment === 'right' ? 'margin-left: auto;' : 'margin-right: auto;'};
|
||||||
|
|
||||||
|
.bp3-button {
|
||||||
|
margin-left: 5px;
|
||||||
|
margin-left: 5px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
@@ -5,7 +5,7 @@ function LoadingContent() {
|
|||||||
return (<div className={Classes.DIALOG_BODY}><Spinner size={30} /></div>);
|
return (<div className={Classes.DIALOG_BODY}><Spinner size={30} /></div>);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function DialogSuspense({
|
export function DialogSuspense({
|
||||||
children
|
children
|
||||||
}) {
|
}) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
7
src/components/Dialog/index.js
Normal file
7
src/components/Dialog/index.js
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
|
||||||
|
|
||||||
|
export * from './Dialog';
|
||||||
|
export * from './DialogFooterActions';
|
||||||
|
export * from './DialogSuspense';
|
||||||
|
export * from './DialogContent';
|
||||||
|
export * from './DialogFooter';
|
||||||
@@ -20,6 +20,18 @@ import ReceiptPdfPreviewDialog from '../containers/Dialogs/ReceiptPdfPreviewDial
|
|||||||
import MoneyInDialog from '../containers/Dialogs/MoneyInDialog';
|
import MoneyInDialog from '../containers/Dialogs/MoneyInDialog';
|
||||||
import MoneyOutDialog from '../containers/Dialogs/MoneyOutDialog';
|
import MoneyOutDialog from '../containers/Dialogs/MoneyOutDialog';
|
||||||
import BadDebtDialog from '../containers/Dialogs/BadDebtDialog';
|
import BadDebtDialog from '../containers/Dialogs/BadDebtDialog';
|
||||||
|
import NotifyInvoiceViaSMSDialog from '../containers/Dialogs/NotifyInvoiceViaSMSDialog';
|
||||||
|
import NotifyReceiptViaSMSDialog from '../containers/Dialogs/NotifyReceiptViaSMSDialog';
|
||||||
|
import NotifyEstimateViaSMSDialog from '../containers/Dialogs/NotifyEstimateViaSMSDialog';
|
||||||
|
import NotifyPaymentReceiveViaSMSDialog from '../containers/Dialogs/NotifyPaymentReceiveViaSMSDialog';
|
||||||
|
import SMSMessageDialog from '../containers/Dialogs/SMSMessageDialog';
|
||||||
|
import RefundCreditNoteDialog from '../containers/Dialogs/RefundCreditNoteDialog';
|
||||||
|
import RefundVendorCreditDialog from '../containers/Dialogs/RefundVendorCreditDialog';
|
||||||
|
import ReconcileCreditNoteDialog from '../containers/Dialogs/ReconcileCreditNoteDialog';
|
||||||
|
import ReconcileVendorCreditDialog from '../containers/Dialogs/ReconcileVendorCreditDialog';
|
||||||
|
import LockingTransactionsDialog from '../containers/Dialogs/LockingTransactionsDialog';
|
||||||
|
import UnlockingTransactionsDialog from '../containers/Dialogs/UnlockingTransactionsDialog';
|
||||||
|
import UnlockingPartialTransactionsDialog from '../containers/Dialogs/UnlockingPartialTransactionsDialog';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dialogs container.
|
* Dialogs container.
|
||||||
@@ -45,7 +57,23 @@ export default function DialogsContainer() {
|
|||||||
<ReceiptPdfPreviewDialog dialogName={'receipt-pdf-preview'} />
|
<ReceiptPdfPreviewDialog dialogName={'receipt-pdf-preview'} />
|
||||||
<MoneyInDialog dialogName={'money-in'} />
|
<MoneyInDialog dialogName={'money-in'} />
|
||||||
<MoneyOutDialog dialogName={'money-out'} />
|
<MoneyOutDialog dialogName={'money-out'} />
|
||||||
|
|
||||||
|
<NotifyInvoiceViaSMSDialog dialogName={'notify-invoice-via-sms'} />
|
||||||
|
<NotifyReceiptViaSMSDialog dialogName={'notify-receipt-via-sms'} />
|
||||||
|
<NotifyEstimateViaSMSDialog dialogName={'notify-estimate-via-sms'} />
|
||||||
|
<NotifyPaymentReceiveViaSMSDialog dialogName={'notify-payment-via-sms'} />
|
||||||
|
|
||||||
<BadDebtDialog dialogName={'write-off-bad-debt'} />
|
<BadDebtDialog dialogName={'write-off-bad-debt'} />
|
||||||
|
<SMSMessageDialog dialogName={'sms-message-form'} />
|
||||||
|
<RefundCreditNoteDialog dialogName={'refund-credit-note'} />
|
||||||
|
<RefundVendorCreditDialog dialogName={'refund-vendor-credit'} />
|
||||||
|
<ReconcileCreditNoteDialog dialogName={'reconcile-credit-note'} />
|
||||||
|
<ReconcileVendorCreditDialog dialogName={'reconcile-vendor-credit'} />
|
||||||
|
<LockingTransactionsDialog dialogName={'locking-transactions'} />
|
||||||
|
<UnlockingTransactionsDialog dialogName={'unlocking-transactions'} />
|
||||||
|
<UnlockingPartialTransactionsDialog
|
||||||
|
dialogName={'unlocking-partial-transactions'}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import ListSelect from "./ListSelect";
|
import intl from 'react-intl-universal';
|
||||||
|
import ListSelect from './ListSelect';
|
||||||
|
|
||||||
export default function DisplayNameList({
|
export default function DisplayNameList({
|
||||||
salutation,
|
salutation,
|
||||||
@@ -9,25 +10,32 @@ export default function DisplayNameList({
|
|||||||
...restProps
|
...restProps
|
||||||
}) {
|
}) {
|
||||||
const formats = [
|
const formats = [
|
||||||
{ format: '{1} {2} {3}', values: [salutation, firstName, lastName], required: [1] },
|
{
|
||||||
|
format: '{1} {2} {3}',
|
||||||
|
values: [salutation, firstName, lastName],
|
||||||
|
required: [1],
|
||||||
|
},
|
||||||
{ format: '{1} {2}', values: [firstName, lastName], required: [] },
|
{ format: '{1} {2}', values: [firstName, lastName], required: [] },
|
||||||
{ format: '{1}, {2}', values: [firstName, lastName], required: [1, 2] },
|
{ format: '{1}, {2}', values: [firstName, lastName], required: [1, 2] },
|
||||||
{ format: '{1}', values: [company], required: [1] }
|
{ format: '{1}', values: [company], required: [1] },
|
||||||
];
|
];
|
||||||
|
|
||||||
const formatOptions = formats
|
const formatOptions = formats
|
||||||
.filter((format) => !format.values.some((value, index) => {
|
.filter(
|
||||||
return !value && format.required.indexOf(index + 1) !== -1;
|
(format) =>
|
||||||
}))
|
!format.values.some((value, index) => {
|
||||||
|
return !value && format.required.indexOf(index + 1) !== -1;
|
||||||
|
}),
|
||||||
|
)
|
||||||
.map((formatOption) => {
|
.map((formatOption) => {
|
||||||
const { format, values } = formatOption;
|
const { format, values } = formatOption;
|
||||||
let label = format;
|
let label = format;
|
||||||
|
|
||||||
values.forEach((value, index) => {
|
values.forEach((value, index) => {
|
||||||
const replaceWith = (value || '');
|
const replaceWith = value || '';
|
||||||
label = label.replace(`{${index + 1}}`, replaceWith).trim();
|
label = label.replace(`{${index + 1}}`, replaceWith).trim();
|
||||||
});
|
});
|
||||||
return { label: label.replace(/\s+/g, " ") };
|
return { label: label.replace(/\s+/g, ' ') };
|
||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -35,9 +43,9 @@ export default function DisplayNameList({
|
|||||||
items={formatOptions}
|
items={formatOptions}
|
||||||
selectedItemProp={'label'}
|
selectedItemProp={'label'}
|
||||||
textProp={'label'}
|
textProp={'label'}
|
||||||
defaultText={'Select display name as'}
|
defaultText={intl.get('select_display_name_as')}
|
||||||
filterable={false}
|
filterable={false}
|
||||||
{ ...restProps }
|
{...restProps}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import { Position, Drawer } from '@blueprintjs/core';
|
|||||||
|
|
||||||
import 'style/components/Drawer.scss';
|
import 'style/components/Drawer.scss';
|
||||||
|
|
||||||
|
import { DrawerProvider } from './DrawerProvider';
|
||||||
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
||||||
import { compose } from 'utils';
|
import { compose } from 'utils';
|
||||||
|
|
||||||
@@ -27,7 +28,7 @@ function DrawerComponent(props) {
|
|||||||
portalClassName={'drawer-portal'}
|
portalClassName={'drawer-portal'}
|
||||||
{...props}
|
{...props}
|
||||||
>
|
>
|
||||||
{children}
|
<DrawerProvider {...props}>{children}</DrawerProvider>
|
||||||
</Drawer>
|
</Drawer>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
12
src/components/Drawer/DrawerActionsBar.js
Normal file
12
src/components/Drawer/DrawerActionsBar.js
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
import DashboardActionsBar from 'components/Dashboard/DashboardActionsBar';
|
||||||
|
|
||||||
|
export function DrawerActionsBar({ ...props }) {
|
||||||
|
return <DrawerActionsBarRoot {...props} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DrawerActionsBarRoot = styled(DashboardActionsBar)`
|
||||||
|
border-bottom: 1px solid #d9d9da;
|
||||||
|
`;
|
||||||
@@ -1,15 +1,54 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Tabs } from '@blueprintjs/core';
|
import { Tabs } from '@blueprintjs/core';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Drawer main tabs.
|
* Drawer main tabs.
|
||||||
*/
|
*/
|
||||||
export function DrawerMainTabs({ children, ...restProps }) {
|
export function DrawerMainTabs({ children, ...restProps }) {
|
||||||
return (
|
return (
|
||||||
<div class="drawer__main-tabs">
|
<DrawerMainTabsRoot>
|
||||||
<Tabs animate={true} large={true} {...restProps}>
|
<Tabs animate={true} large={true} {...restProps}>
|
||||||
{children}
|
{children}
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</div>
|
</DrawerMainTabsRoot>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const DrawerMainTabsRoot = styled.div`
|
||||||
|
.bp3-tabs {
|
||||||
|
.bp3-tab-list {
|
||||||
|
position: relative;
|
||||||
|
background-color: #fff;
|
||||||
|
padding: 0 15px;
|
||||||
|
border-bottom: 2px solid #e1e2e8;
|
||||||
|
|
||||||
|
> *:not(:last-child) {
|
||||||
|
margin-right: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.bp3-large > .bp3-tab {
|
||||||
|
font-size: 15px;
|
||||||
|
color: #7f8596;
|
||||||
|
margin: 0 1rem;
|
||||||
|
|
||||||
|
&[aria-selected='true'],
|
||||||
|
&:not([aria-disabled='true']):hover {
|
||||||
|
color: #0052cc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.bp3-tab-indicator-wrapper .bp3-tab-indicator {
|
||||||
|
height: 2px;
|
||||||
|
bottom: -2px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.bp3-tab-panel {
|
||||||
|
margin-top: 0;
|
||||||
|
|
||||||
|
.card {
|
||||||
|
margin: 15px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|||||||
16
src/components/Drawer/DrawerProvider.js
Normal file
16
src/components/Drawer/DrawerProvider.js
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import React, { createContext, useContext } from 'react';
|
||||||
|
|
||||||
|
const DrawerContext = createContext();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Account form provider.
|
||||||
|
*/
|
||||||
|
function DrawerProvider({ ...props }) {
|
||||||
|
const provider = { ...props };
|
||||||
|
|
||||||
|
return <DrawerContext.Provider value={provider} {...props} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const useDrawerContext = () => useContext(DrawerContext);
|
||||||
|
|
||||||
|
export { DrawerProvider, useDrawerContext };
|
||||||
@@ -13,3 +13,5 @@ export function DrawerLoading({ loading, mount = false, children }) {
|
|||||||
export function DrawerBody({ children }) {
|
export function DrawerBody({ children }) {
|
||||||
return <div className={Classes.DRAWER_BODY}>{children}</div>;
|
return <div className={Classes.DRAWER_BODY}>{children}</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export * from './DrawerActionsBar';
|
||||||
@@ -14,6 +14,13 @@ import CustomerDetailsDrawer from '../containers/Drawers/CustomerDetailsDrawer';
|
|||||||
import VendorDetailsDrawer from '../containers/Drawers/VendorDetailsDrawer';
|
import VendorDetailsDrawer from '../containers/Drawers/VendorDetailsDrawer';
|
||||||
import InventoryAdjustmentDetailDrawer from '../containers/Drawers/InventoryAdjustmentDetailDrawer';
|
import InventoryAdjustmentDetailDrawer from '../containers/Drawers/InventoryAdjustmentDetailDrawer';
|
||||||
import CashflowTransactionDetailDrawer from '../containers/Drawers/CashflowTransactionDetailDrawer';
|
import CashflowTransactionDetailDrawer from '../containers/Drawers/CashflowTransactionDetailDrawer';
|
||||||
|
import QuickCreateCustomerDrawer from '../containers/Drawers/QuickCreateCustomerDrawer';
|
||||||
|
import QuickCreateItemDrawer from '../containers/Drawers/QuickCreateItemDrawer';
|
||||||
|
import QuickWriteVendorDrawer from '../containers/Drawers/QuickWriteVendorDrawer';
|
||||||
|
import CreditNoteDetailDrawer from '../containers/Drawers/CreditNoteDetailDrawer';
|
||||||
|
import VendorCreditDetailDrawer from '../containers/Drawers/VendorCreditDetailDrawer';
|
||||||
|
import RefundCreditNoteDetailDrawer from '../containers/Drawers/RefundCreditNoteDetailDrawer';
|
||||||
|
import RefundVendorCreditDetailDrawer from '../containers/Drawers/RefundVendorCreditDetailDrawer';
|
||||||
|
|
||||||
import { DRAWERS } from 'common/drawers';
|
import { DRAWERS } from 'common/drawers';
|
||||||
|
|
||||||
@@ -38,7 +45,20 @@ export default function DrawersContainer() {
|
|||||||
<InventoryAdjustmentDetailDrawer
|
<InventoryAdjustmentDetailDrawer
|
||||||
name={DRAWERS.INVENTORY_ADJUSTMENT_DRAWER}
|
name={DRAWERS.INVENTORY_ADJUSTMENT_DRAWER}
|
||||||
/>
|
/>
|
||||||
<CashflowTransactionDetailDrawer name={DRAWERS.CASHFLOW_TRNASACTION_DRAWER} />
|
<CashflowTransactionDetailDrawer
|
||||||
|
name={DRAWERS.CASHFLOW_TRNASACTION_DRAWER}
|
||||||
|
/>
|
||||||
|
<QuickCreateCustomerDrawer name={DRAWERS.QUICK_CREATE_CUSTOMER} />
|
||||||
|
<QuickCreateItemDrawer name={DRAWERS.QUICK_CREATE_ITEM} />
|
||||||
|
<QuickWriteVendorDrawer name={DRAWERS.QUICK_WRITE_VENDOR} />
|
||||||
|
<CreditNoteDetailDrawer name={DRAWERS.CREDIT_NOTE_DETAIL_DRAWER} />
|
||||||
|
<VendorCreditDetailDrawer name={DRAWERS.VENDOR_CREDIT_DETAIL_DRAWER} />
|
||||||
|
<RefundCreditNoteDetailDrawer
|
||||||
|
name={DRAWERS.REFUND_CREDIT_NOTE_DETAIL_DRAWER}
|
||||||
|
/>
|
||||||
|
<RefundVendorCreditDetailDrawer
|
||||||
|
name={DRAWERS.REFUND_VENDOR_CREDIT_DETAIL_DRAWER}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
14
src/components/Forms/FormikObserver.js
Normal file
14
src/components/Forms/FormikObserver.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
|
||||||
|
import { useDeepCompareEffect } from 'hooks/utils';
|
||||||
|
|
||||||
|
export function FormikObserver({ onChange, values }) {
|
||||||
|
useDeepCompareEffect(() => {
|
||||||
|
onChange(values);
|
||||||
|
}, [values]);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
FormikObserver.defaultProps = {
|
||||||
|
onChange: () => null,
|
||||||
|
};
|
||||||
@@ -1 +1,2 @@
|
|||||||
export * from './FormObserver';
|
export * from './FormObserver';
|
||||||
|
export * from './FormikObserver';
|
||||||
@@ -1,13 +1,68 @@
|
|||||||
import React, { useState, useCallback, useEffect, useMemo } from 'react';
|
import React, { useState, useCallback, useEffect, useMemo } from 'react';
|
||||||
import { MenuItem } from '@blueprintjs/core';
|
import { MenuItem } from '@blueprintjs/core';
|
||||||
import classNames from 'classnames';
|
|
||||||
import { CLASSES } from 'common/classes';
|
|
||||||
|
|
||||||
import { Suggest } from '@blueprintjs/select';
|
import { Suggest } from '@blueprintjs/select';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import { CLASSES } from 'common/classes';
|
||||||
|
|
||||||
import { FormattedMessage as T } from 'components';
|
import { FormattedMessage as T } from 'components';
|
||||||
|
|
||||||
export default function ItemsSuggestField({
|
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
||||||
|
|
||||||
|
import { DRAWERS } from 'common/drawers';
|
||||||
|
|
||||||
|
// Creates a new item from query.
|
||||||
|
const createNewItemFromQuery = (name) => {
|
||||||
|
return {
|
||||||
|
name,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle quick create new customer.
|
||||||
|
const createNewItemRenderer = (query, active, handleClick) => {
|
||||||
|
return (
|
||||||
|
<MenuItem
|
||||||
|
icon="add"
|
||||||
|
text={intl.get('list.create', { value: `"${query}"` })}
|
||||||
|
active={active}
|
||||||
|
shouldDismissPopover={false}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Item renderer.
|
||||||
|
const itemRenderer = (item, { modifiers, handleClick }) => (
|
||||||
|
<MenuItem
|
||||||
|
key={item.id}
|
||||||
|
text={item.name}
|
||||||
|
label={item.code}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
// Filters items.
|
||||||
|
const filterItemsPredicater = (query, item, _index, exactMatch) => {
|
||||||
|
const normalizedTitle = item.name.toLowerCase();
|
||||||
|
const normalizedQuery = query.toLowerCase();
|
||||||
|
|
||||||
|
if (exactMatch) {
|
||||||
|
return normalizedTitle === normalizedQuery;
|
||||||
|
} else {
|
||||||
|
return `${normalizedTitle} ${item.code}`.indexOf(normalizedQuery) >= 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle input value renderer.
|
||||||
|
const handleInputValueRenderer = (inputValue) => {
|
||||||
|
if (inputValue) {
|
||||||
|
return inputValue.name.toString();
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
};
|
||||||
|
|
||||||
|
function ItemsSuggestField({
|
||||||
items,
|
items,
|
||||||
initialItemId,
|
initialItemId,
|
||||||
selectedItemId,
|
selectedItemId,
|
||||||
@@ -18,6 +73,10 @@ export default function ItemsSuggestField({
|
|||||||
sellable = false,
|
sellable = false,
|
||||||
purchasable = false,
|
purchasable = false,
|
||||||
popoverFill = false,
|
popoverFill = false,
|
||||||
|
|
||||||
|
allowCreate = true,
|
||||||
|
|
||||||
|
openDrawer,
|
||||||
...suggestProps
|
...suggestProps
|
||||||
}) {
|
}) {
|
||||||
// Filters items based on filter props.
|
// Filters items based on filter props.
|
||||||
@@ -36,28 +95,23 @@ export default function ItemsSuggestField({
|
|||||||
// Find initial item object.
|
// Find initial item object.
|
||||||
const initialItem = useMemo(
|
const initialItem = useMemo(
|
||||||
() => filteredItems.some((a) => a.id === initialItemId),
|
() => filteredItems.some((a) => a.id === initialItemId),
|
||||||
[initialItemId],
|
[initialItemId, filteredItems],
|
||||||
);
|
);
|
||||||
|
|
||||||
const [selectedItem, setSelectedItem] = useState(initialItem || null);
|
const [selectedItem, setSelectedItem] = useState(initialItem || null);
|
||||||
|
|
||||||
const onItemSelect = useCallback(
|
const onItemSelect = useCallback(
|
||||||
(item) => {
|
(item) => {
|
||||||
setSelectedItem({ ...item });
|
if (item.id) {
|
||||||
onItemSelected && onItemSelected(item);
|
setSelectedItem({ ...item });
|
||||||
|
onItemSelected && onItemSelected(item);
|
||||||
|
} else {
|
||||||
|
openDrawer(DRAWERS.QUICK_CREATE_ITEM);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[setSelectedItem, onItemSelected],
|
[setSelectedItem, onItemSelected, openDrawer],
|
||||||
);
|
);
|
||||||
|
|
||||||
const itemRenderer = useCallback((item, { modifiers, handleClick }) => (
|
|
||||||
<MenuItem
|
|
||||||
key={item.id}
|
|
||||||
text={item.name}
|
|
||||||
label={item.code}
|
|
||||||
onClick={handleClick}
|
|
||||||
/>
|
|
||||||
));
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (typeof selectedItemId !== 'undefined') {
|
if (typeof selectedItemId !== 'undefined') {
|
||||||
const item = selectedItemId
|
const item = selectedItemId
|
||||||
@@ -67,27 +121,12 @@ export default function ItemsSuggestField({
|
|||||||
}
|
}
|
||||||
}, [selectedItemId, filteredItems, setSelectedItem]);
|
}, [selectedItemId, filteredItems, setSelectedItem]);
|
||||||
|
|
||||||
const handleInputValueRenderer = (inputValue) => {
|
// Maybe inject create new item props to suggest component.
|
||||||
if (inputValue) {
|
const maybeCreateNewItemRenderer = allowCreate ? createNewItemRenderer : null;
|
||||||
return inputValue.name.toString();
|
const maybeCreateNewItemFromQuery = allowCreate
|
||||||
}
|
? createNewItemFromQuery
|
||||||
return '';
|
: null;
|
||||||
};
|
|
||||||
|
|
||||||
// Filters items.
|
|
||||||
const filterItemsPredicater = useCallback(
|
|
||||||
(query, item, _index, exactMatch) => {
|
|
||||||
const normalizedTitle = item.name.toLowerCase();
|
|
||||||
const normalizedQuery = query.toLowerCase();
|
|
||||||
|
|
||||||
if (exactMatch) {
|
|
||||||
return normalizedTitle === normalizedQuery;
|
|
||||||
} else {
|
|
||||||
return `${normalizedTitle} ${item.code}`.indexOf(normalizedQuery) >= 0;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[],
|
|
||||||
);
|
|
||||||
return (
|
return (
|
||||||
<Suggest
|
<Suggest
|
||||||
items={filteredItems}
|
items={filteredItems}
|
||||||
@@ -104,7 +143,12 @@ export default function ItemsSuggestField({
|
|||||||
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
className={classNames(CLASSES.FORM_GROUP_LIST_SELECT, {
|
||||||
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
[CLASSES.SELECT_LIST_FILL_POPOVER]: popoverFill,
|
||||||
})}
|
})}
|
||||||
|
createNewItemRenderer={maybeCreateNewItemRenderer}
|
||||||
|
createNewItemFromQuery={maybeCreateNewItemFromQuery}
|
||||||
|
createNewItemPosition={'top'}
|
||||||
{...suggestProps}
|
{...suggestProps}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export default R.compose(withDrawerActions)(ItemsSuggestField);
|
||||||
|
|||||||
36
src/components/MoreMenutItems.js
Normal file
36
src/components/MoreMenutItems.js
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import {
|
||||||
|
Button,
|
||||||
|
Popover,
|
||||||
|
PopoverInteractionKind,
|
||||||
|
Position,
|
||||||
|
MenuItem,
|
||||||
|
Menu,
|
||||||
|
} from '@blueprintjs/core';
|
||||||
|
|
||||||
|
import { Icon, FormattedMessage as T } from 'components';
|
||||||
|
|
||||||
|
function MoreMenuItems({ payload: { onNotifyViaSMS } }) {
|
||||||
|
return (
|
||||||
|
<Popover
|
||||||
|
minimal={true}
|
||||||
|
content={
|
||||||
|
<Menu>
|
||||||
|
<MenuItem
|
||||||
|
onClick={onNotifyViaSMS}
|
||||||
|
text={<T id={'notify_via_sms.dialog.notify_via_sms'} />}
|
||||||
|
/>
|
||||||
|
</Menu>
|
||||||
|
}
|
||||||
|
interactionKind={PopoverInteractionKind.CLICK}
|
||||||
|
position={Position.BOTTOM_LEFT}
|
||||||
|
modifiers={{
|
||||||
|
offset: { offset: '0, 4' },
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Button icon={<Icon icon="more-vert" iconSize={16} />} minimal={true} />
|
||||||
|
</Popover>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default MoreMenuItems;
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import {
|
|
||||||
Button,
|
|
||||||
PopoverInteractionKind,
|
|
||||||
MenuItem,
|
|
||||||
Position,
|
|
||||||
} from '@blueprintjs/core';
|
|
||||||
|
|
||||||
import { Select } from '@blueprintjs/select';
|
|
||||||
import { Icon } from 'components';
|
|
||||||
|
|
||||||
function MoreVertMenutItems({ text, items, onItemSelect, buttonProps }) {
|
|
||||||
// Menu items renderer.
|
|
||||||
const itemsRenderer = (item, { handleClick, modifiers, query }) => (
|
|
||||||
<MenuItem text={item.name} label={item.label} onClick={handleClick} />
|
|
||||||
);
|
|
||||||
const handleMenuSelect = (type) => {
|
|
||||||
onItemSelect && onItemSelect(type);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Select
|
|
||||||
items={items}
|
|
||||||
itemRenderer={itemsRenderer}
|
|
||||||
onItemSelect={handleMenuSelect}
|
|
||||||
popoverProps={{
|
|
||||||
minimal: true,
|
|
||||||
position: Position.BOTTOM_LEFT,
|
|
||||||
interactionKind: PopoverInteractionKind.CLICK,
|
|
||||||
modifiers: {
|
|
||||||
offset: { offset: '0, 4' },
|
|
||||||
},
|
|
||||||
}}
|
|
||||||
filterable={false}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
text={text}
|
|
||||||
icon={<Icon icon={'more-vert'} iconSize={16} />}
|
|
||||||
minimal={true}
|
|
||||||
{...buttonProps}
|
|
||||||
/>
|
|
||||||
</Select>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default MoreVertMenutItems;
|
|
||||||
@@ -1,6 +1,8 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { ErrorBoundary } from 'react-error-boundary';
|
import { ErrorBoundary } from 'react-error-boundary';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
import { CLASSES } from 'common/classes';
|
import { CLASSES } from 'common/classes';
|
||||||
|
|
||||||
import PreferencesTopbar from 'components/Preferences/PreferencesTopbar';
|
import PreferencesTopbar from 'components/Preferences/PreferencesTopbar';
|
||||||
@@ -8,18 +10,28 @@ import PreferencesContentRoute from 'components/Preferences/PreferencesContentRo
|
|||||||
import DashboardErrorBoundary from 'components/Dashboard/DashboardErrorBoundary';
|
import DashboardErrorBoundary from 'components/Dashboard/DashboardErrorBoundary';
|
||||||
import PreferencesSidebar from 'components/Preferences/PreferencesSidebar';
|
import PreferencesSidebar from 'components/Preferences/PreferencesSidebar';
|
||||||
|
|
||||||
|
import withDashboardActions from 'containers/Dashboard/withDashboardActions';
|
||||||
|
|
||||||
import 'style/pages/Preferences/Page.scss';
|
import 'style/pages/Preferences/Page.scss';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preferences page.
|
* Preferences page.
|
||||||
*/
|
*/
|
||||||
export default function PreferencesPage() {
|
function PreferencesPage({ toggleSidebarExpand }) {
|
||||||
|
// Shrink the dashboard sidebar once open application preferences page.
|
||||||
|
React.useEffect(() => {
|
||||||
|
toggleSidebarExpand(false);
|
||||||
|
}, [toggleSidebarExpand]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ErrorBoundary FallbackComponent={DashboardErrorBoundary}>
|
<ErrorBoundary FallbackComponent={DashboardErrorBoundary}>
|
||||||
<div id={'dashboard'} className={classNames(
|
<div
|
||||||
CLASSES.DASHBOARD_CONTENT,
|
id={'dashboard'}
|
||||||
CLASSES.DASHBOARD_CONTENT_PREFERENCES,
|
className={classNames(
|
||||||
)}>
|
CLASSES.DASHBOARD_CONTENT,
|
||||||
|
CLASSES.DASHBOARD_CONTENT_PREFERENCES,
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div className={classNames(CLASSES.PREFERENCES_PAGE)}>
|
<div className={classNames(CLASSES.PREFERENCES_PAGE)}>
|
||||||
<PreferencesSidebar />
|
<PreferencesSidebar />
|
||||||
|
|
||||||
@@ -32,3 +44,5 @@ export default function PreferencesPage() {
|
|||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export default R.compose(withDashboardActions)(PreferencesPage);
|
||||||
|
|||||||
46
src/components/SMSPreview/index.js
Normal file
46
src/components/SMSPreview/index.js
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
import { Icon } from 'components';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SMS Message preview.
|
||||||
|
*/
|
||||||
|
export function SMSMessagePreview({
|
||||||
|
message,
|
||||||
|
iconWidth = '265px',
|
||||||
|
iconHeight = '287px',
|
||||||
|
iconColor = '#adadad',
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<SMSMessagePreviewBase>
|
||||||
|
<Icon
|
||||||
|
icon={'sms-message-preview'}
|
||||||
|
width={iconWidth}
|
||||||
|
height={iconHeight}
|
||||||
|
color={iconColor}
|
||||||
|
/>
|
||||||
|
<SMSMessageText>{message}</SMSMessageText>
|
||||||
|
</SMSMessagePreviewBase>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const SMSMessageText = styled.div`
|
||||||
|
position: absolute;
|
||||||
|
top: 60px;
|
||||||
|
padding: 12px;
|
||||||
|
color: #fff;
|
||||||
|
border-radius: 12px;
|
||||||
|
margin-left: 12px;
|
||||||
|
margin-right: 12px;
|
||||||
|
word-break: break-word;
|
||||||
|
background: #2fa2e4;
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 1.6;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const SMSMessagePreviewBase = styled.div`
|
||||||
|
position: relative;
|
||||||
|
width: 265px;
|
||||||
|
margin: 0 auto;
|
||||||
|
`;
|
||||||
@@ -2,16 +2,19 @@ import React from 'react';
|
|||||||
import SidebarContainer from 'components/Sidebar/SidebarContainer';
|
import SidebarContainer from 'components/Sidebar/SidebarContainer';
|
||||||
import SidebarHead from 'components/Sidebar/SidebarHead';
|
import SidebarHead from 'components/Sidebar/SidebarHead';
|
||||||
import SidebarMenu from 'components/Sidebar/SidebarMenu';
|
import SidebarMenu from 'components/Sidebar/SidebarMenu';
|
||||||
|
import { useGetSidebarMenu } from './utils';
|
||||||
|
|
||||||
import 'style/containers/Dashboard/Sidebar.scss';
|
import 'style/containers/Dashboard/Sidebar.scss';
|
||||||
|
|
||||||
export default function Sidebar({ dashboardContentRef }) {
|
export default function Sidebar({ dashboardContentRef }) {
|
||||||
|
const menu = useGetSidebarMenu();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SidebarContainer>
|
<SidebarContainer>
|
||||||
<SidebarHead />
|
<SidebarHead />
|
||||||
|
|
||||||
<div className="sidebar__menu">
|
<div className="sidebar__menu">
|
||||||
<SidebarMenu />
|
<SidebarMenu menu={menu} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="sidebar__version">0.0.1-beta version.</div>
|
<div class="sidebar__version">0.0.1-beta version.</div>
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { Button, Popover, Menu, Position } from '@blueprintjs/core';
|
|||||||
import Icon from 'components/Icon';
|
import Icon from 'components/Icon';
|
||||||
import { compose, firstLettersArgs } from 'utils';
|
import { compose, firstLettersArgs } from 'utils';
|
||||||
import withCurrentOrganization from '../../containers/Organization/withCurrentOrganization';
|
import withCurrentOrganization from '../../containers/Organization/withCurrentOrganization';
|
||||||
import { useAuthenticatedUser } from '../Dashboard/AuthenticatedUser';
|
import { useAuthenticatedAccount } from '../../hooks/query';
|
||||||
|
|
||||||
// Popover modifiers.
|
// Popover modifiers.
|
||||||
const POPOVER_MODIFIERS = {
|
const POPOVER_MODIFIERS = {
|
||||||
@@ -18,7 +18,7 @@ function SidebarHead({
|
|||||||
organization,
|
organization,
|
||||||
}) {
|
}) {
|
||||||
// Retrieve authenticated user information.
|
// Retrieve authenticated user information.
|
||||||
const { user } = useAuthenticatedUser();
|
const { data: user } = useAuthenticatedAccount();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="sidebar__head">
|
<div className="sidebar__head">
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Menu, MenuDivider } from '@blueprintjs/core';
|
import { Menu, MenuDivider } from '@blueprintjs/core';
|
||||||
import { useHistory, useLocation } from 'react-router-dom';
|
import { useHistory, useLocation } from 'react-router-dom';
|
||||||
import sidebarMenuList from 'config/sidebarMenu';
|
|
||||||
import { Choose } from 'components';
|
import { Choose } from 'components';
|
||||||
import Icon from 'components/Icon';
|
import Icon from 'components/Icon';
|
||||||
import MenuItem from 'components/MenuItem';
|
import MenuItem from 'components/MenuItem';
|
||||||
@@ -24,7 +24,7 @@ function SidebarMenuItemSpace({ space }) {
|
|||||||
return <div class="bp3-menu-spacer" style={{ height: `${space}px` }} />;
|
return <div class="bp3-menu-spacer" style={{ height: `${space}px` }} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
function SidebarMenu({ isSubscriptionActive }) {
|
function SidebarMenu({ menu, isSubscriptionActive }) {
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
|
|
||||||
@@ -93,7 +93,7 @@ function SidebarMenu({ isSubscriptionActive }) {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const filterItems = sidebarMenuList.filter(
|
const filterItems = menu.filter(
|
||||||
(item) => isSubscriptionActive || item.enableBilling,
|
(item) => isSubscriptionActive || item.enableBilling,
|
||||||
);
|
);
|
||||||
const items = menuItemsMapper(filterItems);
|
const items = menuItemsMapper(filterItems);
|
||||||
|
|||||||
48
src/components/Sidebar/utils.js
Normal file
48
src/components/Sidebar/utils.js
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
import sidebarMenuList from 'config/sidebarMenu';
|
||||||
|
import { isArray, isEmpty } from 'lodash';
|
||||||
|
import { useAbilityContext } from 'hooks/utils';
|
||||||
|
|
||||||
|
export function useGetSidebarMenu() {
|
||||||
|
const ability = useAbilityContext();
|
||||||
|
|
||||||
|
return sidebarMenuList
|
||||||
|
.map((item) => {
|
||||||
|
const children = isArray(item.children)
|
||||||
|
? item.children.filter((childItem) => {
|
||||||
|
return isArray(childItem.permission)
|
||||||
|
? childItem.permission.some((perm) =>
|
||||||
|
ability.can(perm.ability, perm.subject),
|
||||||
|
)
|
||||||
|
: childItem?.permission?.ability && childItem?.permission?.subject
|
||||||
|
? ability.can(
|
||||||
|
childItem.permission.ability,
|
||||||
|
childItem.permission.subject,
|
||||||
|
)
|
||||||
|
: true;
|
||||||
|
})
|
||||||
|
: [];
|
||||||
|
|
||||||
|
return {
|
||||||
|
...item,
|
||||||
|
...(isArray(item.children)
|
||||||
|
? {
|
||||||
|
children,
|
||||||
|
}
|
||||||
|
: {}),
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter((item) => {
|
||||||
|
return isArray(item.permission)
|
||||||
|
? item.permission.some((per) =>
|
||||||
|
ability.can(per.ability, per.subject),
|
||||||
|
)
|
||||||
|
: item?.permission?.ability && item?.permission?.subject
|
||||||
|
? ability.can(item.permission.ability, item.permission.subject)
|
||||||
|
: true;
|
||||||
|
})
|
||||||
|
.filter((item) =>
|
||||||
|
isEmpty(item.children) && !item.href && !item.label && !item.divider
|
||||||
|
? false
|
||||||
|
: true,
|
||||||
|
);
|
||||||
|
}
|
||||||
33
src/components/Table/index.js
Normal file
33
src/components/Table/index.js
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
export const Table = styled.table`
|
||||||
|
width: 100%;
|
||||||
|
vertical-align: top;
|
||||||
|
border-color: #dee2e6;
|
||||||
|
border-spacing: 0;
|
||||||
|
`;
|
||||||
|
export const TBody = styled.tbody``;
|
||||||
|
export const TR = styled.tr``;
|
||||||
|
export const TD = styled.td`
|
||||||
|
padding: 0.5rem 0.5rem;
|
||||||
|
border-bottom-width: 1px;
|
||||||
|
border-bottom-color: inherit;
|
||||||
|
border-bottom-style: solid;
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.textAlign === 'right' &&
|
||||||
|
`
|
||||||
|
text-align: right;`}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const TRDarkSingleLine = styled(TR)`
|
||||||
|
${TD} {
|
||||||
|
border-bottom: 1px solid #000;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const TRDarkDoubleLines = styled(TR)`
|
||||||
|
${TD} {
|
||||||
|
border-bottom: 3px double #000;
|
||||||
|
}
|
||||||
|
`;
|
||||||
11
src/components/Tags/CurrencyTag.js
Normal file
11
src/components/Tags/CurrencyTag.js
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
export const CurrencyTag = styled.span`
|
||||||
|
background: #3e9215;
|
||||||
|
color: #fff;
|
||||||
|
display: inline-block;
|
||||||
|
border-radius: 3px;
|
||||||
|
padding: 2px 4px;
|
||||||
|
line-height: 1;
|
||||||
|
margin-left: 4px;
|
||||||
|
`;
|
||||||
3
src/components/Tags/index.js
Normal file
3
src/components/Tags/index.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
|
||||||
|
|
||||||
|
export * from './CurrencyTag';
|
||||||
28
src/components/TextStatus/index.js
Normal file
28
src/components/TextStatus/index.js
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
export function TextStatus({ intent, children }) {
|
||||||
|
return <TextStatusRoot intent={intent}>{children}</TextStatusRoot>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const TextStatusRoot = styled.span`
|
||||||
|
${(props) =>
|
||||||
|
props.intent === 'warning' &&
|
||||||
|
`
|
||||||
|
color: #ec5b0a;`}
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.intent === 'success' &&
|
||||||
|
`
|
||||||
|
color: #2ba01d;`}
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.intent === 'none' &&
|
||||||
|
`
|
||||||
|
color: #777;`}
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.intent === 'primary' &&
|
||||||
|
`
|
||||||
|
color: #1652c8;`}
|
||||||
|
`;
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
.total_lines {}
|
|
||||||
|
|
||||||
|
|
||||||
.total_line {
|
|
||||||
display: flex;
|
|
||||||
border-bottom: 1px solid #d2dde2;
|
|
||||||
|
|
||||||
:global .amount,
|
|
||||||
:global .title{
|
|
||||||
padding: 8px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,23 +1,93 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import clsx from 'classnames';
|
import styled from 'styled-components';
|
||||||
|
|
||||||
import TotalLinesCls from './TotalLines.module.scss';
|
export const TotalLineBorderStyle = {
|
||||||
|
SingleDark: 'SingleDark',
|
||||||
|
DoubleDark: 'DoubleDark',
|
||||||
|
};
|
||||||
|
|
||||||
export function TotalLines({ children, className }) {
|
export const TotalLineTextStyle = {
|
||||||
|
Regular: 'Regular',
|
||||||
|
Bold: 'Bold',
|
||||||
|
};
|
||||||
|
|
||||||
|
export function TotalLines({
|
||||||
|
children,
|
||||||
|
amountColWidth,
|
||||||
|
labelColWidth,
|
||||||
|
className,
|
||||||
|
}) {
|
||||||
return (
|
return (
|
||||||
<div className={clsx('total_lines', TotalLinesCls.total_lines, className)}>
|
<TotalLinesRoot
|
||||||
|
className={className}
|
||||||
|
amountColWidth={amountColWidth}
|
||||||
|
labelColWidth={labelColWidth}
|
||||||
|
>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</TotalLinesRoot>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function TotalLine({ title, value, className }) {
|
export function TotalLine({ title, value, borderStyle, textStyle, className }) {
|
||||||
return (
|
return (
|
||||||
<div
|
<TotalLineRoot
|
||||||
className={clsx('total_lines_line', TotalLinesCls.total_line, className)}
|
borderStyle={borderStyle}
|
||||||
|
textStyle={textStyle}
|
||||||
|
className={className}
|
||||||
>
|
>
|
||||||
<div class="title">{title}</div>
|
<div class="title">{title}</div>
|
||||||
<div class="amount">{value}</div>
|
<div class="amount">{value}</div>
|
||||||
</div>
|
</TotalLineRoot>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const TotalLinesRoot = styled.div`
|
||||||
|
display: table;
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.amountColWidth &&
|
||||||
|
`
|
||||||
|
.amount{
|
||||||
|
width: ${props.amountColWidth}
|
||||||
|
}
|
||||||
|
`}
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.labelColWidth &&
|
||||||
|
`
|
||||||
|
.title{
|
||||||
|
width: ${props.labelColWidth}
|
||||||
|
}
|
||||||
|
`}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const TotalLineRoot = styled.div`
|
||||||
|
display: table-row;
|
||||||
|
|
||||||
|
.amount,
|
||||||
|
.title {
|
||||||
|
display: table-cell;
|
||||||
|
padding: 8px;
|
||||||
|
border-bottom: 1px solid #d2dde2;
|
||||||
|
|
||||||
|
${(props) =>
|
||||||
|
props.borderStyle === TotalLineBorderStyle.DoubleDark &&
|
||||||
|
`
|
||||||
|
border-bottom: 3px double #000;
|
||||||
|
`}
|
||||||
|
${(props) =>
|
||||||
|
props.borderStyle === TotalLineBorderStyle.SingleDark &&
|
||||||
|
`
|
||||||
|
border-bottom: 1px double #000;
|
||||||
|
`}
|
||||||
|
${(props) =>
|
||||||
|
props.textStyle === TotalLineTextStyle.Bold &&
|
||||||
|
`
|
||||||
|
font-weight: 600;
|
||||||
|
`}
|
||||||
|
}
|
||||||
|
|
||||||
|
.amount {
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|||||||
6
src/components/Typo/Paragraph.js
Normal file
6
src/components/Typo/Paragraph.js
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import clsx from 'classnames';
|
||||||
|
|
||||||
|
export function Paragraph({ className, children }) {
|
||||||
|
return <p className={clsx('paragraph', className)}>{children}</p>;
|
||||||
|
}
|
||||||
2
src/components/Typo/index.js
Normal file
2
src/components/Typo/index.js
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
|
||||||
|
export * from './Paragraph';
|
||||||
13
src/components/Utils/Join.js
Normal file
13
src/components/Utils/Join.js
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
export function Join({ items, sep }) {
|
||||||
|
return items.length > 0
|
||||||
|
? items.reduce((result, item) => (
|
||||||
|
<>
|
||||||
|
{result}
|
||||||
|
{sep}
|
||||||
|
{item}
|
||||||
|
</>
|
||||||
|
))
|
||||||
|
: null;
|
||||||
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
|
|
||||||
|
|
||||||
export * from './FormatNumber';
|
export * from './FormatNumber';
|
||||||
export * from './FormatDate';
|
export * from './FormatDate';
|
||||||
|
export * from './Join';
|
||||||
23
src/components/Vendors/VendorDrawerLink.js
Normal file
23
src/components/Vendors/VendorDrawerLink.js
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
|
import { ButtonLink } from 'components';
|
||||||
|
import withDrawerActions from 'containers/Drawer/withDrawerActions';
|
||||||
|
|
||||||
|
function VendorDrawerLinkComponent({
|
||||||
|
// #ownProps
|
||||||
|
children,
|
||||||
|
vendorId,
|
||||||
|
|
||||||
|
// #withDrawerActions
|
||||||
|
openDrawer,
|
||||||
|
}) {
|
||||||
|
// Handle view customer drawer.
|
||||||
|
const handleVendorDrawer = () => {
|
||||||
|
openDrawer('vendor-details-drawer', { vendorId });
|
||||||
|
};
|
||||||
|
|
||||||
|
return <ButtonLink onClick={handleVendorDrawer}>{children}</ButtonLink>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const VendorDrawerLink = R.compose(withDrawerActions)(VendorDrawerLinkComponent);
|
||||||
1
src/components/Vendors/index.js
Normal file
1
src/components/Vendors/index.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './VendorDrawerLink'
|
||||||
@@ -23,9 +23,6 @@ import AccountsSelectList from './AccountsSelectList';
|
|||||||
import AccountsTypesSelect from './AccountsTypesSelect';
|
import AccountsTypesSelect from './AccountsTypesSelect';
|
||||||
import LoadingIndicator from './LoadingIndicator';
|
import LoadingIndicator from './LoadingIndicator';
|
||||||
import DashboardActionViewsList from './Dashboard/DashboardActionViewsList';
|
import DashboardActionViewsList from './Dashboard/DashboardActionViewsList';
|
||||||
import Dialog from './Dialog/Dialog';
|
|
||||||
import DialogContent from './Dialog/DialogContent';
|
|
||||||
import DialogSuspense from './Dialog/DialogSuspense';
|
|
||||||
import InputPrependButton from './Forms/InputPrependButton';
|
import InputPrependButton from './Forms/InputPrependButton';
|
||||||
import CategoriesSelectList from './CategoriesSelectList';
|
import CategoriesSelectList from './CategoriesSelectList';
|
||||||
import Row from './Grid/Row';
|
import Row from './Grid/Row';
|
||||||
@@ -57,12 +54,12 @@ import Postbox from './Postbox';
|
|||||||
import AccountsSuggestField from './AccountsSuggestField';
|
import AccountsSuggestField from './AccountsSuggestField';
|
||||||
import MaterialProgressBar from './MaterialProgressBar';
|
import MaterialProgressBar from './MaterialProgressBar';
|
||||||
import { MoneyFieldCell } from './DataTableCells';
|
import { MoneyFieldCell } from './DataTableCells';
|
||||||
import Card from './Card';
|
|
||||||
import AvaterCell from './AvaterCell';
|
import AvaterCell from './AvaterCell';
|
||||||
|
|
||||||
import { ItemsMultiSelect } from './Items';
|
import { ItemsMultiSelect } from './Items';
|
||||||
import MoreVertMenutItems from './MoreVertMenutItems';
|
import MoreMenuItems from './MoreMenutItems';
|
||||||
|
|
||||||
|
export * from './Dialog';
|
||||||
export * from './Menu';
|
export * from './Menu';
|
||||||
export * from './AdvancedFilter/AdvancedFilterDropdown';
|
export * from './AdvancedFilter/AdvancedFilterDropdown';
|
||||||
export * from './AdvancedFilter/AdvancedFilterPopover';
|
export * from './AdvancedFilter/AdvancedFilterPopover';
|
||||||
@@ -83,10 +80,21 @@ export * from './MultiSelectTaggable';
|
|||||||
export * from './Utils/FormatNumber';
|
export * from './Utils/FormatNumber';
|
||||||
export * from './Utils/FormatDate';
|
export * from './Utils/FormatDate';
|
||||||
export * from './BankAccounts';
|
export * from './BankAccounts';
|
||||||
export * from './IntersectionObserver'
|
export * from './IntersectionObserver';
|
||||||
export * from './Datatable/CellForceWidth';
|
export * from './Datatable/CellForceWidth';
|
||||||
export * from './Button';
|
export * from './Button';
|
||||||
export * from './IntersectionObserver';
|
export * from './IntersectionObserver';
|
||||||
|
export * from './SMSPreview';
|
||||||
|
export * from './Contacts';
|
||||||
|
export * from './Utils/Join';
|
||||||
|
export * from './Typo';
|
||||||
|
export * from './TextStatus';
|
||||||
|
export * from './Tags';
|
||||||
|
export * from './CommercialDoc';
|
||||||
|
export * from './Card';
|
||||||
|
export * from './Customers'
|
||||||
|
export * from './Vendors'
|
||||||
|
export * from './Table';
|
||||||
|
|
||||||
const Hint = FieldHint;
|
const Hint = FieldHint;
|
||||||
|
|
||||||
@@ -120,9 +128,6 @@ export {
|
|||||||
LoadingIndicator,
|
LoadingIndicator,
|
||||||
DashboardActionViewsList,
|
DashboardActionViewsList,
|
||||||
AppToaster,
|
AppToaster,
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogSuspense,
|
|
||||||
InputPrependButton,
|
InputPrependButton,
|
||||||
CategoriesSelectList,
|
CategoriesSelectList,
|
||||||
Col,
|
Col,
|
||||||
@@ -156,7 +161,6 @@ export {
|
|||||||
MaterialProgressBar,
|
MaterialProgressBar,
|
||||||
MoneyFieldCell,
|
MoneyFieldCell,
|
||||||
ItemsMultiSelect,
|
ItemsMultiSelect,
|
||||||
Card,
|
|
||||||
AvaterCell,
|
AvaterCell,
|
||||||
MoreVertMenutItems,
|
MoreMenuItems,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { FormattedMessage as T } from 'components';
|
import { FormattedMessage as T } from 'components';
|
||||||
|
import { ReportsAction, AbilitySubject } from '../common/abilityOption';
|
||||||
|
|
||||||
export const financialReportMenus = [
|
export const financialReportMenus = [
|
||||||
{
|
{
|
||||||
@@ -11,6 +12,8 @@ export const financialReportMenus = [
|
|||||||
<T id={'reports_a_company_s_assets_liabilities_and_shareholders'} />
|
<T id={'reports_a_company_s_assets_liabilities_and_shareholders'} />
|
||||||
),
|
),
|
||||||
link: '/financial-reports/balance-sheet',
|
link: '/financial-reports/balance-sheet',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_BALANCE_SHEET,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'trial_balance_sheet'} />,
|
title: <T id={'trial_balance_sheet'} />,
|
||||||
@@ -18,11 +21,15 @@ export const financialReportMenus = [
|
|||||||
<T id={'summarizes_the_credit_and_debit_balance_of_each_account'} />
|
<T id={'summarizes_the_credit_and_debit_balance_of_each_account'} />
|
||||||
),
|
),
|
||||||
link: '/financial-reports/trial-balance-sheet',
|
link: '/financial-reports/trial-balance-sheet',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_TRIAL_BALANCE_SHEET,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'profit_loss_report'} />,
|
title: <T id={'profit_loss_report'} />,
|
||||||
desc: <T id={'reports_the_revenues_costs_and_expenses'} />,
|
desc: <T id={'reports_the_revenues_costs_and_expenses'} />,
|
||||||
link: '/financial-reports/profit-loss-sheet',
|
link: '/financial-reports/profit-loss-sheet',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_PROFIT_LOSS,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'cash_flow_statement'} />,
|
title: <T id={'cash_flow_statement'} />,
|
||||||
@@ -30,16 +37,22 @@ export const financialReportMenus = [
|
|||||||
<T id={'reports_inflow_and_outflow_of_cash_and_cash_equivalents'} />
|
<T id={'reports_inflow_and_outflow_of_cash_and_cash_equivalents'} />
|
||||||
),
|
),
|
||||||
link: '/financial-reports/cash-flow',
|
link: '/financial-reports/cash-flow',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CASHFLOW,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'journal_report'} />,
|
title: <T id={'journal_report'} />,
|
||||||
desc: <T id={'the_debit_and_credit_entries_of_system_transactions'} />,
|
desc: <T id={'the_debit_and_credit_entries_of_system_transactions'} />,
|
||||||
link: '/financial-reports/journal-sheet',
|
link: '/financial-reports/journal-sheet',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_JOURNAL,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'general_ledger_report'} />,
|
title: <T id={'general_ledger_report'} />,
|
||||||
desc: <T id={'reports_every_transaction_going_in_and_out_of_your'} />,
|
desc: <T id={'reports_every_transaction_going_in_and_out_of_your'} />,
|
||||||
link: '/financial-reports/general-ledger',
|
link: '/financial-reports/general-ledger',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_GENERAL_LEDGET,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'receivable_aging_summary'} />,
|
title: <T id={'receivable_aging_summary'} />,
|
||||||
@@ -47,11 +60,15 @@ export const financialReportMenus = [
|
|||||||
<T id={'summarize_total_unpaid_balances_of_customers_invoices'} />
|
<T id={'summarize_total_unpaid_balances_of_customers_invoices'} />
|
||||||
),
|
),
|
||||||
link: '/financial-reports/receivable-aging-summary',
|
link: '/financial-reports/receivable-aging-summary',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_AR_AGING_SUMMARY,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'payable_aging_summary'} />,
|
title: <T id={'payable_aging_summary'} />,
|
||||||
desc: <T id={'summarize_total_unpaid_balances_of_vendors_purchase'} />,
|
desc: <T id={'summarize_total_unpaid_balances_of_vendors_purchase'} />,
|
||||||
link: '/financial-reports/payable-aging-summary',
|
link: '/financial-reports/payable-aging-summary',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_AP_AGING_SUMMARY,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -71,6 +88,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/financial-reports/purchases-by-items',
|
link: '/financial-reports/purchases-by-items',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_PURCHASES_BY_ITEMS,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'sales_by_items'} />,
|
title: <T id={'sales_by_items'} />,
|
||||||
@@ -82,6 +101,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/financial-reports/sales-by-items',
|
link: '/financial-reports/sales-by-items',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_SALES_BY_ITEMS,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'inventory_valuation'} />,
|
title: <T id={'inventory_valuation'} />,
|
||||||
@@ -93,6 +114,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/financial-reports/inventory-valuation',
|
link: '/financial-reports/inventory-valuation',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_VALUATION_SUMMARY,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'customers_balance_summary'} />,
|
title: <T id={'customers_balance_summary'} />,
|
||||||
@@ -104,6 +127,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/financial-reports/customers-balance-summary',
|
link: '/financial-reports/customers-balance-summary',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_SUMMARY_BALANCE,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'vendors_balance_summary'} />,
|
title: <T id={'vendors_balance_summary'} />,
|
||||||
@@ -111,6 +136,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
<T id={'summerize_the_total_amount_your_business_owes_each_vendor'} />
|
<T id={'summerize_the_total_amount_your_business_owes_each_vendor'} />
|
||||||
),
|
),
|
||||||
link: '/financial-reports/vendors-balance-summary',
|
link: '/financial-reports/vendors-balance-summary',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_SUMMARY_BALANCE,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'customers_transactions'} />,
|
title: <T id={'customers_transactions'} />,
|
||||||
@@ -120,6 +147,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/financial-reports/transactions-by-customers',
|
link: '/financial-reports/transactions-by-customers',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_TRANSACTIONS,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'vendors_transactions'} />,
|
title: <T id={'vendors_transactions'} />,
|
||||||
@@ -131,6 +160,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
link: '/financial-reports/transactions-by-vendors',
|
link: '/financial-reports/transactions-by-vendors',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_TRANSACTIONS,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: <T id={'inventory_item_details'} />,
|
title: <T id={'inventory_item_details'} />,
|
||||||
@@ -138,6 +169,8 @@ export const SalesAndPurchasesReportMenus = [
|
|||||||
<T id={'reports_every_transaction_going_in_and_out_of_your_items'} />
|
<T id={'reports_every_transaction_going_in_and_out_of_your_items'} />
|
||||||
),
|
),
|
||||||
link: '/financial-reports/inventory-item-details',
|
link: '/financial-reports/inventory-item-details',
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_ITEM_DETAILS,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,29 +1,34 @@
|
|||||||
import React from 'react'
|
import React from 'react';
|
||||||
import { FormattedMessage as T } from 'components';
|
import { FormattedMessage as T } from 'components';
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
{
|
{
|
||||||
text: <T id={'general'}/>,
|
text: <T id={'general'} />,
|
||||||
disabled: false,
|
disabled: false,
|
||||||
href: '/preferences/general',
|
href: '/preferences/general',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'users'}/>,
|
text: <T id={'users'} />,
|
||||||
href: '/preferences/users',
|
href: '/preferences/users',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'currencies'}/>,
|
text: <T id={'currencies'} />,
|
||||||
|
|
||||||
href: '/preferences/currencies',
|
href: '/preferences/currencies',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'accountant'}/>,
|
text: <T id={'accountant'} />,
|
||||||
disabled: false,
|
disabled: false,
|
||||||
href: '/preferences/accountant',
|
href: '/preferences/accountant',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'items'}/>,
|
text: <T id={'items'} />,
|
||||||
disabled: false,
|
disabled: false,
|
||||||
href: '/preferences/items',
|
href: '/preferences/items',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
text: <T id={'sms_integration.label'} />,
|
||||||
|
disabled: false,
|
||||||
|
href: '/preferences/sms-message',
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -1,5 +1,26 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { FormattedMessage as T } from 'components';
|
import { FormattedMessage as T } from 'components';
|
||||||
|
import {
|
||||||
|
ReportsAction,
|
||||||
|
AbilitySubject,
|
||||||
|
ItemAction,
|
||||||
|
InventoryAdjustmentAction,
|
||||||
|
SaleEstimateAction,
|
||||||
|
SaleInvoiceAction,
|
||||||
|
SaleReceiptAction,
|
||||||
|
PaymentReceiveAction,
|
||||||
|
BillAction,
|
||||||
|
PaymentMadeAction,
|
||||||
|
CustomerAction,
|
||||||
|
VendorAction,
|
||||||
|
AccountAction,
|
||||||
|
ManualJournalAction,
|
||||||
|
ExpenseAction,
|
||||||
|
CashflowAction,
|
||||||
|
PreferencesAbility,
|
||||||
|
ExchangeRateAbility,
|
||||||
|
SubscriptionBillingAbility,
|
||||||
|
} from '../common/abilityOption';
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
{
|
{
|
||||||
@@ -11,6 +32,32 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'sales_inventory'} />,
|
text: <T id={'sales_inventory'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.InventoryAdjustment,
|
||||||
|
ability: InventoryAdjustmentAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
ability: PaymentReceiveAction.View,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'items'} />,
|
text: <T id={'items'} />,
|
||||||
@@ -18,37 +65,70 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'items'} />,
|
text: <T id={'items'} />,
|
||||||
href: '/items',
|
href: '/items',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'inventory_adjustments'} />,
|
text: <T id={'inventory_adjustments'} />,
|
||||||
href: '/inventory-adjustments',
|
href: '/inventory-adjustments',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.InventoryAdjustment,
|
||||||
|
ability: InventoryAdjustmentAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'category_list'} />,
|
text: <T id={'category_list'} />,
|
||||||
href: '/items/categories',
|
href: '/items/categories',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New inventory item'} />,
|
text: <T id={'New inventory item'} />,
|
||||||
href: '/items/new',
|
href: '/items/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New service'} />,
|
text: <T id={'New service'} />,
|
||||||
href: '/items/new',
|
href: '/items/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New item category'} />,
|
text: <T id={'New item category'} />,
|
||||||
href: '/items/categories/new',
|
href: '/items/categories/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Item,
|
||||||
|
ability: ItemAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
// {
|
|
||||||
// text: <T id={'New inventory adjustment'} />,
|
|
||||||
// },
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -57,43 +137,117 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'estimates'} />,
|
text: <T id={'estimates'} />,
|
||||||
href: '/estimates',
|
href: '/estimates',
|
||||||
newTabHref: '/estimates/new',
|
permission: {
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'invoices'} />,
|
text: <T id={'invoices'} />,
|
||||||
href: '/invoices',
|
href: '/invoices',
|
||||||
newTabHref: '/invoices/new',
|
permission: {
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'receipts'} />,
|
text: <T id={'receipts'} />,
|
||||||
href: '/receipts',
|
href: '/receipts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: <T id={'sidebar_credit_note'} />,
|
||||||
|
href: '/credit-notes',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'payment_receives'} />,
|
text: <T id={'payment_receives'} />,
|
||||||
href: '/payment-receives',
|
href: '/payment-receives',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
ability: PaymentReceiveAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
ability: PaymentReceiveAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
ability: PaymentReceiveAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_estimate'} />,
|
text: <T id={'new_estimate'} />,
|
||||||
href: '/estimates/new',
|
href: '/estimates/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Estimate,
|
||||||
|
ability: SaleEstimateAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_invoice'} />,
|
text: <T id={'new_invoice'} />,
|
||||||
href: '/invoices/new',
|
href: '/invoices/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Invoice,
|
||||||
|
ability: SaleInvoiceAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_receipt'} />,
|
text: <T id={'new_receipt'} />,
|
||||||
href: '/receipts/new',
|
href: '/receipts/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Receipt,
|
||||||
|
ability: SaleReceiptAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: <T id={'credit_note.label.new_credit_note'} />,
|
||||||
|
href: '/credit-notes/new',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_payment_receive'} />,
|
text: <T id={'new_payment_receive'} />,
|
||||||
href: '/payment-receives/new',
|
href: '/payment-receives/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.PaymentReceive,
|
||||||
|
ability: PaymentReceiveAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -103,27 +257,74 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'bills'} />,
|
text: <T id={'bills'} />,
|
||||||
href: '/bills',
|
href: '/bills',
|
||||||
newTabHref: '/bills/new',
|
permission: {
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: <T id={'sidebar_vendor_credits'} />,
|
||||||
|
href: '/vendor-credits',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'payment_mades'} />,
|
text: <T id={'payment_mades'} />,
|
||||||
href: '/payment-mades',
|
href: '/payment-mades',
|
||||||
newTabHref: '/payment-mades/new',
|
newTabHref: '/payment-mades/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.PaymentMade,
|
||||||
|
ability: PaymentMadeAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.PaymentMade,
|
||||||
|
ability: PaymentMadeAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.Create,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.PaymentMade,
|
||||||
|
ability: PaymentMadeAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New purchase invoice'} />,
|
text: <T id={'New purchase invoice'} />,
|
||||||
href: '/bills/new',
|
href: '/bills/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.Create,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: <T id={'vendor_credits.label.new_vendor_credit'} />,
|
||||||
|
href: '/vendor-credits/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Bill,
|
||||||
|
ability: BillAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_payment_made'} />,
|
text: <T id={'new_payment_made'} />,
|
||||||
href: '/payment-mades/new',
|
href: '/payment-mades/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.PaymentMade,
|
||||||
|
ability: PaymentMadeAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -133,33 +334,77 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'customers'} />,
|
text: <T id={'customers'} />,
|
||||||
href: '/customers',
|
href: '/customers',
|
||||||
newTabHref: '/customers/new',
|
permission: {
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
ability: CustomerAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'vendors'} />,
|
text: <T id={'vendors'} />,
|
||||||
href: '/vendors',
|
href: '/vendors',
|
||||||
newTabHref: '/vendors/new',
|
permission: {
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
ability: VendorAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
ability: CustomerAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
ability: VendorAction.View,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
ability: CustomerAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
ability: VendorAction.View,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_customer'} />,
|
text: <T id={'new_customer'} />,
|
||||||
href: '/customers/new',
|
href: '/customers/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Customer,
|
||||||
|
ability: CustomerAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_vendor'} />,
|
text: <T id={'new_vendor'} />,
|
||||||
href: '/vendors/new',
|
href: '/vendors/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Vendor,
|
||||||
|
ability: VendorAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'accounting'} />,
|
text: <T id={'accounting'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Account,
|
||||||
|
ability: AccountAction.View,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.View,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'financial'} />,
|
text: <T id={'financial'} />,
|
||||||
@@ -167,25 +412,57 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'accounts_chart'} />,
|
text: <T id={'accounts_chart'} />,
|
||||||
href: '/accounts',
|
href: '/accounts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Account,
|
||||||
|
ability: AccountAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'manual_journals'} />,
|
text: <T id={'manual_journals'} />,
|
||||||
href: '/manual-journals',
|
href: '/manual-journals',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.View,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: <T id={'sidebar.transactions_locaking'} />,
|
||||||
|
href: '/transactions-locking',
|
||||||
|
// permission: {
|
||||||
|
// subject: AbilitySubject.ManualJournal,
|
||||||
|
// ability: ManualJournalAction.TransactionLocking,
|
||||||
|
// },
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'exchange_rate'} />,
|
text: <T id={'exchange_rate'} />,
|
||||||
href: '/exchange-rates',
|
href: '/exchange-rates',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.ExchangeRate,
|
||||||
|
ability: ExchangeRateAbility.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'make_journal_entry'} />,
|
text: <T id={'make_journal_entry'} />,
|
||||||
href: '/make-journal-entry',
|
href: '/make-journal-entry',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
ability: ManualJournalAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -195,29 +472,61 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'siebar.cashflow.label_cash_and_bank_accounts'} />,
|
text: <T id={'siebar.cashflow.label_cash_and_bank_accounts'} />,
|
||||||
href: '/cashflow-accounts',
|
href: '/cashflow-accounts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'cash_flow.label.add_money_in'} />,
|
text: <T id={'cash_flow.label.add_money_in'} />,
|
||||||
href: '/cashflow-accounts',
|
href: '/cashflow-accounts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'cash_flow.label.add_money_out'} />,
|
text: <T id={'cash_flow.label.add_money_out'} />,
|
||||||
href: '/cashflow-accounts',
|
href: '/cashflow-accounts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'cash_flow.label.add_cash_account'} />,
|
text: <T id={'cash_flow.label.add_cash_account'} />,
|
||||||
href: '/cashflow-accounts',
|
href: '/cashflow-accounts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'cash_flow.label.add_bank_account'} />,
|
text: <T id={'cash_flow.label.add_bank_account'} />,
|
||||||
href: '/cashflow-accounts',
|
href: '/cashflow-accounts',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Cashflow,
|
||||||
|
ability: CashflowAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -227,17 +536,33 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'expenses'} />,
|
text: <T id={'expenses'} />,
|
||||||
href: '/expenses',
|
href: '/expenses',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
ability: ExpenseAction.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'New tasks'} />,
|
text: <T id={'New tasks'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
ability: ExpenseAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
ability: ExpenseAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'new_expense'} />,
|
text: <T id={'new_expense'} />,
|
||||||
href: '/expenses/new',
|
href: '/expenses/new',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Expense,
|
||||||
|
ability: ExpenseAction.Create,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -247,80 +572,216 @@ export default [
|
|||||||
{
|
{
|
||||||
text: <T id={'balance_sheet'} />,
|
text: <T id={'balance_sheet'} />,
|
||||||
href: '/financial-reports/balance-sheet',
|
href: '/financial-reports/balance-sheet',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_BALANCE_SHEET,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'trial_balance_sheet'} />,
|
text: <T id={'trial_balance_sheet'} />,
|
||||||
href: '/financial-reports/trial-balance-sheet',
|
href: '/financial-reports/trial-balance-sheet',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_TRIAL_BALANCE_SHEET,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'journal'} />,
|
text: <T id={'journal'} />,
|
||||||
href: '/financial-reports/journal-sheet',
|
href: '/financial-reports/journal-sheet',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_JOURNAL,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'general_ledger'} />,
|
text: <T id={'general_ledger'} />,
|
||||||
href: '/financial-reports/general-ledger',
|
href: '/financial-reports/general-ledger',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_GENERAL_LEDGET,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'profit_loss_sheet'} />,
|
text: <T id={'profit_loss_sheet'} />,
|
||||||
href: '/financial-reports/profit-loss-sheet',
|
href: '/financial-reports/profit-loss-sheet',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_PROFIT_LOSS,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'cash_flow_statement'} />,
|
text: <T id={'cash_flow_statement'} />,
|
||||||
href: '/financial-reports/cash-flow',
|
href: '/financial-reports/cash-flow',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CASHFLOW_ACCOUNT_TRANSACTION,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'AR_Aging_Summary'} />,
|
text: <T id={'AR_Aging_Summary'} />,
|
||||||
href: '/financial-reports/receivable-aging-summary',
|
href: '/financial-reports/receivable-aging-summary',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_AR_AGING_SUMMARY,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'AP_Aging_Summary'} />,
|
text: <T id={'AP_Aging_Summary'} />,
|
||||||
href: '/financial-reports/payable-aging-summary',
|
href: '/financial-reports/payable-aging-summary',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_AP_AGING_SUMMARY,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'Sales/Purchases'} />,
|
text: <T id={'Sales/Purchases'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_PURCHASES_BY_ITEMS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_SALES_BY_ITEMS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_TRANSACTIONS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_TRANSACTIONS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_PURCHASES_BY_ITEMS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_SALES_BY_ITEMS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_TRANSACTIONS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_TRANSACTIONS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'purchases_by_items'} />,
|
text: <T id={'purchases_by_items'} />,
|
||||||
href: '/financial-reports/purchases-by-items',
|
href: '/financial-reports/purchases-by-items',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_PURCHASES_BY_ITEMS,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'sales_by_items'} />,
|
text: <T id={'sales_by_items'} />,
|
||||||
href: '/financial-reports/sales-by-items',
|
href: '/financial-reports/sales-by-items',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_SALES_BY_ITEMS,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'customers_transactions'} />,
|
text: <T id={'customers_transactions'} />,
|
||||||
href: '/financial-reports/transactions-by-customers',
|
href: '/financial-reports/transactions-by-customers',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_TRANSACTIONS,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'vendors_transactions'} />,
|
text: <T id={'vendors_transactions'} />,
|
||||||
href: '/financial-reports/transactions-by-vendors',
|
href: '/financial-reports/transactions-by-vendors',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_TRANSACTIONS,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'customers_balance_summary'} />,
|
text: <T id={'customers_balance_summary'} />,
|
||||||
href: '/financial-reports/customers-balance-summary',
|
href: '/financial-reports/customers-balance-summary',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_CUSTOMERS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'vendors_balance_summary'} />,
|
text: <T id={'vendors_balance_summary'} />,
|
||||||
href: '/financial-reports/vendors-balance-summary',
|
href: '/financial-reports/vendors-balance-summary',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_VENDORS_SUMMARY_BALANCE,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'inventory'} />,
|
text: <T id={'inventory'} />,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_ITEM_DETAILS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_VALUATION_SUMMARY,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
divider: true,
|
divider: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_ITEM_DETAILS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_VALUATION_SUMMARY,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'inventory_item_details'} />,
|
text: <T id={'inventory_item_details'} />,
|
||||||
href: '/financial-reports/inventory-item-details',
|
href: '/financial-reports/inventory-item-details',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_ITEM_DETAILS,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'inventory_valuation'} />,
|
text: <T id={'inventory_valuation'} />,
|
||||||
href: '/financial-reports/inventory-valuation',
|
href: '/financial-reports/inventory-valuation',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Report,
|
||||||
|
ability: ReportsAction.READ_INVENTORY_VALUATION_SUMMARY,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -328,14 +789,32 @@ export default [
|
|||||||
text: <T id={'system'} />,
|
text: <T id={'system'} />,
|
||||||
enableBilling: true,
|
enableBilling: true,
|
||||||
label: true,
|
label: true,
|
||||||
|
permission: [
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.Preferences,
|
||||||
|
ability: PreferencesAbility.Mutate,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
subject: AbilitySubject.SubscriptionBilling,
|
||||||
|
ability: SubscriptionBillingAbility.View,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'preferences'} />,
|
text: <T id={'preferences'} />,
|
||||||
href: '/preferences',
|
href: '/preferences',
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.Preferences,
|
||||||
|
ability: PreferencesAbility.Mutate,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text: <T id={'billing'} />,
|
text: <T id={'billing'} />,
|
||||||
href: '/billing',
|
href: '/billing',
|
||||||
enableBilling: true,
|
enableBilling: true,
|
||||||
|
permission: {
|
||||||
|
subject: AbilitySubject.SubscriptionBilling,
|
||||||
|
ability: SubscriptionBillingAbility.View,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -26,8 +26,11 @@ import withManualJournals from './withManualJournals';
|
|||||||
import withSettingsActions from '../../Settings/withSettingsActions';
|
import withSettingsActions from '../../Settings/withSettingsActions';
|
||||||
import withSettings from '../../Settings/withSettings';
|
import withSettings from '../../Settings/withSettings';
|
||||||
|
|
||||||
import { If, DashboardActionViewsList } from 'components';
|
import { Can, If, DashboardActionViewsList } from 'components';
|
||||||
|
import {
|
||||||
|
ManualJournalAction,
|
||||||
|
AbilitySubject,
|
||||||
|
} from '../../../common/abilityOption';
|
||||||
import { compose } from 'utils';
|
import { compose } from 'utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -86,13 +89,14 @@ function ManualJournalActionsBar({
|
|||||||
onChange={handleTabChange}
|
onChange={handleTabChange}
|
||||||
/>
|
/>
|
||||||
<NavbarDivider />
|
<NavbarDivider />
|
||||||
|
<Can I={ManualJournalAction.Create} a={AbilitySubject.ManualJournal}>
|
||||||
<Button
|
<Button
|
||||||
className={Classes.MINIMAL}
|
className={Classes.MINIMAL}
|
||||||
icon={<Icon icon="plus" />}
|
icon={<Icon icon="plus" />}
|
||||||
text={<T id={'journal_entry'} />}
|
text={<T id={'journal_entry'} />}
|
||||||
onClick={onClickNewManualJournal}
|
onClick={onClickNewManualJournal}
|
||||||
/>
|
/>
|
||||||
|
</Can>
|
||||||
<AdvancedFilterPopover
|
<AdvancedFilterPopover
|
||||||
advancedFilterProps={{
|
advancedFilterProps={{
|
||||||
conditions: manualJournalsFilterConditions,
|
conditions: manualJournalsFilterConditions,
|
||||||
|
|||||||
@@ -2,7 +2,11 @@ import React from 'react';
|
|||||||
import { Button, Intent } from '@blueprintjs/core';
|
import { Button, Intent } from '@blueprintjs/core';
|
||||||
import { useHistory } from 'react-router-dom';
|
import { useHistory } from 'react-router-dom';
|
||||||
import { EmptyStatus } from 'components';
|
import { EmptyStatus } from 'components';
|
||||||
import { FormattedMessage as T } from 'components';
|
import { Can, FormattedMessage as T } from 'components';
|
||||||
|
import {
|
||||||
|
AbilitySubject,
|
||||||
|
ManualJournalAction,
|
||||||
|
} from '../../../common/abilityOption';
|
||||||
|
|
||||||
export default function ManualJournalsEmptyStatus() {
|
export default function ManualJournalsEmptyStatus() {
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
@@ -17,19 +21,21 @@ export default function ManualJournalsEmptyStatus() {
|
|||||||
}
|
}
|
||||||
action={
|
action={
|
||||||
<>
|
<>
|
||||||
<Button
|
<Can I={ManualJournalAction.Create} a={AbilitySubject.ManualJournal}>
|
||||||
intent={Intent.PRIMARY}
|
<Button
|
||||||
large={true}
|
intent={Intent.PRIMARY}
|
||||||
onClick={() => {
|
large={true}
|
||||||
history.push('/make-journal-entry');
|
onClick={() => {
|
||||||
}}
|
history.push('/make-journal-entry');
|
||||||
>
|
}}
|
||||||
<T id={'make_journal'} />
|
>
|
||||||
</Button>
|
<T id={'make_journal'} />
|
||||||
|
</Button>
|
||||||
|
|
||||||
<Button intent={Intent.NONE} large={true}>
|
<Button intent={Intent.NONE} large={true}>
|
||||||
<T id={'learn_more'} />
|
<T id={'learn_more'} />
|
||||||
</Button>
|
</Button>
|
||||||
|
</Can>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -46,8 +46,7 @@ function ManualJournalsListProvider({ query, tableStateChanged, ...props }) {
|
|||||||
isEmptyStatus,
|
isEmptyStatus,
|
||||||
};
|
};
|
||||||
|
|
||||||
const isPageLoading =
|
const isPageLoading = isViewsLoading || isResourceMetaLoading;
|
||||||
isManualJournalsLoading || isViewsLoading || isResourceMetaLoading;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DashboardInsider loading={isPageLoading} name={'manual-journals'}>
|
<DashboardInsider loading={isPageLoading} name={'manual-journals'}>
|
||||||
|
|||||||
@@ -13,7 +13,18 @@ import {
|
|||||||
} from '@blueprintjs/core';
|
} from '@blueprintjs/core';
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
|
|
||||||
import { FormattedMessage as T, Choose, Money, If, Icon } from 'components';
|
import {
|
||||||
|
Can,
|
||||||
|
FormattedMessage as T,
|
||||||
|
Choose,
|
||||||
|
Money,
|
||||||
|
If,
|
||||||
|
Icon,
|
||||||
|
} from 'components';
|
||||||
|
import {
|
||||||
|
ManualJournalAction,
|
||||||
|
AbilitySubject,
|
||||||
|
} from '../../../common/abilityOption';
|
||||||
import { safeCallback } from 'utils';
|
import { safeCallback } from 'utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -90,13 +101,13 @@ export const StatusAccessor = (row) => {
|
|||||||
return (
|
return (
|
||||||
<Choose>
|
<Choose>
|
||||||
<Choose.When condition={!!row.is_published}>
|
<Choose.When condition={!!row.is_published}>
|
||||||
<Tag minimal={true}>
|
<Tag minimal={true} round={true}>
|
||||||
<T id={'published'} />
|
<T id={'published'} />
|
||||||
</Tag>
|
</Tag>
|
||||||
</Choose.When>
|
</Choose.When>
|
||||||
|
|
||||||
<Choose.Otherwise>
|
<Choose.Otherwise>
|
||||||
<Tag minimal={true} intent={Intent.WARNING}>
|
<Tag minimal={true} intent={Intent.WARNING} round={true}>
|
||||||
<T id={'draft'} />
|
<T id={'draft'} />
|
||||||
</Tag>
|
</Tag>
|
||||||
</Choose.Otherwise>
|
</Choose.Otherwise>
|
||||||
@@ -150,25 +161,32 @@ export const ActionsMenu = ({
|
|||||||
text={intl.get('view_details')}
|
text={intl.get('view_details')}
|
||||||
onClick={safeCallback(onViewDetails, original)}
|
onClick={safeCallback(onViewDetails, original)}
|
||||||
/>
|
/>
|
||||||
<MenuDivider />
|
<Can I={ManualJournalAction.Edit} a={AbilitySubject.ManualJournal}>
|
||||||
<If condition={!original.is_published}>
|
<MenuDivider />
|
||||||
|
<If condition={!original.is_published}>
|
||||||
|
<MenuItem
|
||||||
|
icon={<Icon icon="arrow-to-top" />}
|
||||||
|
text={intl.get('publish_journal')}
|
||||||
|
onClick={safeCallback(onPublish, original)}
|
||||||
|
/>
|
||||||
|
</If>
|
||||||
|
</Can>
|
||||||
|
<Can I={ManualJournalAction.Edit} a={AbilitySubject.ManualJournal}>
|
||||||
<MenuItem
|
<MenuItem
|
||||||
icon={<Icon icon="arrow-to-top" />}
|
icon={<Icon icon="pen-18" />}
|
||||||
text={intl.get('publish_journal')}
|
text={intl.get('edit_journal')}
|
||||||
onClick={safeCallback(onPublish, original)}
|
onClick={safeCallback(onEdit, original)}
|
||||||
/>
|
/>
|
||||||
</If>
|
</Can>
|
||||||
<MenuItem
|
<Can I={ManualJournalAction.Delete} a={AbilitySubject.ManualJournal}>
|
||||||
icon={<Icon icon="pen-18" />}
|
<MenuDivider />
|
||||||
text={intl.get('edit_journal')}
|
<MenuItem
|
||||||
onClick={safeCallback(onEdit, original)}
|
text={intl.get('delete_journal')}
|
||||||
/>
|
icon={<Icon icon="trash-16" iconSize={16} />}
|
||||||
<MenuItem
|
intent={Intent.DANGER}
|
||||||
text={intl.get('delete_journal')}
|
onClick={safeCallback(onDelete, original)}
|
||||||
icon={<Icon icon="trash-16" iconSize={16} />}
|
/>
|
||||||
intent={Intent.DANGER}
|
</Can>
|
||||||
onClick={safeCallback(onDelete, original)}
|
|
||||||
/>
|
|
||||||
</Menu>
|
</Menu>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -151,6 +151,7 @@ export default function MakeJournalFloatingAction() {
|
|||||||
disabled={isSubmitting}
|
disabled={isSubmitting}
|
||||||
intent={Intent.PRIMARY}
|
intent={Intent.PRIMARY}
|
||||||
onClick={handleSubmitPublishBtnClick}
|
onClick={handleSubmitPublishBtnClick}
|
||||||
|
style={{ minWidth: '85px' }}
|
||||||
text={<T id={'save'} />}
|
text={<T id={'save'} />}
|
||||||
/>
|
/>
|
||||||
<Popover
|
<Popover
|
||||||
|
|||||||
@@ -7,7 +7,8 @@ import {
|
|||||||
useJournal,
|
useJournal,
|
||||||
useCreateJournal,
|
useCreateJournal,
|
||||||
useEditJournal,
|
useEditJournal,
|
||||||
useSettings
|
useSettings,
|
||||||
|
useSettingsManualJournals
|
||||||
} from 'hooks/query';
|
} from 'hooks/query';
|
||||||
|
|
||||||
const MakeJournalFormContext = createContext();
|
const MakeJournalFormContext = createContext();
|
||||||
@@ -40,7 +41,7 @@ function MakeJournalProvider({ journalId, ...props }) {
|
|||||||
const { mutateAsync: editJournalMutate } = useEditJournal();
|
const { mutateAsync: editJournalMutate } = useEditJournal();
|
||||||
|
|
||||||
// Loading the journal settings.
|
// Loading the journal settings.
|
||||||
const { isLoading: isSettingsLoading } = useSettings();
|
const { isLoading: isSettingsLoading } = useSettingsManualJournals();
|
||||||
|
|
||||||
// Submit form payload.
|
// Submit form payload.
|
||||||
const [submitPayload, setSubmitPayload] = useState({});
|
const [submitPayload, setSubmitPayload] = useState({});
|
||||||
|
|||||||
@@ -128,6 +128,7 @@ export const useJournalTableEntriesColumns = () => {
|
|||||||
className: 'account',
|
className: 'account',
|
||||||
disableSortBy: true,
|
disableSortBy: true,
|
||||||
width: 160,
|
width: 160,
|
||||||
|
fieldProps: { allowCreate: true }
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Header: CreditHeaderCell,
|
Header: CreditHeaderCell,
|
||||||
|
|||||||
@@ -1,6 +1,10 @@
|
|||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { RESOURCES_TYPES } from 'common/resourcesTypes';
|
import { RESOURCES_TYPES } from 'common/resourcesTypes';
|
||||||
import withDrawerActions from '../Drawer/withDrawerActions';
|
import withDrawerActions from '../Drawer/withDrawerActions';
|
||||||
|
import {
|
||||||
|
AbilitySubject,
|
||||||
|
ManualJournalAction,
|
||||||
|
} from '../../common/abilityOption';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Universal search manual journal item select action.
|
* Universal search manual journal item select action.
|
||||||
@@ -44,4 +48,8 @@ export const universalSearchJournalBind = () => ({
|
|||||||
optionItemLabel: intl.get('manual_journals'),
|
optionItemLabel: intl.get('manual_journals'),
|
||||||
selectItemAction: JournalUniversalSearchSelectAction,
|
selectItemAction: JournalUniversalSearchSelectAction,
|
||||||
itemSelect: manualJournalsToSearch,
|
itemSelect: manualJournalsToSearch,
|
||||||
|
permission: {
|
||||||
|
ability: ManualJournalAction.View,
|
||||||
|
subject: AbilitySubject.ManualJournal,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { RESOURCES_TYPES } from '../../common/resourcesTypes';
|
|
||||||
import withDrawerActions from '../Drawer/withDrawerActions';
|
import withDrawerActions from '../Drawer/withDrawerActions';
|
||||||
|
|
||||||
|
import { AbilitySubject, AccountAction } from '../../common/abilityOption';
|
||||||
|
import { RESOURCES_TYPES } from '../../common/resourcesTypes';
|
||||||
|
|
||||||
function AccountUniversalSearchItemSelectComponent({
|
function AccountUniversalSearchItemSelectComponent({
|
||||||
// #ownProps
|
// #ownProps
|
||||||
resourceType,
|
resourceType,
|
||||||
@@ -42,4 +45,8 @@ export const universalSearchAccountBind = () => ({
|
|||||||
optionItemLabel: intl.get('accounts'),
|
optionItemLabel: intl.get('accounts'),
|
||||||
selectItemAction: AccountUniversalSearchItemSelect,
|
selectItemAction: AccountUniversalSearchItemSelect,
|
||||||
itemSelect: accountToSearch,
|
itemSelect: accountToSearch,
|
||||||
|
permission: {
|
||||||
|
ability: AccountAction.View,
|
||||||
|
subject: AbilitySubject.Account,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import { FormattedMessage as T } from 'components';
|
|||||||
import {
|
import {
|
||||||
AdvancedFilterPopover,
|
AdvancedFilterPopover,
|
||||||
If,
|
If,
|
||||||
|
Can,
|
||||||
DashboardActionViewsList,
|
DashboardActionViewsList,
|
||||||
DashboardFilterButton,
|
DashboardFilterButton,
|
||||||
DashboardRowsHeightButton,
|
DashboardRowsHeightButton,
|
||||||
@@ -30,6 +31,8 @@ import withAlertActions from 'containers/Alert/withAlertActions';
|
|||||||
import withAccountsTableActions from './withAccountsTableActions';
|
import withAccountsTableActions from './withAccountsTableActions';
|
||||||
import withSettings from '../Settings/withSettings';
|
import withSettings from '../Settings/withSettings';
|
||||||
import withSettingsActions from '../Settings/withSettingsActions';
|
import withSettingsActions from '../Settings/withSettingsActions';
|
||||||
|
import { AccountAction, AbilitySubject } from '../../common/abilityOption';
|
||||||
|
|
||||||
import { compose } from 'utils';
|
import { compose } from 'utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -116,13 +119,14 @@ function AccountsActionsBar({
|
|||||||
onChange={handleTabChange}
|
onChange={handleTabChange}
|
||||||
/>
|
/>
|
||||||
<NavbarDivider />
|
<NavbarDivider />
|
||||||
|
<Can I={AccountAction.Create} a={AbilitySubject.Account}>
|
||||||
<Button
|
<Button
|
||||||
className={Classes.MINIMAL}
|
className={Classes.MINIMAL}
|
||||||
icon={<Icon icon="plus" />}
|
icon={<Icon icon="plus" />}
|
||||||
text={<T id={'new_account'} />}
|
text={<T id={'new_account'} />}
|
||||||
onClick={onClickNewAccount}
|
onClick={onClickNewAccount}
|
||||||
/>
|
/>
|
||||||
|
</Can>
|
||||||
<AdvancedFilterPopover
|
<AdvancedFilterPopover
|
||||||
advancedFilterProps={{
|
advancedFilterProps={{
|
||||||
conditions: accountsFilterConditions,
|
conditions: accountsFilterConditions,
|
||||||
@@ -183,11 +187,13 @@ function AccountsActionsBar({
|
|||||||
onChange={handleTableRowSizeChange}
|
onChange={handleTableRowSizeChange}
|
||||||
/>
|
/>
|
||||||
<NavbarDivider />
|
<NavbarDivider />
|
||||||
<Switch
|
<Can I={AccountAction.Edit} a={AbilitySubject.Account}>
|
||||||
labelElement={<T id={'inactive'} />}
|
<Switch
|
||||||
defaultChecked={accountsInactiveMode}
|
labelElement={<T id={'inactive'} />}
|
||||||
onChange={handleInactiveSwitchChange}
|
defaultChecked={accountsInactiveMode}
|
||||||
/>
|
onChange={handleInactiveSwitchChange}
|
||||||
|
/>
|
||||||
|
</Can>
|
||||||
</NavbarGroup>
|
</NavbarGroup>
|
||||||
<NavbarGroup align={Alignment.RIGHT}>
|
<NavbarGroup align={Alignment.RIGHT}>
|
||||||
<Button
|
<Button
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user