-
Notifications
You must be signed in to change notification settings - Fork 111
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
update support ticket to support categories, project, and target options #6689
base: main
Are you sure you want to change the base?
Changes from 2 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -3,6 +3,15 @@ import type { SupportTicketResolvers } from './../../../__generated__/types'; | |
|
||
export const SupportTicket: SupportTicketResolvers = { | ||
comments: async (ticket, _args, { injector }) => { | ||
return { | ||
edges: [], | ||
pageInfo: { | ||
endCursor: '', | ||
hasNextPage: false, | ||
hasPreviousPage: false, | ||
startCursor: '', | ||
}, | ||
}; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. youll want to remove these stubs. This was just for offline development. |
||
const response = await injector.get(SupportManager).getTicketComments(ticket.id); | ||
|
||
return { | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,6 @@ | ||
import { Select, SelectContent, SelectItem, SelectTrigger } from '@/components/ui/select'; | ||
import { FragmentType, graphql, useFragment } from '@/gql'; | ||
import { SelectValue } from '@radix-ui/react-select'; | ||
import { Link, useRouter } from '@tanstack/react-router'; | ||
|
||
const ProjectSelector_OrganizationConnectionFragment = graphql(` | ||
|
@@ -21,7 +22,15 @@ export function ProjectSelector(props: { | |
currentOrganizationSlug: string; | ||
currentProjectSlug: string; | ||
organizations: FragmentType<typeof ProjectSelector_OrganizationConnectionFragment> | null; | ||
onValueChange?: Function; | ||
optional?: boolean; | ||
showOrganization?: boolean; | ||
}) { | ||
const optional = typeof props.optional !== 'undefined' ? props.optional : false; | ||
const showOrganization = | ||
typeof props.showOrganization !== 'undefined' ? props.showOrganization : true; | ||
const onValueChangeFunc = | ||
typeof props.onValueChange !== undefined ? props.onValueChange : undefined; | ||
const router = useRouter(); | ||
|
||
const organizations = useFragment( | ||
|
@@ -38,47 +47,74 @@ export function ProjectSelector(props: { | |
|
||
return ( | ||
<> | ||
{currentOrganization ? ( | ||
<Link | ||
to="/$organizationSlug" | ||
params={{ organizationSlug: props.currentOrganizationSlug }} | ||
className="max-w-[200px] shrink-0 truncate font-medium" | ||
> | ||
{currentOrganization.slug} | ||
</Link> | ||
{showOrganization ? ( | ||
currentOrganization ? ( | ||
<Link | ||
to="/$organizationSlug" | ||
params={{ organizationSlug: props.currentOrganizationSlug }} | ||
className="max-w-[200px] shrink-0 truncate font-medium" | ||
> | ||
{currentOrganization.slug} | ||
</Link> | ||
) : ( | ||
<div className="h-5 w-48 max-w-[200px] animate-pulse rounded-full bg-gray-800" /> | ||
) | ||
) : ( | ||
<div className="h-5 w-48 max-w-[200px] animate-pulse rounded-full bg-gray-800" /> | ||
'' | ||
)} | ||
{projects?.length && currentProject ? ( | ||
|
||
{(projects?.length && currentProject) || optional ? ( | ||
<> | ||
<div className="italic text-gray-500">/</div> | ||
{showOrganization ? <div className="italic text-gray-500">/</div> : <></>} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. can use |
||
<Select | ||
value={props.currentProjectSlug} | ||
onValueChange={id => { | ||
void router.navigate({ | ||
to: '/$organizationSlug/$projectSlug', | ||
params: { | ||
organizationSlug: props.currentOrganizationSlug, | ||
projectSlug: id, | ||
}, | ||
}); | ||
}} | ||
onValueChange={ | ||
onValueChangeFunc | ||
? id => { | ||
onValueChangeFunc(id); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. minor: to clean up the jsx, I'd move this defaulting logic to the top of the function alongside the other defaulting logic. And you shouldnt need to wrap the |
||
} | ||
: id => { | ||
void router.navigate({ | ||
to: '/$organizationSlug/$projectSlug', | ||
params: { | ||
organizationSlug: props.currentOrganizationSlug, | ||
projectSlug: id, | ||
}, | ||
}); | ||
} | ||
} | ||
> | ||
<SelectTrigger variant="default" data-cy="project-picker-trigger"> | ||
<div className="font-medium" data-cy="project-picker-current"> | ||
{currentProject.slug} | ||
{optional ? <SelectValue placeholder="Pick an option" /> : ''} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. minor: This can be shortened to
|
||
{currentProject && !optional ? currentProject.slug : ''} | ||
</div> | ||
</SelectTrigger> | ||
<SelectContent> | ||
{projects.map(project => ( | ||
{optional ? ( | ||
<SelectItem | ||
key={project.slug} | ||
value={project.slug} | ||
data-cy={`project-picker-option-${project.slug}`} | ||
key={'empty'} | ||
value={'empty'} | ||
data-cy={`project-picker-option-Unassigned`} | ||
> | ||
{project.slug} | ||
Unassigned | ||
</SelectItem> | ||
))} | ||
) : ( | ||
<></> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. null here also |
||
)} | ||
{projects ? ( | ||
projects.map(project => ( | ||
<SelectItem | ||
key={project.slug} | ||
value={project.slug} | ||
data-cy={`project-picker-option-${project.slug}`} | ||
> | ||
{project.slug} | ||
</SelectItem> | ||
)) | ||
) : ( | ||
<></> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. null here also |
||
)} | ||
</SelectContent> | ||
</Select> | ||
</> | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,10 @@ | ||
import { Select, SelectContent, SelectItem, SelectTrigger } from '@/components/ui/select'; | ||
import { | ||
Select, | ||
SelectContent, | ||
SelectItem, | ||
SelectTrigger, | ||
SelectValue, | ||
} from '@/components/ui/select'; | ||
import { FragmentType, graphql, useFragment } from '@/gql'; | ||
import { Link, useRouter } from '@tanstack/react-router'; | ||
|
||
|
@@ -27,10 +33,19 @@ export function TargetSelector(props: { | |
currentOrganizationSlug: string; | ||
currentProjectSlug: string; | ||
currentTargetSlug: string; | ||
optional?: boolean; | ||
showOrganization?: boolean; | ||
showProject?: boolean; | ||
onValueChange?: Function; | ||
organizations: FragmentType<typeof TargetSelector_OrganizationConnectionFragment> | null; | ||
}) { | ||
const router = useRouter(); | ||
|
||
const showOrganization = | ||
typeof props.showOrganization !== 'undefined' ? props.showOrganization : true; | ||
const showProject = typeof props.showProject !== 'undefined' ? props.showProject : true; | ||
const isOptional = typeof props.optional !== undefined ? props.optional : false; | ||
|
||
const organizations = useFragment( | ||
TargetSelector_OrganizationConnectionFragment, | ||
props.organizations, | ||
|
@@ -43,70 +58,102 @@ export function TargetSelector(props: { | |
const projects = currentOrganization?.projects.nodes; | ||
const currentProject = projects?.find(node => node.slug === props.currentProjectSlug); | ||
|
||
const targets = currentProject?.targets.nodes; | ||
const targets = currentProject?.targets?.nodes; | ||
const currentTarget = targets?.find(node => node.slug === props.currentTargetSlug); | ||
const onValueChangeFunc = | ||
typeof props.onValueChange !== undefined ? props.onValueChange : () => {}; | ||
|
||
return ( | ||
<> | ||
{currentOrganization ? ( | ||
<Link | ||
to="/$organizationSlug" | ||
params={{ | ||
organizationSlug: currentOrganization.slug, | ||
}} | ||
className="max-w-[200px] shrink-0 truncate font-medium" | ||
> | ||
{currentOrganization.slug} | ||
</Link> | ||
{showOrganization ? ( | ||
currentOrganization ? ( | ||
<Link | ||
to="/$organizationSlug" | ||
params={{ | ||
organizationSlug: currentOrganization.slug, | ||
}} | ||
className="max-w-[200px] shrink-0 truncate font-medium" | ||
> | ||
{currentOrganization.slug} | ||
</Link> | ||
) : ( | ||
<div className="h-5 w-48 max-w-[200px] animate-pulse rounded-full bg-gray-800" /> | ||
) | ||
) : ( | ||
<div className="h-5 w-48 max-w-[200px] animate-pulse rounded-full bg-gray-800" /> | ||
<></> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. null here also |
||
)} | ||
<div className="italic text-gray-500">/</div> | ||
{currentOrganization && currentProject ? ( | ||
<Link | ||
to="/$organizationSlug/$projectSlug" | ||
params={{ | ||
organizationSlug: props.currentOrganizationSlug, | ||
projectSlug: props.currentProjectSlug, | ||
}} | ||
className="max-w-[200px] shrink-0 truncate font-medium" | ||
> | ||
{currentProject.slug} | ||
</Link> | ||
{showOrganization ? <div className="italic text-gray-500">/</div> : <></>} | ||
{showProject ? ( | ||
currentOrganization && currentProject ? ( | ||
<Link | ||
to="/$organizationSlug/$projectSlug" | ||
params={{ | ||
organizationSlug: props.currentOrganizationSlug, | ||
projectSlug: props.currentProjectSlug, | ||
}} | ||
className="max-w-[200px] shrink-0 truncate font-medium" | ||
> | ||
{currentProject.slug} | ||
</Link> | ||
) : ( | ||
<div className="h-5 w-48 max-w-[200px] animate-pulse rounded-full bg-gray-800" /> | ||
) | ||
) : ( | ||
<div className="h-5 w-48 max-w-[200px] animate-pulse rounded-full bg-gray-800" /> | ||
<></> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. null here also |
||
)} | ||
<div className="italic text-gray-500">/</div> | ||
{targets?.length && currentOrganization && currentProject && currentTarget ? ( | ||
{showProject ? <div className="italic text-gray-500">/</div> : <></>} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. null here also |
||
{(targets?.length && currentOrganization && currentProject && currentTarget) || isOptional ? ( | ||
<> | ||
<Select | ||
value={props.currentTargetSlug} | ||
onValueChange={id => { | ||
void router.navigate({ | ||
to: '/$organizationSlug/$projectSlug/$targetSlug', | ||
params: { | ||
organizationSlug: props.currentOrganizationSlug, | ||
projectSlug: props.currentProjectSlug, | ||
targetSlug: id, | ||
}, | ||
}); | ||
}} | ||
onValueChange={ | ||
onValueChangeFunc | ||
? id => { | ||
onValueChangeFunc(id); | ||
} | ||
: id => { | ||
void router.navigate({ | ||
to: '/$organizationSlug/$projectSlug/$targetSlug', | ||
params: { | ||
organizationSlug: props.currentOrganizationSlug, | ||
projectSlug: props.currentProjectSlug, | ||
targetSlug: id, | ||
}, | ||
}); | ||
} | ||
} | ||
> | ||
<SelectTrigger variant="default" data-cy="target-picker-trigger"> | ||
<div className="font-medium" data-cy="target-picker-current"> | ||
{currentTarget.slug} | ||
{isOptional ? <SelectValue placeholder="Pick an option" /> : ''} | ||
{currentTarget && !isOptional ? currentTarget.slug : ''}{' '} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ^ simplify like the prior comment |
||
</div> | ||
</SelectTrigger> | ||
<SelectContent> | ||
{targets.map(target => ( | ||
{isOptional ? ( | ||
<SelectItem | ||
key={target.slug} | ||
value={target.slug} | ||
data-cy={`target-picker-option-${target.slug}`} | ||
key={'empty'} | ||
value={'empty'} | ||
data-cy={`project-picker-option-Unassigned`} | ||
> | ||
{target.slug} | ||
Unassigned | ||
</SelectItem> | ||
))} | ||
) : ( | ||
<></> | ||
)} | ||
{targets ? ( | ||
targets.map(target => ( | ||
<SelectItem | ||
key={target.slug} | ||
value={target.slug} | ||
data-cy={`target-picker-option-${target.slug}`} | ||
> | ||
{target.slug} | ||
</SelectItem> | ||
)) | ||
) : ( | ||
<></> | ||
)} | ||
</SelectContent> | ||
</Select> | ||
</> | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Minor: I defer to my colleagues, but I think we should default this or make it nullable to avoid taking any downtime during the deployment window.
We don't get a ton of support tickets so I doubt it'd cause an issue, but I'm extra cautious.