mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-19 14:20:31 +00:00
Merge branch 'BIG-374-refactoring-sidebar-menu-with-feature-and-permissions-control' into develop
This commit is contained in:
@@ -2,6 +2,13 @@
|
|||||||
|
|
||||||
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.7.3-rc.2] - 15-04-2022
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
- `BIG-372` Activate branches and warehouses dialog reloading once activating.
|
||||||
|
- `BIG-373` Issue general ledger report select specific account.
|
||||||
|
- `BIG-377` Make readonly details entries as oneline with tooltip for more details.
|
||||||
|
|
||||||
## [1.7.2-rc.2] - 04-04-2022
|
## [1.7.2-rc.2] - 04-04-2022
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|||||||
42
package-lock.json
generated
42
package-lock.json
generated
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "bigcapital-client",
|
"name": "bigcapital-client",
|
||||||
"version": "1.5.8",
|
"version": "1.7.1",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -1043,17 +1043,6 @@
|
|||||||
"to-fast-properties": "^2.0.0"
|
"to-fast-properties": "^2.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@blueprintjs-formik/core": {
|
|
||||||
"version": "0.1.5",
|
|
||||||
"resolved": "https://registry.npmjs.org/@blueprintjs-formik/core/-/core-0.1.5.tgz",
|
|
||||||
"integrity": "sha512-H0aXiNMYC8RwhWR1F2O77dcRcRITijUX5we51G4AK2Vmp1yXCmNb0piN9ftsjL5vVIvIsMKWg+dfbwREmB5VWg==",
|
|
||||||
"requires": {
|
|
||||||
"lodash.get": "^4.4.2",
|
|
||||||
"lodash.keyby": "^4.6.0",
|
|
||||||
"styled-components": "^5.3.3",
|
|
||||||
"web-vitals": "^2.1.4"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"@blueprintjs-formik/select": {
|
"@blueprintjs-formik/select": {
|
||||||
"version": "0.1.4",
|
"version": "0.1.4",
|
||||||
"resolved": "https://registry.npmjs.org/@blueprintjs-formik/select/-/select-0.1.4.tgz",
|
"resolved": "https://registry.npmjs.org/@blueprintjs-formik/select/-/select-0.1.4.tgz",
|
||||||
@@ -1980,6 +1969,16 @@
|
|||||||
"integrity": "sha512-MUc6zSmU3tEVnkQ78q0peeEjKWPUADMlC/t++2bI8WnAG2tvYRPIgHG8lWkXwqc8MsUF6Z2MOf+Mh5sazOmhiQ==",
|
"integrity": "sha512-MUc6zSmU3tEVnkQ78q0peeEjKWPUADMlC/t++2bI8WnAG2tvYRPIgHG8lWkXwqc8MsUF6Z2MOf+Mh5sazOmhiQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"@types/hoist-non-react-statics": {
|
||||||
|
"version": "3.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.1.tgz",
|
||||||
|
"integrity": "sha512-iMIqiko6ooLrTh1joXodJK5X9xeEALT1kM5G3ZLhD3hszxBdIEd5C75U834D9mLcINgD4OyZf5uQXjkuYydWvA==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@types/react": "*",
|
||||||
|
"hoist-non-react-statics": "^3.3.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"@types/http-proxy": {
|
"@types/http-proxy": {
|
||||||
"version": "1.17.3",
|
"version": "1.17.3",
|
||||||
"resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.3.tgz",
|
"resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.3.tgz",
|
||||||
@@ -2240,6 +2239,25 @@
|
|||||||
"resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-1.0.1.tgz",
|
||||||
"integrity": "sha512-l42BggppR6zLmpfU6fq9HEa2oGPEI8yrSPL3GITjfRInppYFahObbIQOQK3UGxEnyQpltZLaPe75046NOZQikw=="
|
"integrity": "sha512-l42BggppR6zLmpfU6fq9HEa2oGPEI8yrSPL3GITjfRInppYFahObbIQOQK3UGxEnyQpltZLaPe75046NOZQikw=="
|
||||||
},
|
},
|
||||||
|
"@types/styled-components": {
|
||||||
|
"version": "5.1.25",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/styled-components/-/styled-components-5.1.25.tgz",
|
||||||
|
"integrity": "sha512-fgwl+0Pa8pdkwXRoVPP9JbqF0Ivo9llnmsm+7TCI330kbPIFd9qv1Lrhr37shf4tnxCOSu+/IgqM7uJXLWZZNQ==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@types/hoist-non-react-statics": "*",
|
||||||
|
"@types/react": "*",
|
||||||
|
"csstype": "^3.0.2"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"csstype": {
|
||||||
|
"version": "3.0.11",
|
||||||
|
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.0.11.tgz",
|
||||||
|
"integrity": "sha512-sa6P2wJ+CAbgyy4KFssIb/JNMLxFvKF1pCYCSXS8ZMuqZnMsrxqI2E5sPyoTpxoPU/gVZMzr2zjOfg8GIZOMsw==",
|
||||||
|
"dev": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@types/testing-library__dom": {
|
"@types/testing-library__dom": {
|
||||||
"version": "6.12.1",
|
"version": "6.12.1",
|
||||||
"resolved": "https://registry.npmjs.org/@types/testing-library__dom/-/testing-library__dom-6.12.1.tgz",
|
"resolved": "https://registry.npmjs.org/@types/testing-library__dom/-/testing-library__dom-6.12.1.tgz",
|
||||||
|
|||||||
@@ -38,6 +38,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",
|
||||||
|
"deepdash": "^5.3.9",
|
||||||
"dependency-graph": "^0.11.0",
|
"dependency-graph": "^0.11.0",
|
||||||
"dotenv": "8.2.0",
|
"dotenv": "8.2.0",
|
||||||
"dotenv-expand": "5.1.0",
|
"dotenv-expand": "5.1.0",
|
||||||
@@ -150,6 +151,7 @@
|
|||||||
"@types/react": "^17.0.0",
|
"@types/react": "^17.0.0",
|
||||||
"@types/react-dom": "^17.0.0",
|
"@types/react-dom": "^17.0.0",
|
||||||
"@types/react-router-dom": "^5.1.8",
|
"@types/react-router-dom": "^5.1.8",
|
||||||
|
"@types/styled-components": "^5.1.25",
|
||||||
"@types/yup": "^0.29.13",
|
"@types/yup": "^0.29.13",
|
||||||
"@welldone-software/why-did-you-render": "^6.0.0-rc.1",
|
"@welldone-software/why-did-you-render": "^6.0.0-rc.1",
|
||||||
"compression-webpack-plugin": "^6.1.0",
|
"compression-webpack-plugin": "^6.1.0",
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { Switch, Route } from 'react-router';
|
|||||||
|
|
||||||
import 'style/pages/Dashboard/Dashboard.scss';
|
import 'style/pages/Dashboard/Dashboard.scss';
|
||||||
|
|
||||||
import Sidebar from 'components/Sidebar/Sidebar';
|
import { Sidebar } from 'containers/Dashboard/Sidebar/Sidebar';
|
||||||
import DashboardContent from 'components/Dashboard/DashboardContent';
|
import DashboardContent from 'components/Dashboard/DashboardContent';
|
||||||
import DialogsContainer from 'components/DialogsContainer';
|
import DialogsContainer from 'components/DialogsContainer';
|
||||||
import PreferencesPage from 'components/Preferences/PreferencesPage';
|
import PreferencesPage from 'components/Preferences/PreferencesPage';
|
||||||
|
|||||||
16
src/components/FlexGrid/Flex.style.tsx
Normal file
16
src/components/FlexGrid/Flex.style.tsx
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import styled from 'styled-components';
|
||||||
|
import { FlexProps } from './interfaces';
|
||||||
|
|
||||||
|
export const FlexStyled = styled.div<FlexProps>`
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-items: ${({ align }) => align || 'center'};
|
||||||
|
|
||||||
|
&:after {
|
||||||
|
content: '';
|
||||||
|
max-width: ${({ col, gap = 1 }) =>
|
||||||
|
col && col < 12 ? `${(100 * col) / 12 - gap}%` : '100%'};
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
`;
|
||||||
26
src/components/FlexGrid/Flex.tsx
Normal file
26
src/components/FlexGrid/Flex.tsx
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
import * as React from 'react';
|
||||||
|
import { FlexProps } from './interfaces';
|
||||||
|
import { FlexItem } from './FlexItem';
|
||||||
|
import { FlexStyled } from './Flex.style';
|
||||||
|
|
||||||
|
export function Flex({
|
||||||
|
children,
|
||||||
|
col = 12,
|
||||||
|
gap,
|
||||||
|
align,
|
||||||
|
className,
|
||||||
|
style,
|
||||||
|
}: FlexProps) {
|
||||||
|
return (
|
||||||
|
<FlexStyled
|
||||||
|
col={col}
|
||||||
|
gap={gap}
|
||||||
|
align={align}
|
||||||
|
className={className}
|
||||||
|
style={style}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<FlexItem col={col} gap={gap} />
|
||||||
|
</FlexStyled>
|
||||||
|
);
|
||||||
|
}
|
||||||
19
src/components/FlexGrid/FlexItem.style.tsx
Normal file
19
src/components/FlexGrid/FlexItem.style.tsx
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import styled from 'styled-components';
|
||||||
|
import { ItemProps } from './interfaces';
|
||||||
|
|
||||||
|
export const FlexItem = styled.div<ItemProps>`
|
||||||
|
width: 100%;
|
||||||
|
max-width: ${({ col, gap = 1 }) =>
|
||||||
|
col && col < 12 ? `${(100 * col) / 12 - gap}%` : '100%'};
|
||||||
|
${({ marginBottom }) =>
|
||||||
|
marginBottom &&
|
||||||
|
`
|
||||||
|
margin-bottom: ${marginBottom}px;
|
||||||
|
`}
|
||||||
|
${({ stretch }) =>
|
||||||
|
stretch &&
|
||||||
|
`
|
||||||
|
display: flex;
|
||||||
|
align-self: stretch;
|
||||||
|
`}
|
||||||
|
`;
|
||||||
1
src/components/FlexGrid/FlexItem.tsx
Normal file
1
src/components/FlexGrid/FlexItem.tsx
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './FlexItem.style';
|
||||||
4
src/components/FlexGrid/index.ts
Normal file
4
src/components/FlexGrid/index.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
export * from './Flex.style';
|
||||||
|
export * from './Flex';
|
||||||
|
export * from './FlexItem.style';
|
||||||
|
export * from './interfaces';
|
||||||
22
src/components/FlexGrid/interfaces.ts
Normal file
22
src/components/FlexGrid/interfaces.ts
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import { HTMLAttributes, Component, StyleHTMLAttributes } from 'react';
|
||||||
|
|
||||||
|
export type Range = 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12;
|
||||||
|
|
||||||
|
export interface ItemProps extends HTMLAttributes<HTMLDivElement> {
|
||||||
|
gap?: number;
|
||||||
|
col: Range;
|
||||||
|
marginBottom?: number;
|
||||||
|
stretch?: boolean;
|
||||||
|
as?: string | Component;
|
||||||
|
className?: string;
|
||||||
|
style?: StyleHTMLAttributes<HTMLDivElement>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FlexProps extends HTMLAttributes<HTMLDivElement> {
|
||||||
|
gap?: number;
|
||||||
|
align?: 'flex-start' | 'flex-end' | 'center' | 'baseline' | 'stretch';
|
||||||
|
col?: Range;
|
||||||
|
className?: string;
|
||||||
|
style?: StyleHTMLAttributes<HTMLDivElement>;
|
||||||
|
as?: string | Component;
|
||||||
|
}
|
||||||
@@ -104,7 +104,7 @@ import {
|
|||||||
// textClassName?: string;
|
// textClassName?: string;
|
||||||
// }
|
// }
|
||||||
|
|
||||||
export default class MenuItem extends AbstractPureComponent2 {
|
export class MenuItem extends AbstractPureComponent2 {
|
||||||
static get defaultProps() {
|
static get defaultProps() {
|
||||||
return {
|
return {
|
||||||
disabled: false,
|
disabled: false,
|
||||||
|
|||||||
@@ -1,123 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import { Menu, MenuDivider } from '@blueprintjs/core';
|
|
||||||
import { useHistory, useLocation } from 'react-router-dom';
|
|
||||||
|
|
||||||
import { Choose } from 'components';
|
|
||||||
import Icon from 'components/Icon';
|
|
||||||
import MenuItem from 'components/MenuItem';
|
|
||||||
import { MenuItemLabel } from 'components';
|
|
||||||
import classNames from 'classnames';
|
|
||||||
import SidebarOverlay from 'components/SidebarOverlay';
|
|
||||||
import { compose } from 'redux';
|
|
||||||
import withSubscriptions from '../../containers/Subscriptions/withSubscriptions';
|
|
||||||
|
|
||||||
const DEFAULT_ITEM = {
|
|
||||||
text: '',
|
|
||||||
href: '',
|
|
||||||
};
|
|
||||||
|
|
||||||
function matchPath(pathname, path, matchExact) {
|
|
||||||
return matchExact ? pathname === path : pathname.indexOf(path) !== -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
function SidebarMenuItemSpace({ space }) {
|
|
||||||
return <div class="bp3-menu-spacer" style={{ height: `${space}px` }} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
function SidebarMenu({ menu, isSubscriptionActive }) {
|
|
||||||
const history = useHistory();
|
|
||||||
const location = useLocation();
|
|
||||||
|
|
||||||
const [isOpen, setIsOpen] = React.useState(false);
|
|
||||||
const [currentItem, setCurrentItem] = React.useState(null);
|
|
||||||
|
|
||||||
const menuItemsMapper = (list) => {
|
|
||||||
return list.map((item, index) => {
|
|
||||||
const hasChildren = Array.isArray(item.children);
|
|
||||||
|
|
||||||
const isActive =
|
|
||||||
(item.children
|
|
||||||
? item.children.some((c) =>
|
|
||||||
matchPath(location.pathname, c.href, item.matchExact),
|
|
||||||
)
|
|
||||||
: item.href &&
|
|
||||||
matchPath(location.pathname, item.href, item.matchExact)) ||
|
|
||||||
currentItem === item;
|
|
||||||
|
|
||||||
const handleItemClick = () => {
|
|
||||||
if (item.href) {
|
|
||||||
history.push(item.href);
|
|
||||||
}
|
|
||||||
if (item.children && item.children.length > 0) {
|
|
||||||
setIsOpen(true);
|
|
||||||
setCurrentItem(item);
|
|
||||||
} else {
|
|
||||||
setIsOpen(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Choose>
|
|
||||||
<Choose.When condition={item.spacer}>
|
|
||||||
<SidebarMenuItemSpace space={item.spacer} />
|
|
||||||
</Choose.When>
|
|
||||||
|
|
||||||
<Choose.When condition={item.divider}>
|
|
||||||
<MenuDivider key={index} title={item.title} />
|
|
||||||
</Choose.When>
|
|
||||||
|
|
||||||
<Choose.When condition={item.label}>
|
|
||||||
<MenuItemLabel key={index} text={item.text} />
|
|
||||||
</Choose.When>
|
|
||||||
|
|
||||||
<Choose.Otherwise>
|
|
||||||
<MenuItem
|
|
||||||
key={index}
|
|
||||||
active={isActive}
|
|
||||||
icon={<Icon icon={item.icon} iconSize={item.iconSize} />}
|
|
||||||
text={item.text}
|
|
||||||
disabled={item.disabled}
|
|
||||||
dropdownType={item.dropdownType || 'collapse'}
|
|
||||||
caretIconSize={16}
|
|
||||||
onClick={handleItemClick}
|
|
||||||
callapseActive={!!isActive}
|
|
||||||
itemClassName={classNames({
|
|
||||||
'is-active': isActive,
|
|
||||||
'has-icon': !hasChildren && item.icon,
|
|
||||||
})}
|
|
||||||
hasSubmenu={hasChildren}
|
|
||||||
/>
|
|
||||||
</Choose.Otherwise>
|
|
||||||
</Choose>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const filterItems = menu.filter(
|
|
||||||
(item) => isSubscriptionActive || item.enableBilling,
|
|
||||||
);
|
|
||||||
const items = menuItemsMapper(filterItems);
|
|
||||||
|
|
||||||
const handleSidebarOverlayClose = () => {
|
|
||||||
setIsOpen(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<Menu className="sidebar-menu">{items}</Menu>{' '}
|
|
||||||
<SidebarOverlay
|
|
||||||
isOpen={isOpen}
|
|
||||||
label={currentItem?.text || ''}
|
|
||||||
items={currentItem?.children || []}
|
|
||||||
onClose={handleSidebarOverlayClose}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default compose(
|
|
||||||
withSubscriptions(
|
|
||||||
({ isSubscriptionActive }) => ({ isSubscriptionActive }),
|
|
||||||
'main',
|
|
||||||
),
|
|
||||||
)(SidebarMenu);
|
|
||||||
@@ -1,48 +0,0 @@
|
|||||||
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,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,146 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import { Overlay } from '@blueprintjs/core';
|
|
||||||
import { Link } from 'react-router-dom';
|
|
||||||
import SidebarOverlayContainer from './SidebarOverlayContainer';
|
|
||||||
interface ISidebarOverlayItem {
|
|
||||||
text: string;
|
|
||||||
href: string;
|
|
||||||
divider?: boolean;
|
|
||||||
label?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ISidebarOverlayProps {
|
|
||||||
isOpen: boolean;
|
|
||||||
items: ISidebarOverlayItem[];
|
|
||||||
label: string;
|
|
||||||
onClose: Function;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ISidebarOverlayItemProps {
|
|
||||||
text: string;
|
|
||||||
href: string;
|
|
||||||
onLinkClick: Function;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ISidebarOverlayItemDivider {
|
|
||||||
divider: boolean;
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Sidebar overlay item.
|
|
||||||
*/
|
|
||||||
function SidebarOverlayItem({
|
|
||||||
text,
|
|
||||||
href,
|
|
||||||
onLinkClick,
|
|
||||||
}: ISidebarOverlayItemProps) {
|
|
||||||
const handleLinkClick = () => {
|
|
||||||
onLinkClick && onLinkClick();
|
|
||||||
};
|
|
||||||
return (
|
|
||||||
<div className="sidebar-overlay__item">
|
|
||||||
<Link onClick={handleLinkClick} to={href}>
|
|
||||||
{text}
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ISidebarOverlayItemLabel {
|
|
||||||
text: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
function SidebarOverlayLabel({ text }: ISidebarOverlayItemLabel) {
|
|
||||||
return <div className="sidebar-overlay__label">{text}</div>;
|
|
||||||
}
|
|
||||||
|
|
||||||
function SidebarOverlayDivider() {
|
|
||||||
return <div className={'sidebar-overlay__divider'}></div>;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sidebar overlay component.
|
|
||||||
*/
|
|
||||||
export default function SidebarOverlay({
|
|
||||||
label,
|
|
||||||
isOpen: controllerdIsOpen,
|
|
||||||
onClose,
|
|
||||||
items,
|
|
||||||
}: ISidebarOverlayProps) {
|
|
||||||
const [isEverOpened, setEverOpened] = React.useState(false);
|
|
||||||
const [isOpen, setIsOpen] = React.useState(controllerdIsOpen);
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
if (
|
|
||||||
typeof controllerdIsOpen !== 'undefined' &&
|
|
||||||
isOpen !== controllerdIsOpen
|
|
||||||
) {
|
|
||||||
setIsOpen(controllerdIsOpen);
|
|
||||||
}
|
|
||||||
}, [controllerdIsOpen, setIsOpen, isOpen]);
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
if (isOpen && !isEverOpened) {
|
|
||||||
setEverOpened(true);
|
|
||||||
}
|
|
||||||
}, [isEverOpened, isOpen]);
|
|
||||||
|
|
||||||
if (!isEverOpened) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle overlay close event.
|
|
||||||
const handleOverlayClose = () => {
|
|
||||||
setIsOpen(false);
|
|
||||||
onClose && onClose();
|
|
||||||
};
|
|
||||||
// Handle overlay open event.
|
|
||||||
const handleOverlayOpen = () => {
|
|
||||||
setIsOpen(true);
|
|
||||||
};
|
|
||||||
// Handle sidebar item link click.
|
|
||||||
const handleItemClick = () => {
|
|
||||||
setIsOpen(false);
|
|
||||||
onClose && onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Overlay
|
|
||||||
isOpen={isOpen}
|
|
||||||
portalContainer={
|
|
||||||
(document.querySelector('.Pane.vertical.Pane2') as HTMLElement) ||
|
|
||||||
document.body
|
|
||||||
}
|
|
||||||
onClose={handleOverlayClose}
|
|
||||||
onOpening={handleOverlayOpen}
|
|
||||||
transitionDuration={100}
|
|
||||||
backdropClassName={'sidebar-overlay-backdrop'}
|
|
||||||
>
|
|
||||||
<div className="sidebar-overlay sidebar-overlay-transition">
|
|
||||||
<SidebarOverlayContainer>
|
|
||||||
<div className="sidebar-overlay__menu">
|
|
||||||
{label && (
|
|
||||||
<>
|
|
||||||
<SidebarOverlayLabel text={label} />
|
|
||||||
<SidebarOverlayDivider />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{items.map((item) =>
|
|
||||||
item.divider ? (
|
|
||||||
<SidebarOverlayDivider />
|
|
||||||
) : item.label ? (
|
|
||||||
<SidebarOverlayLabel text={item.text} />
|
|
||||||
) : (
|
|
||||||
<SidebarOverlayItem
|
|
||||||
onLinkClick={handleItemClick}
|
|
||||||
text={item.text}
|
|
||||||
href={item.href}
|
|
||||||
/>
|
|
||||||
),
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</SidebarOverlayContainer>
|
|
||||||
</div>
|
|
||||||
</Overlay>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -101,10 +101,12 @@ export * from './ExchangeRate';
|
|||||||
export * from './Branches';
|
export * from './Branches';
|
||||||
export * from './Warehouses';
|
export * from './Warehouses';
|
||||||
export * from './Currencies';
|
export * from './Currencies';
|
||||||
export * from './FormTopbar'
|
export * from './FormTopbar';
|
||||||
export * from './Paper';
|
export * from './Paper';
|
||||||
export * from './Accounts'
|
export * from './Accounts';
|
||||||
export * from './DataTableCells';
|
export * from './DataTableCells';
|
||||||
|
export * from './FlexGrid';
|
||||||
|
export * from './MenuItem';
|
||||||
|
|
||||||
const Hint = FieldHint;
|
const Hint = FieldHint;
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,14 +1,19 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import SidebarContainer from 'components/Sidebar/SidebarContainer';
|
import { SidebarContainer } from './SidebarContainer';
|
||||||
import SidebarHead from 'components/Sidebar/SidebarHead';
|
import { SidebarHead } from './SidebarHead';
|
||||||
import SidebarMenu from 'components/Sidebar/SidebarMenu';
|
import { SidebarMenu } from './SidebarMenu';
|
||||||
import { useGetSidebarMenu } from './utils';
|
import { useMainSidebarMenu } from './hooks';
|
||||||
|
import { SidebarOverlayBinded } from '../SidebarOverlay';
|
||||||
|
|
||||||
import 'style/containers/Dashboard/Sidebar.scss';
|
import 'style/containers/Dashboard/Sidebar.scss';
|
||||||
|
|
||||||
export default function Sidebar({ dashboardContentRef }) {
|
/**
|
||||||
const menu = useGetSidebarMenu();
|
* Dashboard sidebar.
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
export function Sidebar() {
|
||||||
|
const menu = useMainSidebarMenu();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SidebarContainer>
|
<SidebarContainer>
|
||||||
@@ -17,14 +22,14 @@ export default function Sidebar({ dashboardContentRef }) {
|
|||||||
<div className="sidebar__menu">
|
<div className="sidebar__menu">
|
||||||
<SidebarMenu menu={menu} />
|
<SidebarMenu menu={menu} />
|
||||||
</div>
|
</div>
|
||||||
|
<SidebarOverlayBinded />
|
||||||
<SidebarFooterVersion />
|
<SidebarFooterVersion />
|
||||||
</SidebarContainer>
|
</SidebarContainer>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sidebar footer version.
|
* Sidebar footer version.
|
||||||
* @returns {React.JSX}
|
* @returns {React.JSX}
|
||||||
*/
|
*/
|
||||||
function SidebarFooterVersion() {
|
function SidebarFooterVersion() {
|
||||||
@@ -1,19 +1,21 @@
|
|||||||
import React, { useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
import { Scrollbar } from 'react-scrollbars-custom';
|
import { Scrollbar } from 'react-scrollbars-custom';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import withDashboardActions from 'containers/Dashboard/withDashboardActions';
|
|
||||||
import withDashboard from 'containers/Dashboard/withDashboard';
|
import withDashboard from 'containers/Dashboard/withDashboard';
|
||||||
|
import withSubscriptions from 'containers/Subscriptions/withSubscriptions';
|
||||||
|
|
||||||
|
import { useObserveSidebarExpendedBodyclass } from './hooks';
|
||||||
import { compose } from 'utils';
|
import { compose } from 'utils';
|
||||||
import withSubscriptions from '../../containers/Subscriptions/withSubscriptions';
|
|
||||||
|
|
||||||
function SidebarContainer({
|
/**
|
||||||
|
* Sidebar container/
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarContainerJSX({
|
||||||
// #ownProps
|
// #ownProps
|
||||||
children,
|
children,
|
||||||
|
|
||||||
// #withDashboardActions
|
|
||||||
toggleSidebarExpend,
|
|
||||||
|
|
||||||
// #withDashboard
|
// #withDashboard
|
||||||
sidebarExpended,
|
sidebarExpended,
|
||||||
|
|
||||||
@@ -22,9 +24,10 @@ function SidebarContainer({
|
|||||||
}) {
|
}) {
|
||||||
const sidebarScrollerRef = React.useRef();
|
const sidebarScrollerRef = React.useRef();
|
||||||
|
|
||||||
useEffect(() => {
|
// Toggles classname to body once sidebar expend/shrink.
|
||||||
document.body.classList.toggle('has-mini-sidebar', !sidebarExpended);
|
useObserveSidebarExpendedBodyclass(sidebarExpended);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
if (!sidebarExpended && sidebarScrollerRef.current) {
|
if (!sidebarExpended && sidebarScrollerRef.current) {
|
||||||
sidebarScrollerRef.current.scrollTo({
|
sidebarScrollerRef.current.scrollTo({
|
||||||
top: 0,
|
top: 0,
|
||||||
@@ -39,9 +42,9 @@ function SidebarContainer({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const scrollerElementRef = (ref) => {
|
const scrollerElementRef = React.useCallback((ref) => {
|
||||||
sidebarScrollerRef.current = ref;
|
sidebarScrollerRef.current = ref;
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@@ -64,8 +67,7 @@ function SidebarContainer({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default compose(
|
export const SidebarContainer = compose(
|
||||||
withDashboardActions,
|
|
||||||
withDashboard(({ sidebarExpended }) => ({
|
withDashboard(({ sidebarExpended }) => ({
|
||||||
sidebarExpended,
|
sidebarExpended,
|
||||||
})),
|
})),
|
||||||
@@ -73,4 +75,4 @@ export default compose(
|
|||||||
({ isSubscriptionActive }) => ({ isSubscriptionActive }),
|
({ isSubscriptionActive }) => ({ isSubscriptionActive }),
|
||||||
'main',
|
'main',
|
||||||
),
|
),
|
||||||
)(SidebarContainer);
|
)(SidebarContainerJSX);
|
||||||
@@ -1,9 +1,11 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Button, Popover, Menu, Position } from '@blueprintjs/core';
|
import { Button, Popover, Menu, Position } from '@blueprintjs/core';
|
||||||
import Icon from 'components/Icon';
|
|
||||||
|
import { Icon } from 'components';
|
||||||
|
|
||||||
|
import withCurrentOrganization from 'containers/Organization/withCurrentOrganization';
|
||||||
|
import { useAuthenticatedAccount } from 'hooks/query';
|
||||||
import { compose, firstLettersArgs } from 'utils';
|
import { compose, firstLettersArgs } from 'utils';
|
||||||
import withCurrentOrganization from '../../containers/Organization/withCurrentOrganization';
|
|
||||||
import { useAuthenticatedAccount } from '../../hooks/query';
|
|
||||||
|
|
||||||
// Popover modifiers.
|
// Popover modifiers.
|
||||||
const POPOVER_MODIFIERS = {
|
const POPOVER_MODIFIERS = {
|
||||||
@@ -13,7 +15,7 @@ const POPOVER_MODIFIERS = {
|
|||||||
/**
|
/**
|
||||||
* Sideabr head.
|
* Sideabr head.
|
||||||
*/
|
*/
|
||||||
function SidebarHead({
|
function SidebarHeadJSX({
|
||||||
// #withCurrentOrganization
|
// #withCurrentOrganization
|
||||||
organization,
|
organization,
|
||||||
}) {
|
}) {
|
||||||
@@ -61,6 +63,6 @@ function SidebarHead({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default compose(
|
export const SidebarHead = compose(
|
||||||
withCurrentOrganization(({ organization }) => ({ organization })),
|
withCurrentOrganization(({ organization }) => ({ organization })),
|
||||||
)(SidebarHead);
|
)(SidebarHeadJSX);
|
||||||
73
src/containers/Dashboard/Sidebar/SidebarMenu.js
Normal file
73
src/containers/Dashboard/Sidebar/SidebarMenu.js
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Menu } from '@blueprintjs/core';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
|
import { MenuItem, MenuItemLabel } from 'components';
|
||||||
|
import { ISidebarMenuItemType } from 'containers/Dashboard/Sidebar/interfaces';
|
||||||
|
import { useIsSidebarMenuItemActive } from './hooks';
|
||||||
|
|
||||||
|
import withSubscriptions from 'containers/Subscriptions/withSubscriptions';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar menu item.
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarMenuItem({ item, index }) {
|
||||||
|
// Detarmine whether the item is active.
|
||||||
|
const isActive = useIsSidebarMenuItemActive(item);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MenuItem
|
||||||
|
key={index}
|
||||||
|
text={item.text}
|
||||||
|
disabled={item.disabled}
|
||||||
|
dropdownType={item.dropdownType || 'collapse'}
|
||||||
|
caretIconSize={16}
|
||||||
|
onClick={item.onClick}
|
||||||
|
active={isActive}
|
||||||
|
hasSubmenu={item.hasChildren}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
SidebarMenuItem.ItemTypes = [
|
||||||
|
ISidebarMenuItemType.Link,
|
||||||
|
ISidebarMenuItemType.Overlay,
|
||||||
|
ISidebarMenuItemType.Dialog,
|
||||||
|
];
|
||||||
|
/**
|
||||||
|
* Detarmines which sidebar menu item type should display.
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarMenuItemComposer({ item, index }) {
|
||||||
|
// Link item type.
|
||||||
|
return SidebarMenuItem.ItemTypes.indexOf(item.type) !== -1 ? (
|
||||||
|
<SidebarMenuItem item={item} index={index} />
|
||||||
|
) : // Group item type.
|
||||||
|
item.type === ISidebarMenuItemType.Group ? (
|
||||||
|
<MenuItemLabel text={item.text} />
|
||||||
|
) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar menu.
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarMenuJSX({ menu }) {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<Menu className="sidebar-menu">
|
||||||
|
{menu.map((item, index) => (
|
||||||
|
<SidebarMenuItemComposer index={index} item={item} />
|
||||||
|
))}
|
||||||
|
</Menu>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SidebarMenu = R.compose(
|
||||||
|
withSubscriptions(
|
||||||
|
({ isSubscriptionActive }) => ({ isSubscriptionActive }),
|
||||||
|
'main',
|
||||||
|
),
|
||||||
|
)(SidebarMenuJSX);
|
||||||
349
src/containers/Dashboard/Sidebar/hooks.js
Normal file
349
src/containers/Dashboard/Sidebar/hooks.js
Normal file
@@ -0,0 +1,349 @@
|
|||||||
|
import _, { isEmpty, includes } from 'lodash';
|
||||||
|
import React from 'react';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
|
import { useAbilityContext } from 'hooks';
|
||||||
|
import { useSidebarSubmenu, useFeatureCan } from 'hooks/state';
|
||||||
|
import { SidebarMenu } from 'config/sidebarMenu';
|
||||||
|
import {
|
||||||
|
ISidebarMenuItemType,
|
||||||
|
ISidebarSubscriptionAbility,
|
||||||
|
} from './interfaces';
|
||||||
|
import {
|
||||||
|
useSidebarSubmnuActions,
|
||||||
|
useDialogActions,
|
||||||
|
useSubscription,
|
||||||
|
} from 'hooks/state';
|
||||||
|
import { filterValuesDeep, deepdash } from 'utils';
|
||||||
|
|
||||||
|
const deepDashConfig = {
|
||||||
|
childrenPath: 'children',
|
||||||
|
pathFormat: 'array',
|
||||||
|
};
|
||||||
|
const ingoreTypesEmpty = [
|
||||||
|
ISidebarMenuItemType.Group,
|
||||||
|
ISidebarMenuItemType.Overlay,
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes the all overlay items from the menu to the main-sidebar.
|
||||||
|
* @param {ISidebarMenuItem[]} menu
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
function removeSidebarOverlayChildren(menu) {
|
||||||
|
return deepdash.mapValuesDeep(
|
||||||
|
menu,
|
||||||
|
(item, key, parent, context) => {
|
||||||
|
if (item.type === ISidebarMenuItemType.Overlay) {
|
||||||
|
context.skipChildren(true);
|
||||||
|
return _.omit(item, ['children']);
|
||||||
|
}
|
||||||
|
return item;
|
||||||
|
},
|
||||||
|
deepDashConfig,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrives the main sidebar pre-menu.
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
export function getMainSidebarMenu() {
|
||||||
|
return R.compose(removeSidebarOverlayChildren)(SidebarMenu);
|
||||||
|
}
|
||||||
|
|
||||||
|
function useFilterSidebarItemFeaturePredicater() {
|
||||||
|
const { featureCan } = useFeatureCan();
|
||||||
|
|
||||||
|
return {
|
||||||
|
predicate: (item) => {
|
||||||
|
if (item.feature && !featureCan(item.feature)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function useFilterSidebarItemAbilityPredicater() {
|
||||||
|
const ability = useAbilityContext();
|
||||||
|
|
||||||
|
return {
|
||||||
|
predicate: (item) => {
|
||||||
|
if (
|
||||||
|
item.permission &&
|
||||||
|
!ability.can(item.permission.ability, item.permission.subject)
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filters the sidebar item based on the subscription state.
|
||||||
|
*/
|
||||||
|
function useFilterSidebarItemSubscriptionPredicater() {
|
||||||
|
const { isSubscriptionActive, isSubscriptionInactive } = useSubscription();
|
||||||
|
|
||||||
|
return {
|
||||||
|
predicate: (item) => {
|
||||||
|
const { subscription } = item;
|
||||||
|
|
||||||
|
if (subscription) {
|
||||||
|
const isActive = includes(subscription, [
|
||||||
|
ISidebarSubscriptionAbility.Active,
|
||||||
|
])
|
||||||
|
? isSubscriptionActive
|
||||||
|
: true;
|
||||||
|
|
||||||
|
const isInactive = includes(subscription, [
|
||||||
|
ISidebarSubscriptionAbility.Inactive,
|
||||||
|
])
|
||||||
|
? isSubscriptionInactive
|
||||||
|
: true;
|
||||||
|
|
||||||
|
return isActive && isInactive;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filters sidebar menu items based on ability of the item permission.
|
||||||
|
* @param {*} menu
|
||||||
|
* @returns {}
|
||||||
|
*/
|
||||||
|
function useFilterSidebarMenuAbility(menu) {
|
||||||
|
const { predicate: predFeature } = useFilterSidebarItemFeaturePredicater();
|
||||||
|
const { predicate: predAbility } = useFilterSidebarItemAbilityPredicater();
|
||||||
|
const { predicate: predSubscription } =
|
||||||
|
useFilterSidebarItemSubscriptionPredicater();
|
||||||
|
|
||||||
|
return deepdash.filterDeep(
|
||||||
|
menu,
|
||||||
|
(item) => {
|
||||||
|
return predFeature(item) && predAbility(item) && predSubscription(item);
|
||||||
|
},
|
||||||
|
deepDashConfig,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flats the sidebar menu groups.
|
||||||
|
* @param {*} menu
|
||||||
|
* @returns {}
|
||||||
|
*/
|
||||||
|
function useFlatSidebarMenu(menu) {
|
||||||
|
return React.useMemo(() => {
|
||||||
|
return deepdash.mapDeep(menu, (item) => item, deepDashConfig);
|
||||||
|
}, [menu]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Binds sidebar link item click action.
|
||||||
|
* @param {ISidebarMenuItem} item
|
||||||
|
*/
|
||||||
|
function useBindSidebarItemLinkClick() {
|
||||||
|
const history = useHistory();
|
||||||
|
const { closeSidebarSubmenu } = useSidebarSubmnuActions();
|
||||||
|
|
||||||
|
// Handle sidebar item click.
|
||||||
|
const onClick = (item) => (event) => {
|
||||||
|
closeSidebarSubmenu();
|
||||||
|
history.push(item.href);
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
bindOnClick: (item) => {
|
||||||
|
return {
|
||||||
|
...item,
|
||||||
|
onClick: onClick(item),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bind sidebar dialog item click action.
|
||||||
|
* @param {ISidebarMenuItem} item
|
||||||
|
*/
|
||||||
|
function useBindSidebarItemDialogClick() {
|
||||||
|
const { closeSidebarSubmenu } = useSidebarSubmnuActions();
|
||||||
|
const { openDialog } = useDialogActions();
|
||||||
|
|
||||||
|
// Handle sidebar item click.
|
||||||
|
const onClick = (item) => (event) => {
|
||||||
|
closeSidebarSubmenu();
|
||||||
|
openDialog(item.dialogName, item.dialogPayload);
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
bindOnClick: (item) => {
|
||||||
|
return {
|
||||||
|
...item,
|
||||||
|
onClick: onClick(item),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function useBindSidebarItemOverlayClick() {
|
||||||
|
const { toggleSidebarSubmenu, closeSidebarSubmenu } =
|
||||||
|
useSidebarSubmnuActions();
|
||||||
|
|
||||||
|
// Handle sidebar item click.
|
||||||
|
const onClick = (item) => (event) => {
|
||||||
|
closeSidebarSubmenu();
|
||||||
|
toggleSidebarSubmenu({ submenuId: item.overlayId });
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
bindOnClick: (item) => {
|
||||||
|
return {
|
||||||
|
...item,
|
||||||
|
onClick: onClick(item),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {} menu
|
||||||
|
* @returns {}
|
||||||
|
*/
|
||||||
|
function useBindSidebarItemClick(menu) {
|
||||||
|
const { bindOnClick: bindLinkClickEvt } = useBindSidebarItemLinkClick();
|
||||||
|
const { bindOnClick: bindOverlayClickEvt } = useBindSidebarItemOverlayClick();
|
||||||
|
const { bindOnClick: bindItemDialogEvt } = useBindSidebarItemDialogClick();
|
||||||
|
|
||||||
|
return React.useMemo(() => {
|
||||||
|
return deepdash.mapValuesDeep(
|
||||||
|
menu,
|
||||||
|
(item) => {
|
||||||
|
return R.compose(
|
||||||
|
R.when(
|
||||||
|
R.propSatisfies(R.equals(ISidebarMenuItemType.Link), 'type'),
|
||||||
|
bindLinkClickEvt,
|
||||||
|
),
|
||||||
|
R.when(
|
||||||
|
R.propSatisfies(R.equals(ISidebarMenuItemType.Overlay), 'type'),
|
||||||
|
bindOverlayClickEvt,
|
||||||
|
),
|
||||||
|
R.when(
|
||||||
|
R.propSatisfies(R.equals(ISidebarMenuItemType.Dialog), 'type'),
|
||||||
|
bindItemDialogEvt,
|
||||||
|
),
|
||||||
|
)(item);
|
||||||
|
},
|
||||||
|
deepDashConfig,
|
||||||
|
);
|
||||||
|
}, [menu]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Finds the given overlay submenu id from the menu graph.
|
||||||
|
* @param {ISidebarMenuOverlayIds}
|
||||||
|
* @param {ISidebarMenuItem[]} menu -
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
const findSubmenuBySubmenuId = R.curry((submenuId, menu) => {
|
||||||
|
const groupItem = deepdash.findDeep(
|
||||||
|
menu,
|
||||||
|
(item) => {
|
||||||
|
return (
|
||||||
|
item.type === ISidebarMenuItemType.Overlay &&
|
||||||
|
item.overlayId === submenuId
|
||||||
|
);
|
||||||
|
},
|
||||||
|
deepDashConfig,
|
||||||
|
);
|
||||||
|
return groupItem?.value?.children || [];
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the main sidebar post-menu.
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
export function useMainSidebarMenu() {
|
||||||
|
return R.compose(
|
||||||
|
useBindSidebarItemClick,
|
||||||
|
useFlatSidebarMenu,
|
||||||
|
removeSidebarOverlayChildren,
|
||||||
|
useAssocSidebarItemHasChildren,
|
||||||
|
filterSidebarItemHasNoChildren,
|
||||||
|
useFilterSidebarMenuAbility,
|
||||||
|
)(SidebarMenu);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Assoc `hasChildren` prop to sidebar menu items.
|
||||||
|
* @param {ISidebarMenuItem[]} items
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
function useAssocSidebarItemHasChildren(items) {
|
||||||
|
return deepdash.mapValuesDeep(
|
||||||
|
items,
|
||||||
|
(item) => {
|
||||||
|
return {
|
||||||
|
...item,
|
||||||
|
hasChildren: !isEmpty(item.children),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
deepDashConfig,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sub-sidebar post-menu.
|
||||||
|
* @param {ISidebarMenuOverlayIds} submenuId
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
export function useSubSidebarMenu(submenuId) {
|
||||||
|
if (!submenuId) return [];
|
||||||
|
|
||||||
|
return R.compose(
|
||||||
|
useBindSidebarItemClick,
|
||||||
|
useFlatSidebarMenu,
|
||||||
|
filterSidebarItemHasNoChildren,
|
||||||
|
useFilterSidebarMenuAbility,
|
||||||
|
findSubmenuBySubmenuId(submenuId),
|
||||||
|
)(SidebarMenu);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Observes the sidebar expending with body class.
|
||||||
|
* @param {boolean} sidebarExpended
|
||||||
|
*/
|
||||||
|
export function useObserveSidebarExpendedBodyclass(sidebarExpended) {
|
||||||
|
React.useEffect(() => {
|
||||||
|
document.body.classList.toggle('has-mini-sidebar', !sidebarExpended);
|
||||||
|
}, [sidebarExpended]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detamrines whether the given sidebar menu item is active.
|
||||||
|
* @returns {boolean}
|
||||||
|
*/
|
||||||
|
export function useIsSidebarMenuItemActive(item) {
|
||||||
|
const { submenuId } = useSidebarSubmenu();
|
||||||
|
return (
|
||||||
|
item.type === ISidebarMenuItemType.Overlay && submenuId === item.overlayId
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filter sidebar specific items types that have no types.
|
||||||
|
* @param {ISidebarMenuItem[]} items -
|
||||||
|
* @returns {ISidebarMenuItem[]}
|
||||||
|
*/
|
||||||
|
export function filterSidebarItemHasNoChildren(items) {
|
||||||
|
return filterValuesDeep((item) => {
|
||||||
|
// If it was group item and has no children items so discard that item.
|
||||||
|
if (ingoreTypesEmpty.indexOf(item.type) !== -1 && isEmpty(item.children)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}, items);
|
||||||
|
}
|
||||||
@@ -2,7 +2,9 @@ export enum ISidebarMenuItemType {
|
|||||||
Label = 'label',
|
Label = 'label',
|
||||||
Link = 'link',
|
Link = 'link',
|
||||||
Group = 'group',
|
Group = 'group',
|
||||||
Overlay = 'overlay'
|
Overlay = 'overlay',
|
||||||
|
Dialog = 'dialog',
|
||||||
|
Drawer = 'drawer',
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ISidebarMenuItemOverlay extends ISidebarMenuItemCommon {
|
export interface ISidebarMenuItemOverlay extends ISidebarMenuItemCommon {
|
||||||
@@ -16,6 +18,18 @@ export interface ISidebarMenuItemLink extends ISidebarMenuItemCommon {
|
|||||||
matchExact?: boolean;
|
matchExact?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ISidebarMenuItemDialog extends ISidebarMenuItemCommon {
|
||||||
|
type: ISidebarMenuItemType.Dialog;
|
||||||
|
dialogName: string;
|
||||||
|
dialogPayload: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ISidebarMenuItemDrawer extends ISidebarMenuItemCommon {
|
||||||
|
type: ISidebarMenuItemType.Drawer;
|
||||||
|
drawerName: string;
|
||||||
|
drawerPayload: any;
|
||||||
|
}
|
||||||
|
|
||||||
export interface ISidebarMenuItemLabel extends ISidebarMenuItemCommon {
|
export interface ISidebarMenuItemLabel extends ISidebarMenuItemCommon {
|
||||||
text?: string;
|
text?: string;
|
||||||
type: ISidebarMenuItemType.Label;
|
type: ISidebarMenuItemType.Label;
|
||||||
@@ -42,4 +56,22 @@ export type ISidebarMenuItem =
|
|||||||
| ISidebarMenuItemLink
|
| ISidebarMenuItemLink
|
||||||
| ISidebarMenuItemLabel
|
| ISidebarMenuItemLabel
|
||||||
| ISidebarMenuItemGroup
|
| ISidebarMenuItemGroup
|
||||||
| ISidebarMenuItemOverlay;
|
| ISidebarMenuItemOverlay
|
||||||
|
| ISidebarMenuItemDialog
|
||||||
|
| ISidebarMenuItemDrawer;
|
||||||
|
|
||||||
|
export enum ISidebarMenuOverlayIds {
|
||||||
|
Items = 'Items',
|
||||||
|
Reports = 'Reports',
|
||||||
|
Sales = 'Sales',
|
||||||
|
Purchases = 'Purchases',
|
||||||
|
Financial = 'Financial',
|
||||||
|
Contacts = 'Contacts',
|
||||||
|
Cashflow = 'Cashflow',
|
||||||
|
Expenses = 'Expenses',
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum ISidebarSubscriptionAbility {
|
||||||
|
Expired = 'SubscriptionExpired',
|
||||||
|
Active = 'SubscriptionActive',
|
||||||
|
}
|
||||||
13
src/containers/Dashboard/Sidebar/withDashboardSidebar.js
Normal file
13
src/containers/Dashboard/Sidebar/withDashboardSidebar.js
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
|
export default (mapState) => {
|
||||||
|
const mapStateToProps = (state, props) => {
|
||||||
|
const mapped = {
|
||||||
|
sidebarSubmenuOpen: state.dashboard.sidebarSubmenu.isOpen,
|
||||||
|
sidebarSubmenuId: state.dashboard.sidebarSubmenu.submenuId,
|
||||||
|
};
|
||||||
|
return mapState ? mapState(mapped, state, props) : mapped;
|
||||||
|
};
|
||||||
|
return connect(mapStateToProps);
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
import { connect } from 'react-redux';
|
||||||
|
import {
|
||||||
|
closeSidebarSubmenu,
|
||||||
|
openSidebarSubmenu,
|
||||||
|
} from 'store/dashboard/dashboard.actions';
|
||||||
|
|
||||||
|
const mapActionsToProps = (dispatch) => ({
|
||||||
|
// Opens the dashboard submenu sidebar.
|
||||||
|
openDashboardSidebarSubmenu: (submenuId) =>
|
||||||
|
dispatch(openSidebarSubmenu(submenuId)),
|
||||||
|
|
||||||
|
// Closes the dashboard submenu sidebar.
|
||||||
|
closeDashboardSidebarSubmenu: () => dispatch(closeSidebarSubmenu()),
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(null, mapActionsToProps);
|
||||||
142
src/containers/Dashboard/SidebarOverlay/SidebarOverlay.tsx
Normal file
142
src/containers/Dashboard/SidebarOverlay/SidebarOverlay.tsx
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
//@ts-nocheck
|
||||||
|
import React from 'react';
|
||||||
|
import { Overlay, OverlayProps } from '@blueprintjs/core';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
import { SidebarOverlayContainer } from './SidebarOverlayContainer';
|
||||||
|
import { ISidebarMenuItem, ISidebarMenuItemType } from '../Sidebar/interfaces';
|
||||||
|
|
||||||
|
export interface ISidebarOverlayItem {
|
||||||
|
text: string;
|
||||||
|
href: string;
|
||||||
|
divider?: boolean;
|
||||||
|
label?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ISidebarOverlayProps {
|
||||||
|
items: ISidebarMenuItem[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ISidebarOverlayItemProps {
|
||||||
|
text: string | JSX.Element;
|
||||||
|
href?: string;
|
||||||
|
onClick?: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ISidebarOverlayItemDivider {
|
||||||
|
divider: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay item.
|
||||||
|
* @param {ISidebarOverlayItemProps}
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
export function SidebarOverlayItemLink({
|
||||||
|
text,
|
||||||
|
href,
|
||||||
|
onClick,
|
||||||
|
}: ISidebarOverlayItemProps) {
|
||||||
|
return (
|
||||||
|
<div className="sidebar-overlay__item">
|
||||||
|
<Link to={href} onClick={onClick}>
|
||||||
|
{text}
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ISidebarOverlayItemLabel {
|
||||||
|
text: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay label item.
|
||||||
|
* @param {ISidebarOverlayItemLabel}
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
export function SidebarOverlayLabel({
|
||||||
|
text,
|
||||||
|
}: ISidebarOverlayItemLabel): JSX.Element {
|
||||||
|
return <div className="sidebar-overlay__label">{text}</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay divider item.
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
export function SidebarOverlayDivider() {
|
||||||
|
return <div className={'sidebar-overlay__divider'}></div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SidebarOverlayItemProps {
|
||||||
|
item: ISidebarMenuItem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay item.
|
||||||
|
* @param {SidebarOverlayItemProps} props -
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarOverlayItem({ item }: SidebarOverlayItemProps) {
|
||||||
|
//
|
||||||
|
return item.type === ISidebarMenuItemType.Group ? (
|
||||||
|
<SidebarOverlayLabel text={item.text} />
|
||||||
|
) : //
|
||||||
|
item.type === ISidebarMenuItemType.Link ||
|
||||||
|
item.type === ISidebarMenuItemType.Dialog ? (
|
||||||
|
<SidebarOverlayItemLink text={item.text} onClick={item.onClick} />
|
||||||
|
) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
export interface ISidebarOverlayMenu {
|
||||||
|
items: ISidebarMenuItem[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay menu.
|
||||||
|
* @param {ISidebarOverlayMenu}
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarOverlayMenu({ items }: ISidebarOverlayMenu) {
|
||||||
|
return (
|
||||||
|
<div className="sidebar-overlay__menu">
|
||||||
|
{items.map((item) => (
|
||||||
|
<SidebarOverlayItem item={item} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SidebarOverlayProps extends OverlayProps {
|
||||||
|
items: ISidebarMenuItem[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay component.
|
||||||
|
* @param {SidebarOverlayProps}
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
export function SidebarOverlay({ items, label, ...rest }: SidebarOverlayProps) {
|
||||||
|
return (
|
||||||
|
<Overlay
|
||||||
|
portalContainer={
|
||||||
|
(document.querySelector('.Pane.vertical.Pane2') as HTMLElement) ||
|
||||||
|
document.body
|
||||||
|
}
|
||||||
|
transitionDuration={100}
|
||||||
|
backdropClassName={'sidebar-overlay-backdrop'}
|
||||||
|
{...rest}
|
||||||
|
>
|
||||||
|
<div className="sidebar-overlay sidebar-overlay-transition">
|
||||||
|
<SidebarOverlayContainer>
|
||||||
|
{label && <SidebarOverlayLabel text={label} />}
|
||||||
|
|
||||||
|
<SidebarOverlayMenu items={items} />
|
||||||
|
</SidebarOverlayContainer>
|
||||||
|
</div>
|
||||||
|
</Overlay>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import React from 'react';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
|
||||||
|
import { SidebarOverlay } from './SidebarOverlay';
|
||||||
|
import withDashboardSidebarActions from 'containers/Dashboard/Sidebar/withDashboardSidebarActions';
|
||||||
|
import withDashboardSidebar from 'containers/Dashboard/Sidebar/withDashboardSidebar';
|
||||||
|
|
||||||
|
import { useSubSidebarMenu } from '../Sidebar/hooks';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard sidebar menu.
|
||||||
|
* @returns {JSX.Element}
|
||||||
|
*/
|
||||||
|
function SidebarOverlayBindedRoot({
|
||||||
|
// #withDashboardSidebar
|
||||||
|
sidebarSubmenuOpen,
|
||||||
|
sidebarSubmenuId,
|
||||||
|
|
||||||
|
// #withDashboardSidebarActions
|
||||||
|
closeDashboardSidebarSubmenu,
|
||||||
|
}) {
|
||||||
|
const handleSidebarClosing = React.useCallback(() => {
|
||||||
|
closeDashboardSidebarSubmenu();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SidebarOverlayBindedRouter
|
||||||
|
sidebarSubmenuId={sidebarSubmenuId}
|
||||||
|
isOpen={sidebarSubmenuOpen}
|
||||||
|
onClose={handleSidebarClosing}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard sidebar submenu router.
|
||||||
|
*/
|
||||||
|
function SidebarOverlayBindedRouter({ sidebarSubmenuId, ...rest }) {
|
||||||
|
const sidebarItems = useSubSidebarMenu(sidebarSubmenuId);
|
||||||
|
|
||||||
|
return <SidebarOverlay items={sidebarItems} {...rest} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar overlay binded with redux.
|
||||||
|
*/
|
||||||
|
export const SidebarOverlayBinded = R.compose(
|
||||||
|
withDashboardSidebar(({ sidebarSubmenuOpen, sidebarSubmenuId }) => ({
|
||||||
|
sidebarSubmenuOpen,
|
||||||
|
sidebarSubmenuId,
|
||||||
|
})),
|
||||||
|
withDashboardSidebarActions,
|
||||||
|
)(SidebarOverlayBindedRoot);
|
||||||
@@ -1,14 +1,16 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Scrollbar } from 'react-scrollbars-custom';
|
import { Scrollbar } from 'react-scrollbars-custom';
|
||||||
|
|
||||||
interface ISidebarOverlayContainerProps {
|
export interface ISidebarOverlayContainerProps {
|
||||||
children: JSX.Element | JSX.Element[],
|
children: JSX.Element | JSX.Element[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sidebar overlay container.
|
* Sidebar overlay container.
|
||||||
*/
|
*/
|
||||||
export default function SidebarOverlayContainer({ children }: ISidebarOverlayContainerProps) {
|
export function SidebarOverlayContainer({
|
||||||
|
children,
|
||||||
|
}: ISidebarOverlayContainerProps) {
|
||||||
return (
|
return (
|
||||||
<div className={'sidebar-overlay__scroll-wrapper'}>
|
<div className={'sidebar-overlay__scroll-wrapper'}>
|
||||||
<Scrollbar noDefaultStyles={true}>
|
<Scrollbar noDefaultStyles={true}>
|
||||||
3
src/containers/Dashboard/SidebarOverlay/index.ts
Normal file
3
src/containers/Dashboard/SidebarOverlay/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from './SidebarOverlay';
|
||||||
|
export * from './SidebarOverlayContainer';
|
||||||
|
export * from './SidebarOverlayBinded'
|
||||||
@@ -4,7 +4,7 @@ import { FastField, Field } from 'formik';
|
|||||||
import { FormGroup, Checkbox } from '@blueprintjs/core';
|
import { FormGroup, Checkbox } from '@blueprintjs/core';
|
||||||
import styled from 'styled-components';
|
import styled from 'styled-components';
|
||||||
|
|
||||||
import { Row, Col, FieldHint, FormattedMessage as T } from 'components';
|
import { Flex, FlexItem, FieldHint, FormattedMessage as T } from 'components';
|
||||||
import {
|
import {
|
||||||
handlePreviousYearCheckBoxChange,
|
handlePreviousYearCheckBoxChange,
|
||||||
handlePreviousYearChangeCheckboxChange,
|
handlePreviousYearChangeCheckboxChange,
|
||||||
@@ -15,11 +15,11 @@ import {
|
|||||||
} from './utils';
|
} from './utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Balance sheet header - Comparison panal.
|
* Balance sheet header - Comparison panal - Comparisons fields.
|
||||||
*/
|
*/
|
||||||
export default function BalanceSheetHeaderComparisonPanal() {
|
function BalanceSheetHeaderComparisonPanalFields() {
|
||||||
return (
|
return (
|
||||||
<BalanceSheetComparisonWrap>
|
<>
|
||||||
{/**----------- Previous Year -----------*/}
|
{/**----------- Previous Year -----------*/}
|
||||||
<Field name={'previousYear'} type={'checkbox'}>
|
<Field name={'previousYear'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
@@ -33,8 +33,9 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</Field>
|
</Field>
|
||||||
<Row>
|
|
||||||
<Col xs={3}>
|
<FlexSubFields align={'left'}>
|
||||||
|
<FlexItem col={6}>
|
||||||
<Field name={'previousYearAmountChange'} type={'checkbox'}>
|
<Field name={'previousYearAmountChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -48,8 +49,9 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</Field>
|
</Field>
|
||||||
</Col>
|
</FlexItem>
|
||||||
<Col xs={3}>
|
|
||||||
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousYearPercentageChange'} type={'checkbox'}>
|
<FastField name={'previousYearPercentageChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -62,8 +64,9 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
</Row>
|
</FlexSubFields>
|
||||||
|
|
||||||
{/*------------ Previous Period -----------*/}
|
{/*------------ Previous Period -----------*/}
|
||||||
<FastField name={'previousPeriod'} type={'checkbox'}>
|
<FastField name={'previousPeriod'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
@@ -78,8 +81,9 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
<Row>
|
|
||||||
<Col xs={3}>
|
<FlexSubFields>
|
||||||
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousPeriodAmountChange'} type={'checkbox'}>
|
<FastField name={'previousPeriodAmountChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -93,8 +97,9 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
<Col xs={3}>
|
|
||||||
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousPeriodPercentageChange'} type={'checkbox'}>
|
<FastField name={'previousPeriodPercentageChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -107,8 +112,8 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
</Row>
|
</FlexSubFields>
|
||||||
|
|
||||||
{/**----------- % of Column -----------*/}
|
{/**----------- % of Column -----------*/}
|
||||||
<FastField name={'percentageOfColumn'} type={'checkbox'}>
|
<FastField name={'percentageOfColumn'} type={'checkbox'}>
|
||||||
@@ -137,19 +142,33 @@ export default function BalanceSheetHeaderComparisonPanal() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Balance sheet header - Comparison panal.
|
||||||
|
*/
|
||||||
|
export default function BalanceSheetHeaderComparisonPanal() {
|
||||||
|
return (
|
||||||
|
<BalanceSheetComparisonWrap>
|
||||||
|
<BalanceSheetComparisonFieldsWrap>
|
||||||
|
<BalanceSheetHeaderComparisonPanalFields />
|
||||||
|
</BalanceSheetComparisonFieldsWrap>
|
||||||
</BalanceSheetComparisonWrap>
|
</BalanceSheetComparisonWrap>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const BalanceSheetComparisonWrap = styled.div`
|
const BalanceSheetComparisonWrap = styled.div`
|
||||||
.row {
|
|
||||||
margin-left: 0.15rem;
|
|
||||||
.col {
|
|
||||||
min-width: 150px !important;
|
|
||||||
max-width: 190px !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.bp3-form-group {
|
.bp3-form-group {
|
||||||
margin-bottom: 3px;
|
margin-bottom: 3px;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
const FlexSubFields = styled(Flex)`
|
||||||
|
padding-left: 20px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const BalanceSheetComparisonFieldsWrap = styled.div`
|
||||||
|
width: 400px;
|
||||||
|
`;
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { FastField, Field } from 'formik';
|
import { FastField } from 'formik';
|
||||||
import { FormGroup, Checkbox } from '@blueprintjs/core';
|
import { FormGroup, Checkbox } from '@blueprintjs/core';
|
||||||
import styled from 'styled-components';
|
import styled from 'styled-components';
|
||||||
|
|
||||||
import { FormattedMessage as T } from 'components';
|
import { Flex, FlexItem, FormattedMessage as T } from 'components';
|
||||||
|
|
||||||
import { Row, Col, FieldHint } from '../../../components';
|
import { FieldHint } from '../../../components';
|
||||||
import {
|
import {
|
||||||
handlePreviousYearCheckBoxChange,
|
handlePreviousYearCheckBoxChange,
|
||||||
handlePreviousPeriodCheckBoxChange,
|
handlePreviousPeriodCheckBoxChange,
|
||||||
@@ -16,11 +16,12 @@ import {
|
|||||||
} from './utils';
|
} from './utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ProfitLoss sheet header -comparison panel.
|
* Profit/loss comparisons panel fields.
|
||||||
|
* @returns {JSX.Element}
|
||||||
*/
|
*/
|
||||||
export default function ProfitLossSheetHeaderComparisonPanel() {
|
function ProfitLossComaprsionPanelFields() {
|
||||||
return (
|
return (
|
||||||
<ProfitLossSheetComparisonWrap>
|
<>
|
||||||
{/**----------- Previous Year -----------*/}
|
{/**----------- Previous Year -----------*/}
|
||||||
<FastField name={'previousYear'} type={'checkbox'}>
|
<FastField name={'previousYear'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
@@ -35,8 +36,9 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
<Row>
|
|
||||||
<Col xs={3}>
|
<FlexSubFields>
|
||||||
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousYearAmountChange'} type={'checkbox'}>
|
<FastField name={'previousYearAmountChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -50,8 +52,8 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
<Col xs={3}>
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousYearPercentageChange'} type={'checkbox'}>
|
<FastField name={'previousYearPercentageChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -65,8 +67,9 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
</Row>
|
</FlexSubFields>
|
||||||
|
|
||||||
{/**----------- Previous Period (PP) -----------*/}
|
{/**----------- Previous Period (PP) -----------*/}
|
||||||
<FastField name={'previousPeriod'} type={'checkbox'}>
|
<FastField name={'previousPeriod'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
@@ -81,8 +84,9 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
<Row>
|
|
||||||
<Col xs={3}>
|
<FlexSubFields>
|
||||||
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousPeriodAmountChange'} type={'checkbox'}>
|
<FastField name={'previousPeriodAmountChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -96,8 +100,8 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
<Col xs={3}>
|
<FlexItem col={6}>
|
||||||
<FastField name={'previousPeriodPercentageChange'} type={'checkbox'}>
|
<FastField name={'previousPeriodPercentageChange'} type={'checkbox'}>
|
||||||
{({ form, field }) => (
|
{({ form, field }) => (
|
||||||
<FormGroup labelInfo={<FieldHint />}>
|
<FormGroup labelInfo={<FieldHint />}>
|
||||||
@@ -111,8 +115,9 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
</Col>
|
</FlexItem>
|
||||||
</Row>
|
</FlexSubFields>
|
||||||
|
|
||||||
{/**----------- % of Column -----------*/}
|
{/**----------- % of Column -----------*/}
|
||||||
<FastField name={'percentageColumn'} type={'checkbox'}>
|
<FastField name={'percentageColumn'} type={'checkbox'}>
|
||||||
{({ field }) => (
|
{({ field }) => (
|
||||||
@@ -126,6 +131,7 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
|
|
||||||
{/**----------- % of Row -----------*/}
|
{/**----------- % of Row -----------*/}
|
||||||
<FastField name={'percentageRow'} type={'checkbox'}>
|
<FastField name={'percentageRow'} type={'checkbox'}>
|
||||||
{({ field }) => (
|
{({ field }) => (
|
||||||
@@ -139,6 +145,7 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
|
|
||||||
{/**----------- % of Expense -----------*/}
|
{/**----------- % of Expense -----------*/}
|
||||||
<FastField name={'percentageExpense'} type={'checkbox'}>
|
<FastField name={'percentageExpense'} type={'checkbox'}>
|
||||||
{({ field }) => (
|
{({ field }) => (
|
||||||
@@ -152,6 +159,7 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
|
|
||||||
{/**----------- % of Income -----------*/}
|
{/**----------- % of Income -----------*/}
|
||||||
<FastField name={'percentageIncome'} type={'checkbox'}>
|
<FastField name={'percentageIncome'} type={'checkbox'}>
|
||||||
{({ field }) => (
|
{({ field }) => (
|
||||||
@@ -165,19 +173,33 @@ export default function ProfitLossSheetHeaderComparisonPanel() {
|
|||||||
</FormGroup>
|
</FormGroup>
|
||||||
)}
|
)}
|
||||||
</FastField>
|
</FastField>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ProfitLoss sheet header -comparison panel.
|
||||||
|
*/
|
||||||
|
export default function ProfitLossSheetHeaderComparisonPanel() {
|
||||||
|
return (
|
||||||
|
<ProfitLossSheetComparisonWrap>
|
||||||
|
<ProfitLossComaprsionFieldsWrap>
|
||||||
|
<ProfitLossComaprsionPanelFields />
|
||||||
|
</ProfitLossComaprsionFieldsWrap>
|
||||||
</ProfitLossSheetComparisonWrap>
|
</ProfitLossSheetComparisonWrap>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const ProfitLossSheetComparisonWrap = styled.div`
|
const ProfitLossSheetComparisonWrap = styled.div`
|
||||||
.row {
|
|
||||||
margin-left: 0.15rem;
|
|
||||||
.col {
|
|
||||||
min-width: 150px !important;
|
|
||||||
max-width: 190px !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.bp3-form-group {
|
.bp3-form-group {
|
||||||
margin-bottom: 3px;
|
margin-bottom: 3px;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
const FlexSubFields = styled(Flex)`
|
||||||
|
padding-left: 20px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const ProfitLossComaprsionFieldsWrap = styled.div`
|
||||||
|
max-width: 400px;
|
||||||
|
`;
|
||||||
|
|||||||
@@ -1,9 +1,14 @@
|
|||||||
import { useCallback } from 'react';
|
import { useCallback } from 'react';
|
||||||
import { useDispatch } from 'react-redux';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
|
import { createSelector } from 'reselect';
|
||||||
import {
|
import {
|
||||||
splashStopLoading,
|
splashStopLoading,
|
||||||
splashStartLoading,
|
splashStartLoading,
|
||||||
dashboardPageTitle,
|
dashboardPageTitle,
|
||||||
|
openSidebarSubmenu,
|
||||||
|
closeSidebarSubmenu,
|
||||||
|
openDialog,
|
||||||
|
closeDialog,
|
||||||
} from '../../store/dashboard/dashboard.actions';
|
} from '../../store/dashboard/dashboard.actions';
|
||||||
|
|
||||||
export const useDispatchAction = (action) => {
|
export const useDispatchAction = (action) => {
|
||||||
@@ -30,3 +35,44 @@ export const useSplashLoading = () => {
|
|||||||
useDispatchAction(splashStopLoading),
|
useDispatchAction(splashStopLoading),
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sidebar submenu actions.
|
||||||
|
*/
|
||||||
|
export const useSidebarSubmnuActions = () => {
|
||||||
|
return {
|
||||||
|
openSidebarSubmenu: useDispatchAction(openSidebarSubmenu),
|
||||||
|
closeSidebarSubmenu: useDispatchAction(closeSidebarSubmenu),
|
||||||
|
toggleSidebarSubmenu: useDispatchAction(openSidebarSubmenu),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sidebar submenu selector.
|
||||||
|
*/
|
||||||
|
const sidebarSubmenuSelector = createSelector(
|
||||||
|
(state) => state.dashboard.sidebarSubmenu,
|
||||||
|
(sidebarSubmenu) => sidebarSubmenu,
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sidebar submenu selector.
|
||||||
|
*/
|
||||||
|
export const useSidebarSubmenu = () => {
|
||||||
|
const sidebarSubmenu = useSelector(sidebarSubmenuSelector);
|
||||||
|
|
||||||
|
return {
|
||||||
|
isOpen: sidebarSubmenu?.isOpen || false,
|
||||||
|
submenuId: sidebarSubmenu?.submenuId || null,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dialogs actions.
|
||||||
|
*/
|
||||||
|
export const useDialogActions = () => {
|
||||||
|
return {
|
||||||
|
openDialog: useDispatchAction(openDialog),
|
||||||
|
closeDialog: useDispatchAction(closeDialog),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|||||||
@@ -111,3 +111,16 @@ export const setFeatureDashboardMeta = ({ features }) => {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export function openSidebarSubmenu({ submenuId }) {
|
||||||
|
return {
|
||||||
|
type: t.SIDEBAR_SUBMENU_OPEN,
|
||||||
|
payload: { submenuId },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function closeSidebarSubmenu() {
|
||||||
|
return {
|
||||||
|
type: t.SIDEBAR_SUBMENU_CLOSE,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|||||||
@@ -20,10 +20,8 @@ const initialState = {
|
|||||||
splashScreenLoading: null,
|
splashScreenLoading: null,
|
||||||
appIsLoading: true,
|
appIsLoading: true,
|
||||||
appIntlIsLoading: true,
|
appIntlIsLoading: true,
|
||||||
features: {
|
sidebarSubmenu: { isOpen: false, submenuId: null },
|
||||||
// branches: true,
|
features: {},
|
||||||
// warehouses: true,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const STORAGE_KEY = 'bigcapital:dashboard';
|
const STORAGE_KEY = 'bigcapital:dashboard';
|
||||||
@@ -131,6 +129,16 @@ const reducerInstance = createReducer(initialState, {
|
|||||||
state.splashScreenLoading = Math.max(state.splashScreenLoading, 0);
|
state.splashScreenLoading = Math.max(state.splashScreenLoading, 0);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
[t.SIDEBAR_SUBMENU_OPEN]: (state, action) => {
|
||||||
|
state.sidebarSubmenu.isOpen = true;
|
||||||
|
state.sidebarSubmenu.submenuId = action.payload.submenuId;
|
||||||
|
},
|
||||||
|
|
||||||
|
[t.SIDEBAR_SUBMENU_CLOSE]: (state, action) => {
|
||||||
|
state.sidebarSubmenu.isOpen = false;
|
||||||
|
state.sidebarSubmenu.submenuId = null;
|
||||||
|
},
|
||||||
|
|
||||||
[t.RESET]: () => {
|
[t.RESET]: () => {
|
||||||
purgeStoredState(CONFIG);
|
purgeStoredState(CONFIG);
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -13,6 +13,9 @@ export default {
|
|||||||
ALTER_DASHBOARD_PAGE_SUBTITLE: 'ALTER_DASHBOARD_PAGE_SUBTITLE',
|
ALTER_DASHBOARD_PAGE_SUBTITLE: 'ALTER_DASHBOARD_PAGE_SUBTITLE',
|
||||||
SET_TOPBAR_EDIT_VIEW: 'SET_TOPBAR_EDIT_VIEW',
|
SET_TOPBAR_EDIT_VIEW: 'SET_TOPBAR_EDIT_VIEW',
|
||||||
SIDEBAR_EXPEND_TOGGLE: 'SIDEBAR_EXPEND_TOGGLE',
|
SIDEBAR_EXPEND_TOGGLE: 'SIDEBAR_EXPEND_TOGGLE',
|
||||||
|
SIDEBAR_SUBMENU_OPEN: 'SIDEBAR_SUBMENU_OPEN',
|
||||||
|
SIDEBAR_SUBMENU_CLOSE: 'SIDEBAR_SUBMENU_CLOSE',
|
||||||
|
SIDEBAR_SUBMENU_TOGGLE: 'SIDEBAR_SUBMENU_TOGGLE',
|
||||||
SET_DASHBOARD_BACK_LINK: 'SET_DASHBOARD_BACK_LINK',
|
SET_DASHBOARD_BACK_LINK: 'SET_DASHBOARD_BACK_LINK',
|
||||||
SPLASH_START_LOADING: 'SPLASH_START_LOADING',
|
SPLASH_START_LOADING: 'SPLASH_START_LOADING',
|
||||||
SPLASH_STOP_LOADING: 'SPLASH_STOP_LOADING',
|
SPLASH_STOP_LOADING: 'SPLASH_STOP_LOADING',
|
||||||
|
|||||||
@@ -75,6 +75,8 @@
|
|||||||
padding: 14px 20px 10px;
|
padding: 14px 20px 10px;
|
||||||
letter-spacing: 1px;
|
letter-spacing: 1px;
|
||||||
color: #707a85;
|
color: #707a85;
|
||||||
|
border-bottom: 1px solid #e2e5ec;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
|
||||||
html[lang^="ar"] & {
|
html[lang^="ar"] & {
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
|
|||||||
33
src/utils/deep.js
Normal file
33
src/utils/deep.js
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
import _ from 'lodash';
|
||||||
|
import Deepdash from 'deepdash';
|
||||||
|
|
||||||
|
export const deepdash = Deepdash(_);
|
||||||
|
|
||||||
|
export const filterValuesDeep = (predicate, nodes) => {
|
||||||
|
return deepdash.condense(
|
||||||
|
deepdash.reduceDeep(
|
||||||
|
nodes,
|
||||||
|
(accumulator, value, key, parent, context) => {
|
||||||
|
const newValue = { ...value };
|
||||||
|
|
||||||
|
if (newValue.children) {
|
||||||
|
_.set(newValue, 'children', deepdash.condense(value.children));
|
||||||
|
}
|
||||||
|
const isTrue = predicate(newValue, key, parent, context);
|
||||||
|
|
||||||
|
if (isTrue === true) {
|
||||||
|
_.set(accumulator, context.path, newValue);
|
||||||
|
} else if (isTrue === false) {
|
||||||
|
_.unset(accumulator, context.path);
|
||||||
|
}
|
||||||
|
return accumulator;
|
||||||
|
},
|
||||||
|
[],
|
||||||
|
{
|
||||||
|
childrenPath: 'children',
|
||||||
|
pathFormat: 'array',
|
||||||
|
callbackAfterIterate: true,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -14,6 +14,8 @@ import { isEqual } from 'lodash';
|
|||||||
|
|
||||||
import jsCookie from 'js-cookie';
|
import jsCookie from 'js-cookie';
|
||||||
|
|
||||||
|
export * from './deep';
|
||||||
|
|
||||||
export const getCookie = (name, defaultValue) =>
|
export const getCookie = (name, defaultValue) =>
|
||||||
_.defaultTo(jsCookie.get(name), defaultValue);
|
_.defaultTo(jsCookie.get(name), defaultValue);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user