From f6b6aa0ae8ea663c4a132e2ebff7a31cfb16ecf8 Mon Sep 17 00:00:00 2001
From: andreas-unleash <104830839+andreas-unleash@users.noreply.github.com>
Date: Fri, 22 Jul 2022 09:40:08 +0300
Subject: [PATCH] Refactored to single responsibility components
---
.../ConstraintAccordion.styles.ts | 4 +-
.../RestrictiveLegalValues.tsx | 54 ++--
.../SingleLegalValue/SingleLegalValue.tsx | 10 +-
.../ConstraintAccordionEditHeader.tsx | 21 +-
.../ConstraintAccordionHeaderActions.tsx | 63 +++++
.../ConstraintAccordionViewHeader.tsx | 230 ++----------------
.../ConstraintAccordionViewHeaderInfo.tsx | 65 +++++
.../ConstraintViewHeaderOperator.tsx | 36 +++
...raintAccordionViewHeaderMultipleValues.tsx | 102 ++++++++
...ontraintAccordionViewHeaderSingleValue.tsx | 49 ++++
.../StyledIconWrapper/StyledIconWrapper.tsx | 16 ++
.../{utils.ts => helpers.ts} | 0
12 files changed, 389 insertions(+), 261 deletions(-)
create mode 100644 frontend/src/component/common/ConstraintAccordion/ConstraintAccordionHeaderActions/ConstraintAccordionHeaderActions.tsx
create mode 100644 frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeaderInfo/ConstraintAccordionViewHeaderInfo.tsx
create mode 100644 frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintViewHeaderOperator/ConstraintViewHeaderOperator.tsx
create mode 100644 frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderMultipleValues/ConstraintAccordionViewHeaderMultipleValues.tsx
create mode 100644 frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderSingleValue/ContraintAccordionViewHeaderSingleValue.tsx
create mode 100644 frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/StyledIconWrapper/StyledIconWrapper.tsx
rename frontend/src/component/common/ConstraintAccordion/{utils.ts => helpers.ts} (100%)
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordion.styles.ts b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordion.styles.ts
index 0aa5b91745..7ed5067e05 100644
--- a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordion.styles.ts
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordion.styles.ts
@@ -17,8 +17,8 @@ export const useStyles = makeStyles()(theme => ({
fill: '#fff',
},
accordion: {
- border: `1px solid ${theme.palette.grey[300]}`,
- borderRadius: '5px',
+ border: `1px solid ${theme.palette.grey[400]}`,
+ borderRadius: '8px',
backgroundColor: '#fff',
boxShadow: 'none',
margin: 0,
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/RestrictiveLegalValues/RestrictiveLegalValues.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/RestrictiveLegalValues/RestrictiveLegalValues.tsx
index f269a3c171..c32fa68a75 100644
--- a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/RestrictiveLegalValues/RestrictiveLegalValues.tsx
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/RestrictiveLegalValues/RestrictiveLegalValues.tsx
@@ -64,29 +64,37 @@ export const RestrictiveLegalValues = ({
};
return (
- <>
-
- Select values from a predefined set
-
-
- {filteredValues.map(match => (
- onChange(match.value)}
- name={match.value}
- color="primary"
+ 500)}
+ show={
+ <>
+
+ Select values from a predefined set
+
+
+ {filteredValues.map(match => (
+ onChange(match.value)}
+ name={match.value}
+ color="primary"
+ />
+ }
/>
- }
- />
- ))}
- {error}
}
- />
- >
+ ))}
+ {error}}
+ />
+ >
+ }
+ />
);
};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/SingleLegalValue/SingleLegalValue.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/SingleLegalValue/SingleLegalValue.tsx
index 158cd221af..f62dc67d3c 100644
--- a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/SingleLegalValue/SingleLegalValue.tsx
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionEdit/ConstraintAccordionEditBody/SingleLegalValue/SingleLegalValue.tsx
@@ -36,7 +36,15 @@ export const SingleLegalValue = ({
Add a single {type.toLowerCase()} value
-
+ 500)}
+ show={
+
+ }
+ />
{
- event.stopPropagation();
- onDelete();
- });
-
return (
@@ -155,16 +147,7 @@ export const ConstraintAccordionEditHeader = ({
}
/>
-
-
-
-
-
-
-
-
-
-
+
);
};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionHeaderActions/ConstraintAccordionHeaderActions.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionHeaderActions/ConstraintAccordionHeaderActions.tsx
new file mode 100644
index 0000000000..202eafc065
--- /dev/null
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionHeaderActions/ConstraintAccordionHeaderActions.tsx
@@ -0,0 +1,63 @@
+import React from 'react';
+import { ConditionallyRender } from '../../ConditionallyRender/ConditionallyRender';
+import { IconButton, Tooltip } from '@mui/material';
+import { Delete, Edit } from '@mui/icons-material';
+import { useStyles } from '../ConstraintAccordion.styles';
+
+interface ConstraintAccordionHeaderActionsProps {
+ onDelete?: () => void;
+ onEdit?: () => void;
+}
+
+export const ConstraintAccordionHeaderActions = ({
+ onEdit,
+ onDelete,
+}: ConstraintAccordionHeaderActionsProps) => {
+ const { classes: styles } = useStyles();
+ const onEditClick =
+ onEdit &&
+ ((event: React.SyntheticEvent) => {
+ event.stopPropagation();
+ onEdit();
+ });
+
+ const onDeleteClick =
+ onDelete &&
+ ((event: React.SyntheticEvent) => {
+ event.stopPropagation();
+ onDelete();
+ });
+
+ return (
+
+ (
+
+
+
+
+
+ )}
+ />
+ (
+
+
+
+
+
+ )}
+ />
+
+ );
+};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeader.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeader.tsx
index 69d682ea08..c0061cc650 100644
--- a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeader.tsx
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeader.tsx
@@ -1,72 +1,10 @@
-import { Chip, IconButton, Tooltip, styled } from '@mui/material';
import { ConstraintIcon } from 'component/common/ConstraintAccordion/ConstraintIcon';
-import { Delete, Edit } from '@mui/icons-material';
import { IConstraint } from 'interfaces/strategy';
import { useStyles } from 'component/common/ConstraintAccordion/ConstraintAccordion.styles';
-import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
-import React, { useEffect, useRef, useState } from 'react';
-import { formatConstraintValue } from 'utils/formatConstraintValue';
-import { useLocationSettings } from 'hooks/useLocationSettings';
-import { ConstraintOperator } from 'component/common/ConstraintAccordion/ConstraintOperator/ConstraintOperator';
-import classnames from 'classnames';
-import { getTextWidth } from '../../utils';
-import { ReactComponent as NegatedIcon } from 'assets/icons/24_Negator.svg';
-import { ReactComponent as CaseSensitive } from 'assets/icons/24_Text format.svg';
-import { stringOperators } from 'constants/operators';
-import { oneOf } from 'utils/oneOf';
-
-const StyledHeaderText = styled('span')(({ theme }) => ({
- display: '-webkit-box',
- WebkitLineClamp: 3,
- WebkitBoxOrient: 'vertical',
- overflow: 'hidden',
- maxWidth: '100px',
- minWidth: '100px',
- marginRight: '10px',
- wordBreak: 'break-word',
- fontSize: theme.fontSizes.smallBody,
- [theme.breakpoints.down(710)]: {
- textAlign: 'center',
- padding: theme.spacing(1, 0),
- marginRight: 'inherit',
- maxWidth: 'inherit',
- },
-}));
-
-const StyledValuesSpan = styled('span')(({ theme }) => ({
- display: '-webkit-box',
- WebkitLineClamp: 2,
- WebkitBoxOrient: 'vertical',
- overflow: 'hidden',
- wordBreak: 'break-word',
- fontSize: theme.fontSizes.smallBody,
- [theme.breakpoints.down(710)]: {
- margin: theme.spacing(1, 0),
- textAlign: 'center',
- },
-}));
-
-const StyledSingleValueChip = styled(Chip)(({ theme }) => ({
- [theme.breakpoints.down(710)]: {
- margin: theme.spacing(1, 0),
- },
-}));
-
-const StyledIconWrapper = styled('div')<{
- marginRight?: string;
- marginTop?: string;
-}>(({ theme, marginRight, marginTop }) => ({
- backgroundColor: theme.palette.grey[200],
- width: 28,
- height: 47,
- display: 'inline-flex',
- justifyContent: 'center',
- padding: '10px 0',
- color: theme.palette.primary.main,
- marginRight: marginRight ? marginRight : '0.75rem',
- marginTop: marginTop ? marginTop : 0,
-}));
+import React from 'react';
+import { ConstraintAccordionViewHeaderInfo } from './ConstraintAccordionViewHeaderInfo/ConstraintAccordionViewHeaderInfo';
+import { ConstraintAccordionHeaderActions } from '../../ConstraintAccordionHeaderActions/ConstraintAccordionHeaderActions';
interface IConstraintAccordionViewHeaderProps {
compact: boolean;
@@ -88,161 +26,21 @@ export const ConstraintAccordionViewHeader = ({
expanded,
}: IConstraintAccordionViewHeaderProps) => {
const { classes: styles } = useStyles();
- const { locationSettings } = useLocationSettings();
- const [width, setWidth] = useState(0);
- const [textWidth, setTextWidth] = useState(0);
- const [expandable, setExpandable] = useState(false);
- const elementRef = useRef(null);
-
- const onEditClick =
- onEdit &&
- ((event: React.SyntheticEvent) => {
- event.stopPropagation();
- onEdit();
- });
-
- const onDeleteClick =
- onDelete &&
- ((event: React.SyntheticEvent) => {
- event.stopPropagation();
- onDelete();
- });
-
- useEffect(() => {
- if (elementRef && elementRef.current != null) {
- setTextWidth(
- Math.round(getTextWidth(elementRef.current.innerText) / 2) // 2 lines
- );
- setWidth(elementRef.current.clientWidth);
- }
- }, []);
-
- useEffect(() => {
- if (textWidth && width) {
- setExpandable(textWidth > width);
- allowExpand(textWidth > width);
- }
- }, [textWidth, width, allowExpand]);
return (
-
-
-
- {constraint.contextName}
-
-
-
-
-
-
-
-
- }
- />
-
-
-
-
-
-
-
- {' '}
-
-
- }
- />
-
-
- }
- elseShow={
-
-
-
- {' '}
-
-
- }
- />
-
-
- {constraint?.values
- ?.map(value => value)
- .join(', ')}
-
-
- {!expanded
- ? `Expand to view all (
- ${constraint?.values?.length})`
- : 'Collapse to view less'}
-
- }
- />
-
-
- }
- />
-
-
- (
-
-
-
-
-
- )}
- />
- (
-
-
-
-
-
- )}
- />
-
+
+
);
};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeaderInfo/ConstraintAccordionViewHeaderInfo.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeaderInfo/ConstraintAccordionViewHeaderInfo.tsx
new file mode 100644
index 0000000000..3e2fc4e282
--- /dev/null
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintAccordionViewHeaderInfo/ConstraintAccordionViewHeaderInfo.tsx
@@ -0,0 +1,65 @@
+import { styled, Tooltip } from '@mui/material';
+import { ConstraintViewHeaderOperator } from '../ConstraintViewHeaderOperator/ConstraintViewHeaderOperator';
+import { ConditionallyRender } from '../../../../ConditionallyRender/ConditionallyRender';
+import { ContraintAccordionViewHeaderSingleValue } from '../ContraintAccordionViewHeaderSingleValue/ContraintAccordionViewHeaderSingleValue';
+import { ConstraintAccordionViewHeaderMultipleValues } from '../ContraintAccordionViewHeaderMultipleValues/ConstraintAccordionViewHeaderMultipleValues';
+import React from 'react';
+import { IConstraint } from '../../../../../../interfaces/strategy';
+import { useStyles } from '../../../ConstraintAccordion.styles';
+
+const StyledHeaderText = styled('span')(({ theme }) => ({
+ display: '-webkit-box',
+ WebkitLineClamp: 3,
+ WebkitBoxOrient: 'vertical',
+ overflow: 'hidden',
+ maxWidth: '100px',
+ minWidth: '100px',
+ marginRight: '10px',
+ wordBreak: 'break-word',
+ fontSize: theme.fontSizes.smallBody,
+ [theme.breakpoints.down(710)]: {
+ textAlign: 'center',
+ padding: theme.spacing(1, 0),
+ marginRight: 'inherit',
+ maxWidth: 'inherit',
+ },
+}));
+
+interface ConstraintAccordionViewHeaderMetaInfoProps {
+ constraint: IConstraint;
+ singleValue: boolean;
+ expanded: boolean;
+ allowExpand: (shouldExpand: boolean) => void;
+}
+
+export const ConstraintAccordionViewHeaderInfo = ({
+ constraint,
+ singleValue,
+ allowExpand,
+ expanded,
+}: ConstraintAccordionViewHeaderMetaInfoProps) => {
+ const { classes: styles } = useStyles();
+ return (
+
+
+ {constraint.contextName}
+
+
+
+ }
+ elseShow={
+
+ }
+ />
+
+ );
+};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintViewHeaderOperator/ConstraintViewHeaderOperator.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintViewHeaderOperator/ConstraintViewHeaderOperator.tsx
new file mode 100644
index 0000000000..98022eddeb
--- /dev/null
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ConstraintViewHeaderOperator/ConstraintViewHeaderOperator.tsx
@@ -0,0 +1,36 @@
+import { IConstraint } from '../../../../../../interfaces/strategy';
+import { ConditionallyRender } from '../../../../ConditionallyRender/ConditionallyRender';
+import { Tooltip } from '@mui/material';
+import { ReactComponent as NegatedIcon } from '../../../../../../assets/icons/24_Negator.svg';
+import { ConstraintOperator } from '../../../ConstraintOperator/ConstraintOperator';
+import React from 'react';
+import { useStyles } from '../../../ConstraintAccordion.styles';
+import { StyledIconWrapper } from '../StyledIconWrapper/StyledIconWrapper';
+
+interface ConstraintViewHeaderOperatorProps {
+ constraint: IConstraint;
+}
+
+export const ConstraintViewHeaderOperator = ({
+ constraint,
+}: ConstraintViewHeaderOperatorProps) => {
+ const { classes: styles } = useStyles();
+
+ return (
+
+
+
+
+
+
+ }
+ />
+
+
+
+
+ );
+};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderMultipleValues/ConstraintAccordionViewHeaderMultipleValues.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderMultipleValues/ConstraintAccordionViewHeaderMultipleValues.tsx
new file mode 100644
index 0000000000..606c6c099d
--- /dev/null
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderMultipleValues/ConstraintAccordionViewHeaderMultipleValues.tsx
@@ -0,0 +1,102 @@
+import { ConditionallyRender } from '../../../../ConditionallyRender/ConditionallyRender';
+import { oneOf } from '../../../../../../utils/oneOf';
+import { stringOperators } from '../../../../../../constants/operators';
+import { styled, Tooltip } from '@mui/material';
+import { ReactComponent as CaseSensitive } from '../../../../../../assets/icons/24_Text format.svg';
+import React, { useEffect, useRef, useState } from 'react';
+import classnames from 'classnames';
+import { StyledIconWrapper } from '../StyledIconWrapper/StyledIconWrapper';
+import { IConstraint } from '../../../../../../interfaces/strategy';
+import { useStyles } from '../../../ConstraintAccordion.styles';
+import { getTextWidth } from '../../../helpers';
+
+const StyledValuesSpan = styled('span')(({ theme }) => ({
+ display: '-webkit-box',
+ WebkitLineClamp: 2,
+ WebkitBoxOrient: 'vertical',
+ overflow: 'hidden',
+ wordBreak: 'break-word',
+ fontSize: theme.fontSizes.smallBody,
+ [theme.breakpoints.down(710)]: {
+ margin: theme.spacing(1, 0),
+ textAlign: 'center',
+ },
+}));
+
+interface ConstraintSingleValueProps {
+ constraint: IConstraint;
+ expanded: boolean;
+ allowExpand: (shouldExpand: boolean) => void;
+}
+
+export const ConstraintAccordionViewHeaderMultipleValues = ({
+ constraint,
+ expanded,
+ allowExpand,
+}: ConstraintSingleValueProps) => {
+ const { classes: styles } = useStyles();
+
+ const elementRef = useRef(null);
+
+ const [width, setWidth] = useState(0);
+ const [textWidth, setTextWidth] = useState(0);
+ const [expandable, setExpandable] = useState(false);
+
+ useEffect(() => {
+ if (elementRef && elementRef.current != null) {
+ setTextWidth(
+ Math.round(getTextWidth(elementRef.current.innerText) / 2) // 2 lines
+ );
+ setWidth(elementRef.current.clientWidth);
+ }
+ }, []);
+
+ useEffect(() => {
+ if (textWidth && width) {
+ setExpandable(textWidth > width);
+ }
+ }, [textWidth, width]);
+
+ useEffect(() => {
+ allowExpand(expandable);
+ }, [expandable, allowExpand]);
+
+ return (
+
+
+
+ {' '}
+
+
+ }
+ />
+
+
+ {constraint?.values?.map(value => value).join(', ')}
+
+
+ {!expanded
+ ? `Expand to view all (
+ ${constraint?.values?.length})`
+ : 'Collapse to view less'}
+
+ }
+ />
+
+
+ );
+};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderSingleValue/ContraintAccordionViewHeaderSingleValue.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderSingleValue/ContraintAccordionViewHeaderSingleValue.tsx
new file mode 100644
index 0000000000..00818af583
--- /dev/null
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/ContraintAccordionViewHeaderSingleValue/ContraintAccordionViewHeaderSingleValue.tsx
@@ -0,0 +1,49 @@
+import { ConditionallyRender } from '../../../../ConditionallyRender/ConditionallyRender';
+import { oneOf } from '../../../../../../utils/oneOf';
+import { stringOperators } from '../../../../../../constants/operators';
+import { Chip, styled, Tooltip } from '@mui/material';
+import { ReactComponent as CaseSensitive } from '../../../../../../assets/icons/24_Text format.svg';
+import { formatConstraintValue } from '../../../../../../utils/formatConstraintValue';
+import React from 'react';
+import { useStyles } from '../../../ConstraintAccordion.styles';
+import { StyledIconWrapper } from '../StyledIconWrapper/StyledIconWrapper';
+import { IConstraint } from '../../../../../../interfaces/strategy';
+import { useLocationSettings } from '../../../../../../hooks/useLocationSettings';
+
+const StyledSingleValueChip = styled(Chip)(({ theme }) => ({
+ [theme.breakpoints.down(710)]: {
+ margin: theme.spacing(1, 0),
+ },
+}));
+
+interface ConstraintSingleValueProps {
+ constraint: IConstraint;
+}
+
+export const ContraintAccordionViewHeaderSingleValue = ({
+ constraint,
+}: ConstraintSingleValueProps) => {
+ const { locationSettings } = useLocationSettings();
+ const { classes: styles } = useStyles();
+
+ return (
+
+
+
+ {' '}
+
+
+ }
+ />
+
+
+ );
+};
diff --git a/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/StyledIconWrapper/StyledIconWrapper.tsx b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/StyledIconWrapper/StyledIconWrapper.tsx
new file mode 100644
index 0000000000..0d84e094fb
--- /dev/null
+++ b/frontend/src/component/common/ConstraintAccordion/ConstraintAccordionView/ConstraintAccordionViewHeader/StyledIconWrapper/StyledIconWrapper.tsx
@@ -0,0 +1,16 @@
+import { styled } from '@mui/material';
+
+export const StyledIconWrapper = styled('div')<{
+ marginRight?: string;
+ marginTop?: string;
+}>(({ theme, marginRight, marginTop }) => ({
+ backgroundColor: theme.palette.grey[200],
+ width: 28,
+ height: 47,
+ display: 'inline-flex',
+ justifyContent: 'center',
+ padding: '10px 0',
+ color: theme.palette.primary.main,
+ marginRight: marginRight ? marginRight : '0.75rem',
+ marginTop: marginTop ? marginTop : 0,
+}));
diff --git a/frontend/src/component/common/ConstraintAccordion/utils.ts b/frontend/src/component/common/ConstraintAccordion/helpers.ts
similarity index 100%
rename from frontend/src/component/common/ConstraintAccordion/utils.ts
rename to frontend/src/component/common/ConstraintAccordion/helpers.ts