Use Graphql types in FE and complete mappers removal (#348)

Fix Typescript build issues
This commit is contained in:
Charles Bochet 2023-06-21 10:52:00 -07:00 committed by GitHub
parent b179d1f1f0
commit 8a330b9746
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
35 changed files with 398 additions and 574 deletions

View File

@ -1,8 +1,16 @@
import { atom } from 'recoil'; import { atom } from 'recoil';
import { User } from '@/users/interfaces/user.interface'; import { User, Workspace, WorkspaceMember } from '~/generated/graphql';
export const currentUserState = atom<User | null>({ type CurrentUser = Pick<User, 'id' | 'email' | 'displayName'> & {
workspaceMember?:
| (Pick<WorkspaceMember, 'id'> & {
workspace: Pick<Workspace, 'id' | 'displayName' | 'logo'>;
})
| null;
};
export const currentUserState = atom<CurrentUser | null>({
key: 'currentUserState', key: 'currentUserState',
default: null, default: null,
}); });

View File

@ -2,19 +2,24 @@ import { CellCommentChip } from '@/comments/components/CellCommentChip';
import { useOpenCommentRightDrawer } from '@/comments/hooks/useOpenCommentRightDrawer'; import { useOpenCommentRightDrawer } from '@/comments/hooks/useOpenCommentRightDrawer';
import EditableChip from '@/ui/components/editable-cell/types/EditableChip'; import EditableChip from '@/ui/components/editable-cell/types/EditableChip';
import { getLogoUrlFromDomainName } from '@/utils/utils'; import { getLogoUrlFromDomainName } from '@/utils/utils';
import { CommentableType } from '~/generated/graphql'; import {
CommentableType,
import { Company } from '../interfaces/company.interface'; GetCompaniesQuery,
import { updateCompany } from '../services'; useUpdateCompanyMutation,
} from '~/generated/graphql';
import CompanyChip from './CompanyChip'; import CompanyChip from './CompanyChip';
type OwnProps = { type OwnProps = {
company: Company; company: Pick<
GetCompaniesQuery['companies'][0],
'id' | 'name' | 'domainName' | '_commentCount' | 'accountOwner'
>;
}; };
export function CompanyEditableNameChipCell({ company }: OwnProps) { export function CompanyEditableNameChipCell({ company }: OwnProps) {
const openCommentRightDrawer = useOpenCommentRightDrawer(); const openCommentRightDrawer = useOpenCommentRightDrawer();
const [updateCompany] = useUpdateCompanyMutation();
function handleCommentClick(event: React.MouseEvent<HTMLDivElement>) { function handleCommentClick(event: React.MouseEvent<HTMLDivElement>) {
event.preventDefault(); event.preventDefault();
@ -35,8 +40,11 @@ export function CompanyEditableNameChipCell({ company }: OwnProps) {
picture={getLogoUrlFromDomainName(company.domainName)} picture={getLogoUrlFromDomainName(company.domainName)}
changeHandler={(value: string) => { changeHandler={(value: string) => {
updateCompany({ updateCompany({
...company, variables: {
name: value, ...company,
name: value,
accountOwnerId: company.accountOwner?.id,
},
}); });
}} }}
ChipComponent={CompanyChip} ChipComponent={CompanyChip}

View File

@ -1,13 +0,0 @@
import { Company as GQLCompany } from '../../../generated/graphql';
import { DeepPartial } from '../../utils/utils';
export type Company = DeepPartial<GQLCompany> & { id: string };
export type GraphqlQueryCompany = Company;
export type GraphqlMutationCompany = Company;
export const mapToCompany = (company: GraphqlQueryCompany): Company => company;
export const mapToGqlCompany = (company: Company): GraphqlMutationCompany =>
company;

View File

@ -1,12 +1,4 @@
import { FetchResult, gql } from '@apollo/client'; import { gql } from '@apollo/client';
import { getOperationName } from '@apollo/client/utilities';
import { apiClient } from '~/apollo';
import { UpdateCompanyMutationVariables } from '../../../generated/graphql';
import { Company, mapToGqlCompany } from '../interfaces/company.interface';
import { GET_COMPANIES } from './select';
export const UPDATE_COMPANY = gql` export const UPDATE_COMPANY = gql`
mutation UpdateCompany( mutation UpdateCompany(
@ -80,36 +72,3 @@ export const DELETE_COMPANIES = gql`
} }
} }
`; `;
export async function updateCompany(
company: UpdateCompanyMutationVariables,
): Promise<FetchResult<Company>> {
const result = await apiClient.mutate({
mutation: UPDATE_COMPANY,
variables: company,
});
return result;
}
export async function insertCompany(
company: Company,
): Promise<FetchResult<Company>> {
const result = await apiClient.mutate({
mutation: INSERT_COMPANY,
variables: mapToGqlCompany(company),
refetchQueries: [getOperationName(GET_COMPANIES) ?? ''],
});
return result;
}
export async function deleteCompanies(
peopleIds: string[],
): Promise<FetchResult<Company>> {
const result = await apiClient.mutate({
mutation: DELETE_COMPANIES,
variables: { ids: peopleIds },
});
return result;
}

View File

@ -1,23 +1,17 @@
import { SortOrder as Order_By } from '~/generated/graphql'; import { SortOrder as Order_By } from '~/generated/graphql';
import { BoolExpType } from '../utils/interfaces/generic.interface';
import { import {
FilterableFieldsType,
FilterWhereType, FilterWhereType,
SelectedFilterType, SelectedFilterType,
} from './interfaces/filters/interface'; } from './interfaces/filters/interface';
import { SelectedSortType } from './interfaces/sorts/interface'; import { SelectedSortType } from './interfaces/sorts/interface';
export const reduceFiltersToWhere = < export const reduceFiltersToWhere = <WhereTemplateType extends FilterWhereType>(
ValueType extends FilterableFieldsType, filters: Array<SelectedFilterType<WhereTemplateType>>,
WhereTemplateType extends FilterWhereType, ): Record<string, any> => {
>(
filters: Array<SelectedFilterType<ValueType, WhereTemplateType>>,
): BoolExpType<WhereTemplateType> => {
const where = filters.reduce((acc, filter) => { const where = filters.reduce((acc, filter) => {
return { ...acc, ...filter.operand.whereTemplate(filter.value) }; return { ...acc, ...filter.operand.whereTemplate(filter.value) };
}, {} as BoolExpType<WhereTemplateType>); }, {} as Record<string, any>);
return where; return where;
}; };

View File

@ -1,80 +1,64 @@
import { ReactNode } from 'react'; import { ReactNode } from 'react';
import { SearchConfigType } from '@/search/interfaces/interface'; import { SearchConfigType } from '@/search/interfaces/interface';
import {
AnyEntity,
BoolExpType,
UnknownType,
} from '@/utils/interfaces/generic.interface';
export type FilterableFieldsType = AnyEntity; export type FilterableFieldsType = any;
export type FilterWhereRelationType = AnyEntity; export type FilterWhereRelationType = any;
export type FilterWhereType = FilterWhereRelationType | string | UnknownType; export type FilterWhereType = FilterWhereRelationType | string | unknown;
export type FilterConfigType< export type FilterConfigType<WhereType extends FilterWhereType = unknown> = {
FilteredType extends FilterableFieldsType,
WhereType extends FilterWhereType = UnknownType,
> = {
key: string; key: string;
label: string; label: string;
icon: ReactNode; icon: ReactNode;
type: WhereType extends UnknownType type: WhereType extends unknown
? 'relation' | 'text' | 'date' ? 'relation' | 'text' | 'date'
: WhereType extends AnyEntity : WhereType extends any
? 'relation' ? 'relation'
: WhereType extends string : WhereType extends string
? 'text' | 'date' ? 'text' | 'date'
: never; : never;
operands: FilterOperandType<FilteredType, WhereType>[]; operands: FilterOperandType<WhereType>[];
} & (WhereType extends UnknownType } & (WhereType extends unknown
? { searchConfig?: SearchConfigType<UnknownType> } ? { searchConfig?: SearchConfigType }
: WhereType extends AnyEntity : WhereType extends any
? { searchConfig: SearchConfigType<WhereType> } ? { searchConfig: SearchConfigType }
: WhereType extends string : WhereType extends string
? object ? object
: never) & : never) &
(WhereType extends UnknownType (WhereType extends unknown
? { selectedValueRender?: (selected: any) => string } ? { selectedValueRender?: (selected: any) => string }
: WhereType extends AnyEntity : WhereType extends any
? { selectedValueRender: (selected: WhereType) => string } ? { selectedValueRender: (selected: WhereType) => string }
: WhereType extends string : WhereType extends string
? object ? object
: never); : never);
export type FilterOperandType< export type FilterOperandType<WhereType extends FilterWhereType = unknown> =
FilteredType extends FilterableFieldsType, WhereType extends unknown
WhereType extends FilterWhereType = UnknownType, ? any
> = WhereType extends UnknownType : WhereType extends FilterWhereRelationType
? any ? FilterOperandRelationType<WhereType>
: WhereType extends FilterWhereRelationType : WhereType extends string
? FilterOperandRelationType<FilteredType, WhereType> ? FilterOperandFieldType
: WhereType extends string : never;
? FilterOperandFieldType<FilteredType>
: never;
type FilterOperandRelationType< type FilterOperandRelationType<WhereType extends FilterWhereType> = {
FilteredType extends FilterableFieldsType,
WhereType extends FilterWhereType,
> = {
label: 'Is' | 'Is not'; label: 'Is' | 'Is not';
id: 'is' | 'is_not'; id: 'is' | 'is_not';
whereTemplate: (value: WhereType) => BoolExpType<FilteredType>; whereTemplate: (value: WhereType) => any;
}; };
type FilterOperandFieldType<FilteredType extends FilterableFieldsType> = { type FilterOperandFieldType = {
label: 'Contains' | 'Does not contain' | 'Greater than' | 'Less than'; label: 'Contains' | 'Does not contain' | 'Greater than' | 'Less than';
id: 'like' | 'not_like' | 'greater_than' | 'less_than'; id: 'like' | 'not_like' | 'greater_than' | 'less_than';
whereTemplate: (value: string) => BoolExpType<FilteredType>; whereTemplate: (value: string) => any;
}; };
export type SelectedFilterType< export type SelectedFilterType<WhereType> = {
FilteredType extends FilterableFieldsType,
WhereType extends FilterWhereType = UnknownType,
> = {
key: string; key: string;
value: WhereType extends UnknownType ? any : WhereType; value: WhereType;
displayValue: string; displayValue: string;
label: string; label: string;
icon: ReactNode; icon: ReactNode;
operand: FilterOperandType<FilteredType, WhereType>; operand: FilterOperandType<WhereType>;
}; };

View File

@ -4,14 +4,12 @@ import styled from '@emotion/styled';
import { CellCommentChip } from '@/comments/components/CellCommentChip'; import { CellCommentChip } from '@/comments/components/CellCommentChip';
import { useOpenCommentRightDrawer } from '@/comments/hooks/useOpenCommentRightDrawer'; import { useOpenCommentRightDrawer } from '@/comments/hooks/useOpenCommentRightDrawer';
import { EditableDoubleText } from '@/ui/components/editable-cell/types/EditableDoubleText'; import { EditableDoubleText } from '@/ui/components/editable-cell/types/EditableDoubleText';
import { CommentableType } from '~/generated/graphql'; import { CommentableType, Person } from '~/generated/graphql';
import { Person } from '../interfaces/person.interface';
import { PersonChip } from './PersonChip'; import { PersonChip } from './PersonChip';
type OwnProps = { type OwnProps = {
person: Person; person: Pick<Person, 'id' | 'firstname' | 'lastname' | '_commentCount'>;
onChange: (firstname: string, lastname: string) => void; onChange: (firstname: string, lastname: string) => void;
}; };

View File

@ -4,27 +4,25 @@ import { v4 } from 'uuid';
import CompanyChip, { import CompanyChip, {
CompanyChipPropsType, CompanyChipPropsType,
} from '@/companies/components/CompanyChip'; } from '@/companies/components/CompanyChip';
import {
Company,
mapToCompany,
} from '@/companies/interfaces/company.interface';
import { SearchConfigType } from '@/search/interfaces/interface'; import { SearchConfigType } from '@/search/interfaces/interface';
import { SEARCH_COMPANY_QUERY } from '@/search/services/search'; import { SEARCH_COMPANY_QUERY } from '@/search/services/search';
import { EditableRelation } from '@/ui/components/editable-cell/types/EditableRelation'; import { EditableRelation } from '@/ui/components/editable-cell/types/EditableRelation';
import { logError } from '@/utils/logs/logError'; import { logError } from '@/utils/logs/logError';
import { getLogoUrlFromDomainName } from '@/utils/utils'; import { getLogoUrlFromDomainName } from '@/utils/utils';
import { import {
Company,
Person,
QueryMode, QueryMode,
useInsertCompanyMutation, useInsertCompanyMutation,
useUpdatePeopleMutation, useUpdatePeopleMutation,
} from '~/generated/graphql'; } from '~/generated/graphql';
import { mapToGqlPerson, Person } from '../interfaces/person.interface';
import { PeopleCompanyCreateCell } from './PeopleCompanyCreateCell'; import { PeopleCompanyCreateCell } from './PeopleCompanyCreateCell';
export type OwnProps = { export type OwnProps = {
people: Person; people: Pick<Person, 'id'> & {
company?: Pick<Company, 'id' | 'name'> | null;
};
}; };
export function PeopleCompanyCell({ people }: OwnProps) { export function PeopleCompanyCell({ people }: OwnProps) {
@ -52,7 +50,7 @@ export function PeopleCompanyCell({ people }: OwnProps) {
await updatePeople({ await updatePeople({
variables: { variables: {
...mapToGqlPerson(people), ...people,
companyId: newCompanyId, companyId: newCompanyId,
}, },
}); });
@ -75,7 +73,7 @@ export function PeopleCompanyCell({ people }: OwnProps) {
onCreate={handleCompanyCreate} onCreate={handleCompanyCreate}
/> />
) : ( ) : (
<EditableRelation<Company, CompanyChipPropsType> <EditableRelation<any, CompanyChipPropsType>
relation={people.company} relation={people.company}
searchPlaceholder="Company" searchPlaceholder="Company"
ChipComponent={CompanyChip} ChipComponent={CompanyChip}
@ -88,7 +86,7 @@ export function PeopleCompanyCell({ people }: OwnProps) {
onChange={async (relation) => { onChange={async (relation) => {
await updatePeople({ await updatePeople({
variables: { variables: {
...mapToGqlPerson(people), ...people,
companyId: relation.id, companyId: relation.id,
}, },
}); });
@ -101,10 +99,10 @@ export function PeopleCompanyCell({ people }: OwnProps) {
name: { contains: `%${searchInput}%`, mode: QueryMode.Insensitive }, name: { contains: `%${searchInput}%`, mode: QueryMode.Insensitive },
}), }),
resultMapper: (company) => ({ resultMapper: (company) => ({
render: (company) => company.name, render: (company: any) => company.name,
value: mapToCompany(company), value: company,
}), }),
} satisfies SearchConfigType<Company> } satisfies SearchConfigType
} }
onCreate={() => { onCreate={() => {
setIsCreating(true); setIsCreating(true);

View File

@ -1,12 +0,0 @@
import { Person as GQLPerson } from '../../../generated/graphql';
import { DeepPartial } from '../../utils/utils';
export type Person = DeepPartial<GQLPerson> & { id: GQLPerson['id'] };
export type GraphqlQueryPerson = Person;
export type GraphqlMutationPerson = Person;
export const mapToPerson = (person: GraphqlQueryPerson): Person => person;
export const mapToGqlPerson = (person: Person): GraphqlMutationPerson => person;

View File

@ -1,50 +0,0 @@
import {
GraphqlMutationPerson,
GraphqlQueryPerson,
} from '../../interfaces/person.interface';
import { updatePerson } from '../update';
jest.mock('~/apollo', () => {
const personInterface = jest.requireActual(
'@/people/interfaces/person.interface',
);
return {
apiClient: {
mutate: (arg: {
mutation: unknown;
variables: GraphqlMutationPerson;
}) => {
const gqlPerson = arg.variables as unknown as GraphqlQueryPerson;
return { data: personInterface.mapToPerson(gqlPerson) };
},
},
};
});
it('updates a person', async () => {
const result = await updatePerson({
firstname: 'John',
lastname: 'Doe',
id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6c',
email: 'john@example.com',
company: {
id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6b',
name: 'ACME',
domainName: 'example.com',
__typename: 'Company',
},
phone: '+1 (555) 123-4567',
pipes: [
{
id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6d',
name: 'Customer',
icon: '!',
},
],
createdAt: new Date().toISOString(),
city: 'San Francisco',
__typename: 'Person',
});
expect(result.data).toBeDefined();
result.data && expect(result.data.email).toBe('john@example.com');
});

View File

@ -1,10 +1,4 @@
import { FetchResult, gql } from '@apollo/client'; import { gql } from '@apollo/client';
import { getOperationName } from '@apollo/client/utilities';
import { apiClient } from '../../../apollo';
import { mapToGqlPerson, Person } from '../interfaces/person.interface';
import { GET_PEOPLE } from './select';
export const UPDATE_PERSON = gql` export const UPDATE_PERSON = gql`
mutation UpdatePeople( mutation UpdatePeople(
@ -90,36 +84,3 @@ export const DELETE_PEOPLE = gql`
} }
} }
`; `;
export async function updatePerson(
person: Person,
): Promise<FetchResult<Person>> {
const result = await apiClient.mutate({
mutation: UPDATE_PERSON,
variables: person,
});
return result;
}
export async function insertPerson(
person: Person,
): Promise<FetchResult<Person>> {
const result = await apiClient.mutate({
mutation: INSERT_PERSON,
variables: mapToGqlPerson(person),
refetchQueries: [getOperationName(GET_PEOPLE) ?? ''],
});
return result;
}
export async function deletePeople(
peopleIds: string[],
): Promise<FetchResult<Person>> {
const result = await apiClient.mutate({
mutation: DELETE_PEOPLE,
variables: { ids: peopleIds },
});
return result;
}

View File

@ -1,26 +1,7 @@
import { ReactNode } from 'react';
import { DocumentNode } from 'graphql'; import { DocumentNode } from 'graphql';
import { export type SearchConfigType = {
AnyEntity, query: DocumentNode;
BoolExpType, template: (searchInput: string) => any;
GqlType, resultMapper: (data: any) => any;
UnknownType, };
} from '@/utils/interfaces/generic.interface';
export type SearchConfigType<
SearchType extends AnyEntity | UnknownType = UnknownType,
> = SearchType extends UnknownType
? {
query: DocumentNode;
template: (searchInput: string) => any;
resultMapper: (data: any) => any;
}
: {
query: DocumentNode;
template: (searchInput: string) => BoolExpType<SearchType>;
resultMapper: (data: GqlType<SearchType>) => {
value: SearchType;
render: (value: SearchType) => ReactNode;
};
};

View File

@ -2,7 +2,6 @@ import { useMemo, useState } from 'react';
import { gql, useQuery } from '@apollo/client'; import { gql, useQuery } from '@apollo/client';
import { debounce } from '@/utils/debounce'; import { debounce } from '@/utils/debounce';
import { AnyEntity, UnknownType } from '@/utils/interfaces/generic.interface';
import { SearchConfigType } from '../interfaces/interface'; import { SearchConfigType } from '../interfaces/interface';
@ -64,22 +63,21 @@ export const SEARCH_COMPANY_QUERY = gql`
} }
`; `;
export type SearchResultsType<T extends AnyEntity | UnknownType = UnknownType> = export type SearchResultsType<T> = {
{ results: {
results: { render: (value: T) => string;
render: (value: T) => string; value: T;
value: T; }[];
}[]; loading: boolean;
loading: boolean; };
};
export const useSearch = <T extends AnyEntity | UnknownType = UnknownType>(): [ export const useSearch = <T>(): [
SearchResultsType<T>, SearchResultsType<T>,
React.Dispatch<React.SetStateAction<string>>, React.Dispatch<React.SetStateAction<string>>,
React.Dispatch<React.SetStateAction<SearchConfigType<T> | null>>, React.Dispatch<React.SetStateAction<SearchConfigType | null>>,
string, string,
] => { ] => {
const [searchConfig, setSearchConfig] = useState<SearchConfigType<T> | null>( const [searchConfig, setSearchConfig] = useState<SearchConfigType | null>(
null, null,
); );
const [searchInput, setSearchInput] = useState<string>(''); const [searchInput, setSearchInput] = useState<string>('');

View File

@ -8,7 +8,6 @@ import { useSearch } from '@/search/services/search';
import { IconPlus } from '@/ui/icons/index'; import { IconPlus } from '@/ui/icons/index';
import { textInputStyle } from '@/ui/layout/styles/themes'; import { textInputStyle } from '@/ui/layout/styles/themes';
import { isSomeInputInEditModeState } from '@/ui/tables/states/isSomeInputInEditModeState'; import { isSomeInputInEditModeState } from '@/ui/tables/states/isSomeInputInEditModeState';
import { AnyEntity } from '@/utils/interfaces/generic.interface';
import { isDefined } from '@/utils/type-guards/isDefined'; import { isDefined } from '@/utils/type-guards/isDefined';
import { isNonEmptyString } from '@/utils/type-guards/isNonEmptyString'; import { isNonEmptyString } from '@/utils/type-guards/isNonEmptyString';
@ -86,13 +85,10 @@ const StyledCreateButtonText = styled.div`
color: ${(props) => props.theme.text60}; color: ${(props) => props.theme.text60};
`; `;
export type EditableRelationProps< export type EditableRelationProps<RelationType, ChipComponentPropsType> = {
RelationType extends AnyEntity, relation?: any;
ChipComponentPropsType,
> = {
relation?: RelationType | null;
searchPlaceholder: string; searchPlaceholder: string;
searchConfig: SearchConfigType<RelationType>; searchConfig: SearchConfigType;
onChange: (relation: RelationType) => void; onChange: (relation: RelationType) => void;
onChangeSearchInput?: (searchInput: string) => void; onChangeSearchInput?: (searchInput: string) => void;
editModeHorizontalAlign?: 'left' | 'right'; editModeHorizontalAlign?: 'left' | 'right';
@ -105,10 +101,7 @@ export type EditableRelationProps<
}; };
// TODO: split this component // TODO: split this component
export function EditableRelation< export function EditableRelation<RelationType, ChipComponentPropsType>({
RelationType extends AnyEntity,
ChipComponentPropsType,
>({
relation, relation,
searchPlaceholder, searchPlaceholder,
searchConfig, searchConfig,

View File

@ -40,7 +40,8 @@ export const FilterDropdownButton = <TData extends FilterableFieldsType>({
FilterOperandType<TData> | undefined FilterOperandType<TData> | undefined
>(undefined); >(undefined);
const [filterSearchResults, setSearchInput, setFilterSearch] = useSearch(); const [filterSearchResults, setSearchInput, setFilterSearch] =
useSearch<TData>();
const resetState = useCallback(() => { const resetState = useCallback(() => {
setIsOperandSelectionUnfolded(false); setIsOperandSelectionUnfolded(false);
@ -79,7 +80,7 @@ export const FilterDropdownButton = <TData extends FilterableFieldsType>({
)); ));
const renderSearchResults = ( const renderSearchResults = (
filterSearchResults: SearchResultsType, filterSearchResults: SearchResultsType<TData>,
selectedFilter: FilterConfigType<TData>, selectedFilter: FilterConfigType<TData>,
selectedFilterOperand: FilterOperandType<TData>, selectedFilterOperand: FilterOperandType<TData>,
) => { ) => {
@ -155,7 +156,7 @@ export const FilterDropdownButton = <TData extends FilterableFieldsType>({
displayValue: event.target.value, displayValue: event.target.value,
icon: selectedFilter.icon, icon: selectedFilter.icon,
operand: selectedFilterOperand, operand: selectedFilterOperand,
}); } as SelectedFilterType<TData>);
} }
} }
}} }}
@ -172,7 +173,7 @@ export const FilterDropdownButton = <TData extends FilterableFieldsType>({
displayValue: humanReadableDate(date), displayValue: humanReadableDate(date),
icon: selectedFilter.icon, icon: selectedFilter.icon,
operand: selectedFilterOperand, operand: selectedFilterOperand,
}); } as SelectedFilterType<TData>);
}} }}
customInput={<></>} customInput={<></>}
customCalendarContainer={styled.div` customCalendarContainer={styled.div`
@ -200,7 +201,7 @@ export const FilterDropdownButton = <TData extends FilterableFieldsType>({
setIsUnfolded={setIsUnfolded} setIsUnfolded={setIsUnfolded}
resetState={resetState} resetState={resetState}
> >
{selectedFilter {selectedFilter && selectedFilterOperand
? isOperandSelectionUnfolded ? isOperandSelectionUnfolded
? renderOperandSelection ? renderOperandSelection
: renderValueSelection(selectedFilter, selectedFilterOperand) : renderValueSelection(selectedFilter, selectedFilterOperand)

View File

@ -1,12 +0,0 @@
import { User as GQLUser } from '../../../generated/graphql';
import { DeepPartial } from '../../utils/utils';
export type User = DeepPartial<GQLUser> & { id: string };
export type GraphqlQueryUser = User;
export type GraphqlMutationUser = User;
export const mapToUser = (user: GraphqlQueryUser): User => user;
export const mapToGqlUser = (user: User): GraphqlMutationUser => user;

View File

@ -1,18 +0,0 @@
import { WorkspaceMember as GQLWorkspaceMember } from '../../../generated/graphql';
import { DeepPartial } from '../../utils/utils';
export type WorkspaceMember = DeepPartial<GQLWorkspaceMember> & {
id: GQLWorkspaceMember['id'];
};
export type GraphqlQueryWorkspaceMember = WorkspaceMember;
export type GraphqlMutationWorkspaceMember = WorkspaceMember;
export const mapToWorkspaceMember = (
workspaceMember: GraphqlQueryWorkspaceMember,
): WorkspaceMember => workspaceMember;
export const mapToGqlWorkspaceMember = (
workspaceMember: WorkspaceMember,
): GraphqlMutationWorkspaceMember => workspaceMember;

View File

@ -1,9 +0,0 @@
import { gql } from '@apollo/client';
export const GET_CURRENT_USER = gql`
query getUsers {
findManyUser {
id
}
}
`;

View File

@ -1,37 +0,0 @@
import {
Company,
GraphqlQueryCompany,
} from '@/companies/interfaces/company.interface';
import {
GraphqlQueryPerson,
Person,
} from '@/people/interfaces/person.interface';
import { GraphqlQueryUser, User } from '@/users/interfaces/user.interface';
import {
CompanyWhereInput as Companies_Bool_Exp,
PersonWhereInput as People_Bool_Exp,
UserWhereInput as Users_Bool_Exp,
} from '~/generated/graphql';
export type AnyEntity = {
id: string;
__typename?: string;
} & Record<string, any>;
export type UnknownType = void;
export type GqlType<T> = T extends Company
? GraphqlQueryCompany
: T extends Person
? GraphqlQueryPerson
: T extends User
? GraphqlQueryUser
: never;
export type BoolExpType<T> = T extends Company
? Companies_Bool_Exp
: T extends Person
? People_Bool_Exp
: T extends User
? Users_Bool_Exp
: never;

View File

@ -13,9 +13,3 @@ export const getLogoUrlFromDomainName = (domainName?: string): string => {
export const browserPrefersDarkMode = (): boolean => { export const browserPrefersDarkMode = (): boolean => {
return window.matchMedia('(prefers-color-scheme: dark)').matches; return window.matchMedia('(prefers-color-scheme: dark)').matches;
}; };
export type DeepPartial<T> = T extends object
? {
[P in keyof T]?: DeepPartial<T[P]>;
}
: T;

View File

@ -1,21 +0,0 @@
import { Workspace as GQLWorkspace } from '../../../generated/graphql';
import { DeepPartial } from '../../utils/utils';
export type Workspace = DeepPartial<GQLWorkspace> & { id: GQLWorkspace['id'] };
export type GraphqlQueryWorkspace = Workspace;
export type GraphqlMutationWorkspace = Workspace;
export const mapToWorkspace = (
workspace: GraphqlQueryWorkspace,
): Workspace => ({
id: workspace.id,
domainName: workspace.domainName,
displayName: workspace.displayName,
logo: workspace.logo,
});
export const mapToGqlWorkspace = (
workspace: Workspace,
): GraphqlMutationWorkspace => workspace;

View File

@ -3,14 +3,9 @@ import { useTheme } from '@emotion/react';
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
import {
Company,
mapToCompany,
} from '@/companies/interfaces/company.interface';
import { import {
CompaniesSelectedSortType, CompaniesSelectedSortType,
defaultOrderBy, defaultOrderBy,
insertCompany,
useCompaniesQuery, useCompaniesQuery,
} from '@/companies/services'; } from '@/companies/services';
import { import {
@ -23,8 +18,13 @@ import { EntityTable } from '@/ui/components/table/EntityTable';
import { IconBuildingSkyscraper } from '@/ui/icons/index'; import { IconBuildingSkyscraper } from '@/ui/icons/index';
import { IconList } from '@/ui/icons/index'; import { IconList } from '@/ui/icons/index';
import { WithTopBarContainer } from '@/ui/layout/containers/WithTopBarContainer'; import { WithTopBarContainer } from '@/ui/layout/containers/WithTopBarContainer';
import { BoolExpType } from '@/utils/interfaces/generic.interface'; import {
import { CompanyOrderByWithRelationInput as Companies_Order_By } from '~/generated/graphql'; CompanyOrderByWithRelationInput as Companies_Order_By,
CompanyWhereInput,
GetCompaniesQuery,
InsertCompanyMutationVariables,
useInsertCompanyMutation,
} from '~/generated/graphql';
import { TableActionBarButtonCreateCommentThreadCompany } from './table/TableActionBarButtonCreateCommentThreadCompany'; import { TableActionBarButtonCreateCommentThreadCompany } from './table/TableActionBarButtonCreateCommentThreadCompany';
import { TableActionBarButtonDeleteCompanies } from './table/TableActionBarButtonDeleteCompanies'; import { TableActionBarButtonDeleteCompanies } from './table/TableActionBarButtonDeleteCompanies';
@ -38,15 +38,16 @@ const StyledCompaniesContainer = styled.div`
`; `;
export function Companies() { export function Companies() {
const [insertCompany] = useInsertCompanyMutation();
const [orderBy, setOrderBy] = useState<Companies_Order_By[]>(defaultOrderBy); const [orderBy, setOrderBy] = useState<Companies_Order_By[]>(defaultOrderBy);
const [where, setWhere] = useState<BoolExpType<Company>>({}); const [where, setWhere] = useState<CompanyWhereInput>({});
const updateSorts = useCallback((sorts: Array<CompaniesSelectedSortType>) => { const updateSorts = useCallback((sorts: Array<CompaniesSelectedSortType>) => {
setOrderBy(sorts.length ? reduceSortsToOrderBy(sorts) : defaultOrderBy); setOrderBy(sorts.length ? reduceSortsToOrderBy(sorts) : defaultOrderBy);
}, []); }, []);
const updateFilters = useCallback( const updateFilters = useCallback(
(filters: Array<SelectedFilterType<Company>>) => { (filters: Array<SelectedFilterType<GetCompaniesQuery['companies'][0]>>) => {
setWhere(reduceFiltersToWhere(filters)); setWhere(reduceFiltersToWhere(filters));
}, },
[], [],
@ -54,21 +55,19 @@ export function Companies() {
const { data } = useCompaniesQuery(orderBy, where); const { data } = useCompaniesQuery(orderBy, where);
const companies = data?.companies.map(mapToCompany) ?? []; const companies = data?.companies ?? [];
async function handleAddButtonClick() { async function handleAddButtonClick() {
const newCompany: Company = { const newCompany: InsertCompanyMutationVariables = {
id: uuidv4(), id: uuidv4(),
name: '', name: '',
domainName: '', domainName: '',
employees: null, employees: null,
address: '', address: '',
createdAt: new Date().toISOString(), createdAt: new Date().toISOString(),
accountOwner: null,
__typename: 'Company',
}; };
await insertCompany(newCompany); await insertCompany({ variables: newCompany });
} }
const companiesColumns = useCompaniesColumns(); const companiesColumns = useCompaniesColumns();

View File

@ -2,7 +2,6 @@ import { useMemo } from 'react';
import { createColumnHelper } from '@tanstack/react-table'; import { createColumnHelper } from '@tanstack/react-table';
import { CompanyEditableNameChipCell } from '@/companies/components/CompanyEditableNameCell'; import { CompanyEditableNameChipCell } from '@/companies/components/CompanyEditableNameCell';
import { updateCompany } from '@/companies/services';
import { import {
PersonChip, PersonChip,
PersonChipPropsType, PersonChipPropsType,
@ -22,12 +21,16 @@ import {
IconUsers, IconUsers,
} from '@/ui/icons/index'; } from '@/ui/icons/index';
import { getCheckBoxColumn } from '@/ui/tables/utils/getCheckBoxColumn'; import { getCheckBoxColumn } from '@/ui/tables/utils/getCheckBoxColumn';
import { mapToUser, User } from '@/users/interfaces/user.interface'; import {
import { GetCompaniesQueryHookResult, QueryMode } from '~/generated/graphql'; GetCompaniesQuery,
QueryMode,
useUpdateCompanyMutation,
} from '~/generated/graphql';
const columnHelper = createColumnHelper<GetCompaniesQueryHookResult>(); const columnHelper = createColumnHelper<GetCompaniesQuery['companies'][0]>();
export const useCompaniesColumns = () => { export const useCompaniesColumns = () => {
const [updateCompany] = useUpdateCompanyMutation();
return useMemo(() => { return useMemo(() => {
return [ return [
getCheckBoxColumn(), getCheckBoxColumn(),
@ -54,7 +57,12 @@ export const useCompaniesColumns = () => {
changeHandler={(value) => { changeHandler={(value) => {
const company = { ...props.row.original }; const company = { ...props.row.original };
company.domainName = value; company.domainName = value;
updateCompany(company); updateCompany({
variables: {
...company,
accountOwnerId: company.accountOwner?.id,
},
});
}} }}
/> />
), ),
@ -71,13 +79,13 @@ export const useCompaniesColumns = () => {
changeHandler={(value) => { changeHandler={(value) => {
const company = { ...props.row.original }; const company = { ...props.row.original };
if (value === '') { updateCompany({
company.employees = null; variables: {
updateCompany(company); ...company,
} else if (!Number.isNaN(Number(value))) { employees: value === '' ? null : Number(value),
company.employees = Number(value); accountOwnerId: company.accountOwner?.id,
updateCompany(company); },
} });
}} }}
/> />
), ),
@ -94,7 +102,12 @@ export const useCompaniesColumns = () => {
changeHandler={(value) => { changeHandler={(value) => {
const company = { ...props.row.original }; const company = { ...props.row.original };
company.address = value; company.address = value;
updateCompany(company); updateCompany({
variables: {
...company,
accountOwnerId: company.accountOwner?.id,
},
});
}} }}
/> />
), ),
@ -117,7 +130,12 @@ export const useCompaniesColumns = () => {
changeHandler={(value: Date) => { changeHandler={(value: Date) => {
const company = { ...props.row.original }; const company = { ...props.row.original };
company.createdAt = value.toISOString(); company.createdAt = value.toISOString();
updateCompany(company); updateCompany({
variables: {
...company,
accountOwnerId: company.accountOwner?.id,
},
});
}} }}
/> />
), ),
@ -131,21 +149,24 @@ export const useCompaniesColumns = () => {
/> />
), ),
cell: (props) => ( cell: (props) => (
<EditableRelation<User, PersonChipPropsType> <EditableRelation<any, PersonChipPropsType>
relation={props.row.original.accountOwner} relation={props.row.original.accountOwner}
searchPlaceholder="Account Owner" searchPlaceholder="Account Owner"
ChipComponent={PersonChip} ChipComponent={PersonChip}
chipComponentPropsMapper={( chipComponentPropsMapper={(
accountOwner: User, accountOwner: any,
): PersonChipPropsType => { ): PersonChipPropsType => {
return { return {
name: accountOwner.displayName || '', name: accountOwner.displayName || '',
}; };
}} }}
onChange={(relation: User) => { onChange={(relation: any) => {
const company = { ...props.row.original }; updateCompany({
company.accountOwnerId = relation.id; variables: {
updateCompany(company); ...props.row.original,
accountOwnerId: relation.id,
},
});
}} }}
searchConfig={ searchConfig={
{ {
@ -156,15 +177,15 @@ export const useCompaniesColumns = () => {
mode: QueryMode.Insensitive, mode: QueryMode.Insensitive,
}, },
}), }),
resultMapper: (accountOwner) => ({ resultMapper: (accountOwner: any) => ({
render: (accountOwner) => accountOwner.displayName, render: (accountOwner: any) => accountOwner.displayName,
value: mapToUser(accountOwner), value: accountOwner,
}), }),
} satisfies SearchConfigType<User> } satisfies SearchConfigType
} }
/> />
), ),
}), }),
]; ];
}, []); }, [updateCompany]);
}; };

View File

@ -1,4 +1,3 @@
import { Company } from '@/companies/interfaces/company.interface';
import { FilterConfigType } from '@/filters-and-sorts/interfaces/filters/interface'; import { FilterConfigType } from '@/filters-and-sorts/interfaces/filters/interface';
import { SEARCH_USER_QUERY } from '@/search/services/search'; import { SEARCH_USER_QUERY } from '@/search/services/search';
import { import {
@ -9,8 +8,7 @@ import {
IconUser, IconUser,
IconUsers, IconUsers,
} from '@/ui/icons/index'; } from '@/ui/icons/index';
import { mapToUser, User } from '@/users/interfaces/user.interface'; import { QueryMode, User } from '~/generated/graphql';
import { QueryMode } from '~/generated/graphql';
export const nameFilter = { export const nameFilter = {
key: 'name', key: 'name',
@ -21,14 +19,14 @@ export const nameFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
name: { contains: `%${searchString}%`, mode: QueryMode.Insensitive }, name: { contains: `%${searchString}%`, mode: QueryMode.Insensitive },
}), }),
}, },
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
name: { name: {
@ -40,7 +38,7 @@ export const nameFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, string>; } satisfies FilterConfigType<string>;
export const employeesFilter = { export const employeesFilter = {
key: 'employees', key: 'employees',
@ -51,7 +49,7 @@ export const employeesFilter = {
{ {
label: 'Greater than', label: 'Greater than',
id: 'greater_than', id: 'greater_than',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
employees: { employees: {
gte: isNaN(Number(searchString)) ? undefined : Number(searchString), gte: isNaN(Number(searchString)) ? undefined : Number(searchString),
}, },
@ -60,14 +58,14 @@ export const employeesFilter = {
{ {
label: 'Less than', label: 'Less than',
id: 'less_than', id: 'less_than',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
employees: { employees: {
lte: isNaN(Number(searchString)) ? undefined : Number(searchString), lte: isNaN(Number(searchString)) ? undefined : Number(searchString),
}, },
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, string>; } satisfies FilterConfigType<string>;
export const urlFilter = { export const urlFilter = {
key: 'domainName', key: 'domainName',
@ -78,7 +76,7 @@ export const urlFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
domainName: { domainName: {
contains: `%${searchString}%`, contains: `%${searchString}%`,
mode: QueryMode.Insensitive, mode: QueryMode.Insensitive,
@ -88,7 +86,7 @@ export const urlFilter = {
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
domainName: { domainName: {
@ -100,7 +98,7 @@ export const urlFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, string>; } satisfies FilterConfigType<string>;
export const addressFilter = { export const addressFilter = {
key: 'address', key: 'address',
@ -111,14 +109,14 @@ export const addressFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
address: { contains: `%${searchString}%`, mode: QueryMode.Insensitive }, address: { contains: `%${searchString}%`, mode: QueryMode.Insensitive },
}), }),
}, },
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
address: { address: {
@ -130,7 +128,7 @@ export const addressFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, string>; } satisfies FilterConfigType<string>;
export const ccreatedAtFilter = { export const ccreatedAtFilter = {
key: 'createdAt', key: 'createdAt',
@ -141,7 +139,7 @@ export const ccreatedAtFilter = {
{ {
label: 'Greater than', label: 'Greater than',
id: 'greater_than', id: 'greater_than',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
createdAt: { createdAt: {
gte: searchString, gte: searchString,
}, },
@ -150,14 +148,14 @@ export const ccreatedAtFilter = {
{ {
label: 'Less than', label: 'Less than',
id: 'less_than', id: 'less_than',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
createdAt: { createdAt: {
lte: searchString, lte: searchString,
}, },
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, string>; } satisfies FilterConfigType<string>;
export const accountOwnerFilter = { export const accountOwnerFilter = {
key: 'accountOwner', key: 'accountOwner',
@ -172,24 +170,24 @@ export const accountOwnerFilter = {
mode: QueryMode.Insensitive, mode: QueryMode.Insensitive,
}, },
}), }),
resultMapper: (data) => ({ resultMapper: (data: any) => ({
value: mapToUser(data), value: data,
render: (owner) => owner.displayName, render: (owner: any) => owner.displayName,
}), }),
}, },
selectedValueRender: (owner) => owner.displayName || '', selectedValueRender: (owner: any) => owner.displayName || '',
operands: [ operands: [
{ {
label: 'Is', label: 'Is',
id: 'is', id: 'is',
whereTemplate: (owner) => ({ whereTemplate: (owner: any) => ({
accountOwner: { is: { displayName: { equals: owner.displayName } } }, accountOwner: { is: { displayName: { equals: owner.displayName } } },
}), }),
}, },
{ {
label: 'Is not', label: 'Is not',
id: 'is_not', id: 'is_not',
whereTemplate: (owner) => ({ whereTemplate: (owner: any) => ({
NOT: [ NOT: [
{ {
accountOwner: { accountOwner: {
@ -200,7 +198,7 @@ export const accountOwnerFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, User>; } satisfies FilterConfigType<User>;
export const availableFilters = [ export const availableFilters = [
nameFilter, nameFilter,

View File

@ -1,4 +1,5 @@
import { useCallback, useState } from 'react'; import { useCallback, useState } from 'react';
import { getOperationName } from '@apollo/client/utilities';
import { useTheme } from '@emotion/react'; import { useTheme } from '@emotion/react';
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { v4 as uuidv4 } from 'uuid'; import { v4 as uuidv4 } from 'uuid';
@ -8,10 +9,9 @@ import {
reduceSortsToOrderBy, reduceSortsToOrderBy,
} from '@/filters-and-sorts/helpers'; } from '@/filters-and-sorts/helpers';
import { SelectedFilterType } from '@/filters-and-sorts/interfaces/filters/interface'; import { SelectedFilterType } from '@/filters-and-sorts/interfaces/filters/interface';
import { Person } from '@/people/interfaces/person.interface';
import { import {
defaultOrderBy, defaultOrderBy,
insertPerson, GET_PEOPLE,
PeopleSelectedSortType, PeopleSelectedSortType,
usePeopleQuery, usePeopleQuery,
} from '@/people/services'; } from '@/people/services';
@ -19,7 +19,11 @@ import { EntityTableActionBar } from '@/ui/components/table/action-bar/EntityTab
import { EntityTable } from '@/ui/components/table/EntityTable'; import { EntityTable } from '@/ui/components/table/EntityTable';
import { IconList, IconUser } from '@/ui/icons/index'; import { IconList, IconUser } from '@/ui/icons/index';
import { WithTopBarContainer } from '@/ui/layout/containers/WithTopBarContainer'; import { WithTopBarContainer } from '@/ui/layout/containers/WithTopBarContainer';
import { BoolExpType } from '@/utils/interfaces/generic.interface'; import {
GetPeopleQuery,
PersonWhereInput,
useInsertPersonMutation,
} from '~/generated/graphql';
import { TableActionBarButtonCreateCommentThreadPeople } from './table/TableActionBarButtonCreateCommentThreadPeople'; import { TableActionBarButtonCreateCommentThreadPeople } from './table/TableActionBarButtonCreateCommentThreadPeople';
import { TableActionBarButtonDeletePeople } from './table/TableActionBarButtonDeletePeople'; import { TableActionBarButtonDeletePeople } from './table/TableActionBarButtonDeletePeople';
@ -35,37 +39,38 @@ const StyledPeopleContainer = styled.div`
export function People() { export function People() {
const [orderBy, setOrderBy] = useState(defaultOrderBy); const [orderBy, setOrderBy] = useState(defaultOrderBy);
const [where, setWhere] = useState<BoolExpType<Person>>({}); const [where, setWhere] = useState<PersonWhereInput>({});
const updateSorts = useCallback((sorts: Array<PeopleSelectedSortType>) => { const updateSorts = useCallback((sorts: Array<PeopleSelectedSortType>) => {
setOrderBy(sorts.length ? reduceSortsToOrderBy(sorts) : defaultOrderBy); setOrderBy(sorts.length ? reduceSortsToOrderBy(sorts) : defaultOrderBy);
}, []); }, []);
const updateFilters = useCallback( const updateFilters = useCallback(
(filters: Array<SelectedFilterType<Person>>) => { (filters: Array<SelectedFilterType<GetPeopleQuery['people'][0]>>) => {
setWhere(reduceFiltersToWhere(filters)); setWhere(reduceFiltersToWhere(filters));
}, },
[], [],
); );
const [insertPersonMutation] = useInsertPersonMutation();
const { data } = usePeopleQuery(orderBy, where); const { data } = usePeopleQuery(orderBy, where);
const people = data?.people ?? []; const people = data?.people ?? [];
async function handleAddButtonClick() { async function handleAddButtonClick() {
const newPerson = { await insertPersonMutation({
__typename: 'Person', variables: {
id: uuidv4(), id: uuidv4(),
firstname: '', firstname: '',
lastname: '', lastname: '',
email: '', email: '',
phone: '', phone: '',
company: null, createdAt: new Date().toISOString(),
createdAt: new Date().toISOString(), city: '',
city: '', },
} as const; refetchQueries: [getOperationName(GET_PEOPLE) ?? ''],
});
await insertPerson(newPerson);
} }
const peopleColumns = usePeopleColumns(); const peopleColumns = usePeopleColumns();

View File

@ -3,7 +3,6 @@ import type { Meta } from '@storybook/react';
import { userEvent, within } from '@storybook/testing-library'; import { userEvent, within } from '@storybook/testing-library';
import { graphql } from 'msw'; import { graphql } from 'msw';
import { GraphqlQueryCompany } from '@/companies/interfaces/company.interface';
import { graphqlMocks } from '~/testing/graphqlMocks'; import { graphqlMocks } from '~/testing/graphqlMocks';
import { fetchOneFromData } from '~/testing/mock-data'; import { fetchOneFromData } from '~/testing/mock-data';
import { mockedPeopleData } from '~/testing/mock-data/people'; import { mockedPeopleData } from '~/testing/mock-data/people';
@ -141,7 +140,7 @@ export const EditRelation: Story = {
name: 'Airbnb', name: 'Airbnb',
domainName: 'airbnb.com', domainName: 'airbnb.com',
__typename: 'Company', __typename: 'Company',
} satisfies GraphqlQueryCompany, },
}, },
}, },
}), }),
@ -196,7 +195,7 @@ export const SelectRelationWithKeys: Story = {
name: 'Aircall', name: 'Aircall',
domainName: 'aircall.io', domainName: 'aircall.io',
__typename: 'Company', __typename: 'Company',
} satisfies GraphqlQueryCompany, },
}, },
}, },
}), }),

View File

@ -4,10 +4,7 @@ describe('PeopleFilter', () => {
it(`should render the filter ${companyFilter.key} which relation search`, () => { it(`should render the filter ${companyFilter.key} which relation search`, () => {
expect( expect(
companyFilter.operands[0].whereTemplate({ companyFilter.operands[0].whereTemplate({
id: 'test-id',
name: 'test-name', name: 'test-name',
domainName: 'test-domain-name',
__typename: 'Company',
}), }),
).toMatchSnapshot(); ).toMatchSnapshot();
}); });

View File

@ -3,7 +3,6 @@ import { createColumnHelper } from '@tanstack/react-table';
import { EditablePeopleFullName } from '@/people/components/EditablePeopleFullName'; import { EditablePeopleFullName } from '@/people/components/EditablePeopleFullName';
import { PeopleCompanyCell } from '@/people/components/PeopleCompanyCell'; import { PeopleCompanyCell } from '@/people/components/PeopleCompanyCell';
import { updatePerson } from '@/people/services';
import { EditableDate } from '@/ui/components/editable-cell/types/EditableDate'; import { EditableDate } from '@/ui/components/editable-cell/types/EditableDate';
import { EditablePhone } from '@/ui/components/editable-cell/types/EditablePhone'; import { EditablePhone } from '@/ui/components/editable-cell/types/EditablePhone';
import { EditableText } from '@/ui/components/editable-cell/types/EditableText'; import { EditableText } from '@/ui/components/editable-cell/types/EditableText';
@ -17,12 +16,13 @@ import {
IconUser, IconUser,
} from '@/ui/icons/index'; } from '@/ui/icons/index';
import { getCheckBoxColumn } from '@/ui/tables/utils/getCheckBoxColumn'; import { getCheckBoxColumn } from '@/ui/tables/utils/getCheckBoxColumn';
import { GetPeopleQuery, useUpdatePeopleMutation } from '~/generated/graphql';
import { GetPeopleQueryHookResult } from '../../generated/graphql'; const columnHelper = createColumnHelper<GetPeopleQuery['people'][0]>();
const columnHelper = createColumnHelper<GetPeopleQueryHookResult>();
export const usePeopleColumns = () => { export const usePeopleColumns = () => {
const [updatePerson] = useUpdatePeopleMutation();
return useMemo(() => { return useMemo(() => {
return [ return [
getCheckBoxColumn(), getCheckBoxColumn(),
@ -36,9 +36,14 @@ export const usePeopleColumns = () => {
person={props.row.original} person={props.row.original}
onChange={async (firstName: string, lastName: string) => { onChange={async (firstName: string, lastName: string) => {
const person = { ...props.row.original }; const person = { ...props.row.original };
person.firstname = firstName; await updatePerson({
person.lastname = lastName; variables: {
await updatePerson(person); ...person,
firstname: firstName,
lastname: lastName,
companyId: person.company?.id,
},
});
}} }}
/> />
</> </>
@ -53,10 +58,15 @@ export const usePeopleColumns = () => {
<EditableText <EditableText
placeholder="Email" placeholder="Email"
content={props.row.original.email || ''} content={props.row.original.email || ''}
changeHandler={(value: string) => { changeHandler={async (value: string) => {
const person = props.row.original; const person = props.row.original;
person.email = value; await updatePerson({
updatePerson(person); variables: {
...person,
email: value,
companyId: person.company?.id,
},
});
}} }}
/> />
), ),
@ -80,10 +90,15 @@ export const usePeopleColumns = () => {
<EditablePhone <EditablePhone
placeholder="Phone" placeholder="Phone"
value={props.row.original.phone || ''} value={props.row.original.phone || ''}
changeHandler={(value: string) => { changeHandler={async (value: string) => {
const person = { ...props.row.original }; const person = { ...props.row.original };
person.phone = value; await updatePerson({
updatePerson(person); variables: {
...person,
phone: value,
companyId: person.company?.id,
},
});
}} }}
/> />
), ),
@ -103,10 +118,15 @@ export const usePeopleColumns = () => {
? new Date(props.row.original.createdAt) ? new Date(props.row.original.createdAt)
: new Date() : new Date()
} }
changeHandler={(value: Date) => { changeHandler={async (value: Date) => {
const person = { ...props.row.original }; const person = { ...props.row.original };
person.createdAt = value.toISOString(); await updatePerson({
updatePerson(person); variables: {
...person,
createdAt: value.toISOString(),
companyId: person.company?.id,
},
});
}} }}
/> />
), ),
@ -121,14 +141,19 @@ export const usePeopleColumns = () => {
editModeHorizontalAlign="right" editModeHorizontalAlign="right"
placeholder="City" placeholder="City"
content={props.row.original.city || ''} content={props.row.original.city || ''}
changeHandler={(value: string) => { changeHandler={async (value: string) => {
const person = { ...props.row.original }; const person = { ...props.row.original };
person.city = value; await updatePerson({
updatePerson(person); variables: {
...person,
city: value,
companyId: person.company?.id,
},
});
}} }}
/> />
), ),
}), }),
]; ];
}, []); }, [updatePerson]);
}; };

View File

@ -1,9 +1,4 @@
import {
Company,
mapToCompany,
} from '@/companies/interfaces/company.interface';
import { FilterConfigType } from '@/filters-and-sorts/interfaces/filters/interface'; import { FilterConfigType } from '@/filters-and-sorts/interfaces/filters/interface';
import { Person } from '@/people/interfaces/person.interface';
import { SEARCH_COMPANY_QUERY } from '@/search/services/search'; import { SEARCH_COMPANY_QUERY } from '@/search/services/search';
import { import {
IconBuildingSkyscraper, IconBuildingSkyscraper,
@ -13,7 +8,7 @@ import {
IconPhone, IconPhone,
IconUser, IconUser,
} from '@/ui/icons/index'; } from '@/ui/icons/index';
import { QueryMode } from '~/generated/graphql'; import { Company, QueryMode } from '~/generated/graphql';
export const fullnameFilter = { export const fullnameFilter = {
key: 'fullname', key: 'fullname',
@ -24,7 +19,7 @@ export const fullnameFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
OR: [ OR: [
{ {
firstname: { firstname: {
@ -44,7 +39,7 @@ export const fullnameFilter = {
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
AND: [ AND: [
@ -66,7 +61,7 @@ export const fullnameFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Person, string>; } satisfies FilterConfigType<string>;
export const emailFilter = { export const emailFilter = {
key: 'email', key: 'email',
@ -77,14 +72,14 @@ export const emailFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
email: { contains: `%${searchString}%`, mode: QueryMode.Insensitive }, email: { contains: `%${searchString}%`, mode: QueryMode.Insensitive },
}), }),
}, },
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
email: { email: {
@ -96,7 +91,7 @@ export const emailFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Person, string>; } satisfies FilterConfigType<string>;
export const companyFilter = { export const companyFilter = {
key: 'company_name', key: 'company_name',
@ -109,8 +104,8 @@ export const companyFilter = {
name: { contains: `%${searchString}%`, mode: QueryMode.Insensitive }, name: { contains: `%${searchString}%`, mode: QueryMode.Insensitive },
}), }),
resultMapper: (data) => ({ resultMapper: (data) => ({
value: mapToCompany(data), value: data,
render: (company) => company.name, render: (company: { name: string }) => company.name,
}), }),
}, },
selectedValueRender: (company) => company.name || '', selectedValueRender: (company) => company.name || '',
@ -118,19 +113,19 @@ export const companyFilter = {
{ {
label: 'Is', label: 'Is',
id: 'is', id: 'is',
whereTemplate: (company) => ({ whereTemplate: (company: { name: string }) => ({
company: { is: { name: { equals: company.name } } }, company: { is: { name: { equals: company.name } } },
}), }),
}, },
{ {
label: 'Is not', label: 'Is not',
id: 'is_not', id: 'is_not',
whereTemplate: (company) => ({ whereTemplate: (company: { name: string }) => ({
NOT: [{ company: { is: { name: { equals: company.name } } } }], NOT: [{ company: { is: { name: { equals: company.name } } } }],
}), }),
}, },
], ],
} satisfies FilterConfigType<Person, Company>; } satisfies FilterConfigType<Company>;
export const phoneFilter = { export const phoneFilter = {
key: 'phone', key: 'phone',
@ -141,14 +136,14 @@ export const phoneFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
phone: { contains: `%${searchString}%`, mode: QueryMode.Insensitive }, phone: { contains: `%${searchString}%`, mode: QueryMode.Insensitive },
}), }),
}, },
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
phone: { phone: {
@ -160,7 +155,7 @@ export const phoneFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Person, string>; } satisfies FilterConfigType<string>;
export const createdAtFilter = { export const createdAtFilter = {
key: 'createdAt', key: 'createdAt',
@ -171,7 +166,7 @@ export const createdAtFilter = {
{ {
label: 'Greater than', label: 'Greater than',
id: 'greater_than', id: 'greater_than',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
createdAt: { createdAt: {
gte: searchString, gte: searchString,
}, },
@ -180,14 +175,14 @@ export const createdAtFilter = {
{ {
label: 'Less than', label: 'Less than',
id: 'less_than', id: 'less_than',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
createdAt: { createdAt: {
lte: searchString, lte: searchString,
}, },
}), }),
}, },
], ],
} satisfies FilterConfigType<Company, string>; } satisfies FilterConfigType<string>;
export const cityFilter = { export const cityFilter = {
key: 'city', key: 'city',
@ -198,14 +193,14 @@ export const cityFilter = {
{ {
label: 'Contains', label: 'Contains',
id: 'like', id: 'like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
city: { contains: `%${searchString}%`, mode: QueryMode.Insensitive }, city: { contains: `%${searchString}%`, mode: QueryMode.Insensitive },
}), }),
}, },
{ {
label: 'Does not contain', label: 'Does not contain',
id: 'not_like', id: 'not_like',
whereTemplate: (searchString) => ({ whereTemplate: (searchString: string) => ({
NOT: [ NOT: [
{ {
city: { city: {
@ -217,7 +212,7 @@ export const cityFilter = {
}), }),
}, },
], ],
} satisfies FilterConfigType<Person, string>; } satisfies FilterConfigType<string>;
export const availableFilters = [ export const availableFilters = [
fullnameFilter, fullnameFilter,
@ -226,4 +221,4 @@ export const availableFilters = [
phoneFilter, phoneFilter,
createdAtFilter, createdAtFilter,
cityFilter, cityFilter,
] satisfies FilterConfigType<Person>[]; ];

View File

@ -1,8 +1,20 @@
import { getOperationName } from '@apollo/client/utilities';
import { graphql } from 'msw'; import { graphql } from 'msw';
import { GraphqlQueryCompany } from '@/companies/interfaces/company.interface'; import { GET_COMPANIES } from '@/companies/services';
import { GraphqlQueryPerson } from '@/people/interfaces/person.interface'; import { GET_PEOPLE, UPDATE_PERSON } from '@/people/services';
import { GraphqlQueryUser } from '@/users/interfaces/user.interface'; import {
SEARCH_COMPANY_QUERY,
SEARCH_USER_QUERY,
} from '@/search/services/search';
import { GET_CURRENT_USER } from '@/users/services';
import {
GetCompaniesQuery,
GetPeopleQuery,
GetUsersQuery,
SearchCompanyQuery,
SearchUserQuery,
} from '~/generated/graphql';
import { mockedCompaniesData } from './mock-data/companies'; import { mockedCompaniesData } from './mock-data/companies';
import { mockedPeopleData } from './mock-data/people'; import { mockedPeopleData } from './mock-data/people';
@ -10,8 +22,10 @@ import { mockedUsersData } from './mock-data/users';
import { filterAndSortData, updateOneFromData } from './mock-data'; import { filterAndSortData, updateOneFromData } from './mock-data';
export const graphqlMocks = [ export const graphqlMocks = [
graphql.query('GetCompanies', (req, res, ctx) => { graphql.query(getOperationName(GET_COMPANIES) ?? '', (req, res, ctx) => {
const returnedMockedData = filterAndSortData<GraphqlQueryCompany>( const returnedMockedData = filterAndSortData<
GetCompaniesQuery['companies'][0]
>(
mockedCompaniesData, mockedCompaniesData,
req.variables.where, req.variables.where,
req.variables.orderBy, req.variables.orderBy,
@ -23,21 +37,28 @@ export const graphqlMocks = [
}), }),
); );
}), }),
graphql.query('SearchCompany', (req, res, ctx) => { graphql.query(
const returnedMockedData = filterAndSortData<GraphqlQueryCompany>( getOperationName(SEARCH_COMPANY_QUERY) ?? '',
mockedCompaniesData, (req, res, ctx) => {
req.variables.where, const returnedMockedData = filterAndSortData<
req.variables.orderBy, SearchCompanyQuery['searchResults'][0]
req.variables.limit, >(
); mockedCompaniesData,
return res( req.variables.where,
ctx.data({ req.variables.orderBy,
searchResults: returnedMockedData, req.variables.limit,
}), );
); return res(
}), ctx.data({
graphql.query('SearchUser', (req, res, ctx) => { searchResults: returnedMockedData,
const returnedMockedData = filterAndSortData<GraphqlQueryUser>( }),
);
},
),
graphql.query(getOperationName(SEARCH_USER_QUERY) ?? '', (req, res, ctx) => {
const returnedMockedData = filterAndSortData<
SearchUserQuery['searchResults'][0]
>(
mockedUsersData, mockedUsersData,
req.variables.where, req.variables.where,
req.variables.orderBy, req.variables.orderBy,
@ -49,7 +70,7 @@ export const graphqlMocks = [
}), }),
); );
}), }),
graphql.query('GetCurrentUser', (req, res, ctx) => { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', (req, res, ctx) => {
const customWhere = { const customWhere = {
...req.variables.where, ...req.variables.where,
id: { id: {
@ -57,20 +78,17 @@ export const graphqlMocks = [
}, },
}; };
const returnedMockedData = filterAndSortData<GraphqlQueryUser>( const returnedMockedData = filterAndSortData<
mockedUsersData, GetUsersQuery['findManyUser'][0]
customWhere, >(mockedUsersData, customWhere, req.variables.orderBy, req.variables.limit);
req.variables.orderBy,
req.variables.limit,
);
return res( return res(
ctx.data({ ctx.data({
users: returnedMockedData, users: returnedMockedData,
}), }),
); );
}), }),
graphql.query('GetPeople', (req, res, ctx) => { graphql.query(getOperationName(GET_PEOPLE) ?? '', (req, res, ctx) => {
const returnedMockedData = filterAndSortData<GraphqlQueryPerson>( const returnedMockedData = filterAndSortData<GetPeopleQuery['people'][0]>(
mockedPeopleData, mockedPeopleData,
req.variables.where, req.variables.where,
req.variables.orderBy, req.variables.orderBy,
@ -82,7 +100,7 @@ export const graphqlMocks = [
}), }),
); );
}), }),
graphql.mutation('UpdatePeople', (req, res, ctx) => { graphql.mutation(getOperationName(UPDATE_PERSON) ?? '', (req, res, ctx) => {
return res( return res(
ctx.data({ ctx.data({
updateOnePerson: updateOneFromData( updateOnePerson: updateOneFromData(

View File

@ -1,6 +1,28 @@
import { CommentableType, CommentThread } from '~/generated/graphql'; import {
CommentableType,
CommentThread,
CommentThreadTarget,
} from '~/generated/graphql';
export const mockedCommentThreads: Array<CommentThread> = [ type MockedCommentThread = Pick<
CommentThread,
'id' | 'createdAt' | 'updatedAt' | '__typename'
> & {
commentThreadTargets: Array<
Pick<
CommentThreadTarget,
| 'id'
| '__typename'
| 'createdAt'
| 'updatedAt'
| 'commentableType'
| 'commentableId'
| 'commentThreadId'
> & { commentThread: Pick<CommentThread, 'id' | 'createdAt' | 'updatedAt'> }
>;
};
export const mockedCommentThreads: Array<MockedCommentThread> = [
{ {
id: '89bb825c-171e-4bcc-9cf7-43448d6fb230', id: '89bb825c-171e-4bcc-9cf7-43448d6fb230',
createdAt: '2023-04-26T10:12:42.33625+00:00', createdAt: '2023-04-26T10:12:42.33625+00:00',

View File

@ -1,6 +1,23 @@
import { Company } from '../../generated/graphql'; import { Company, User } from '../../generated/graphql';
export const mockedCompaniesData = [ type MockedCompany = Pick<
Company,
| 'id'
| 'name'
| 'domainName'
| '__typename'
| 'createdAt'
| 'address'
| 'employees'
| '_commentCount'
> & {
accountOwner: Pick<
User,
'id' | 'email' | 'displayName' | '__typename'
> | null;
};
export const mockedCompaniesData: Array<MockedCompany> = [
{ {
id: '89bb825c-171e-4bcc-9cf7-43448d6fb278', id: '89bb825c-171e-4bcc-9cf7-43448d6fb278',
domainName: 'airbnb.com', domainName: 'airbnb.com',
@ -83,4 +100,4 @@ export const mockedCompaniesData = [
accountOwner: null, accountOwner: null,
__typename: 'Company', __typename: 'Company',
}, },
] as Array<Company>; ];

View File

@ -1,9 +1,5 @@
import { GraphQLVariables } from 'msw'; import { GraphQLVariables } from 'msw';
import { Company } from '@/companies/interfaces/company.interface';
import { Person } from '@/people/interfaces/person.interface';
import { User } from '@/users/interfaces/user.interface';
import { BoolExpType } from '@/utils/interfaces/generic.interface';
import { import {
CompanyOrderByWithRelationInput, CompanyOrderByWithRelationInput,
PersonOrderByWithRelationInput, PersonOrderByWithRelationInput,
@ -13,7 +9,7 @@ import {
function filterData<DataT>( function filterData<DataT>(
data: Array<DataT>, data: Array<DataT>,
where: BoolExpType<Company> | BoolExpType<Person>, where: Record<string, any>,
): Array<DataT> { ): Array<DataT> {
return data.filter((item) => { return data.filter((item) => {
// { firstname: {contains: '%string%' }} // { firstname: {contains: '%string%' }}
@ -76,7 +72,7 @@ function filterData<DataT>(
export function filterAndSortData<DataT>( export function filterAndSortData<DataT>(
data: Array<DataT>, data: Array<DataT>,
where?: BoolExpType<Company> | BoolExpType<Person> | BoolExpType<User>, where?: Record<string, any>,
orderBy?: Array< orderBy?: Array<
PersonOrderByWithRelationInput & PersonOrderByWithRelationInput &
CompanyOrderByWithRelationInput & CompanyOrderByWithRelationInput &

View File

@ -1,6 +1,21 @@
import { Person } from '../../modules/people/interfaces/person.interface'; import { Company, Person } from '~/generated/graphql';
export const mockedPeopleData = [ type MockedPerson = Pick<
Person,
| 'id'
| 'firstname'
| 'lastname'
| 'email'
| '__typename'
| 'phone'
| 'city'
| '_commentCount'
| 'createdAt'
> & {
company: Pick<Company, 'id' | 'name' | 'domainName' | '__typename'>;
};
export const mockedPeopleData: Array<MockedPerson> = [
{ {
id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6b', id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6b',
__typename: 'Person', __typename: 'Person',
@ -73,4 +88,4 @@ export const mockedPeopleData = [
city: 'Paris', city: 'Paris',
}, },
] satisfies Array<Person>; ];

View File

@ -1,6 +1,18 @@
import { GraphqlQueryUser } from '@/users/interfaces/user.interface'; import { User, Workspace, WorkspaceMember } from '~/generated/graphql';
export const mockedUsersData: Array<GraphqlQueryUser> = [ type MockedUser = Pick<
User,
'id' | 'email' | 'displayName' | 'avatarUrl' | '__typename'
> & {
workspaceMember: Pick<WorkspaceMember, 'id' | '__typename'> & {
workspace: Pick<
Workspace,
'id' | 'displayName' | 'domainName' | 'logo' | '__typename'
>;
};
};
export const mockedUsersData: Array<MockedUser> = [
{ {
id: '374fe3a5-df1e-4119-afe0-2a62a2ba481e', id: '374fe3a5-df1e-4119-afe0-2a62a2ba481e',
__typename: 'User', __typename: 'User',
@ -37,4 +49,4 @@ export const mockedUsersData: Array<GraphqlQueryUser> = [
}, },
}, },
}, },
] as GraphqlQueryUser[]; ];