-
Notifications
You must be signed in to change notification settings - Fork 2
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
feat: add select for sorting flow list #4137
Conversation
f235e5b
to
73f0b7c
Compare
@@ -390,6 +400,13 @@ export const editorStore: StateCreator< | |||
lastName: last_name | |||
} | |||
} | |||
publishedFlows: published_flows( | |||
order_by: { created_at: desc } | |||
limit: 1 |
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.
Seen this was only used by the Team page, so though it felt better to expand this initial query
}) => { | ||
const [sortBy, setSortBy] = useState<SortObject>(sortArray[0]); | ||
const [sortDirection, setSortDirection] = useState<SortDirection>("asc"); | ||
useEffect(() => { |
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.
Went with a useEffect here to provide stability for adding more sorting types, would be interested to know if there's cleaner techniques
if (!b[sortKey][0]) return -1; | ||
|
||
const aValue = a[sortKey][0][nestedKey]; | ||
const bValue = b[sortKey][0][nestedKey]; |
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.
Although this possibly looks overly complicated, it allowed me to use the type safety to build the sortArray
while not having to map sorting names to data fields with ternaries or conditional statements.
I iterated this a bit and found that the extra code came a lot from the nested keys in the flows
array, but the extra lines always provided better maintainability for adding or removing sorting criteria while remaining aligned with what data is provided to us from the getFlows
query
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.
Thanks for the the insightful comment on how you get here 👍 This makes things easier to review and understand - much appreciated!
First off - a useEffect()
seems fine for this in order to maintain that reactivity - no real issues here.
The logic here is pretty complex and dense - I think removing the entire concept of nestedKeys
(see previous comments) would help out a lot here as there's just a single level to deal with.
Removed vultr server and associated DNS entries |
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.
A few general comments on the code that might simplify things a lot here.
However, there's a more general design concept to consider here. We know that we're going to need this sort of sort component in a few places fairly soon. The logic of the component is very closely coupled with the incoming data - I'd recommend splitting these up.
Think about a pattern like this -
- A "sort" component which takes in a generic
T[]
- the items to be sorted - It can call a callback function to update state when the user interacts with the object
- Sort keys have to be
keyof typeof T
(I think type-fest allows deep keys also) - We pass in a list of flows, and the setFlows function, and a list of keys
We now would have a component which could easily be used for the feedback, submission and flows pages with no extra code or unit tests required.
Right now, it would have a static UI (this is actually fine) but a future enhancement could to make it a bit more headless or to have variants (e.g. "compact")
Super happy to talk this through tomorrow if you'd like 👍
import TrendingDownIcon from "@mui/icons-material/TrendingDown"; | ||
import TrendingUpIcon from "@mui/icons-material/TrendingUp"; |
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.
I actually don't associate these icons with sorting, I'd expect something more like a caret or just an "asc" / "dec" label somewhere (e.g. another section in the dropdown).
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.
Ah yeah! This was mostly for Ian to see it working and then we'll think about how to add it to the sort. Good feedback to have in tho
const addToSearchParams = (sortKey: SortTypes) => { | ||
navigation.navigate( | ||
{ | ||
pathname: window.location.pathname, | ||
search: `?sort=${sortKey}`, | ||
}, | ||
{ | ||
replace: true, | ||
}, | ||
); | ||
}; |
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.
Adding this to the URL is great to see 👍
Two thoughts -
- We should add direction also (now)
- This could be a great hook once we hit a second use case (
useSearchParamState()
?) (future)
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.
Just seen this is listed still a to do on the PR description 👌
|
||
type SortDirection = "asc" | "desc"; | ||
type SortKeys = keyof FlowSummary; | ||
type SortNestedKeys = keyof PublishedFlowSummary | keyof FlowSummaryOperations; |
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.
I think a lot of the complexity in this component is coming from the fact we have a slightly tricky object to work with initially returned by our GetFlows
query.
We can't totally unnest / flatten natively via a GraphQL query, but don't be shy about transforming a database object to a more appropriate model for use in the frontend.
For example, if our FlowSummary
had properties publishedAt
or status
, we can simplify a whole lot of things in this component.
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.
Alternatively, we can use key paths as assessors - type-fest
and lodash
can really help here.
For example, we could use "publishedFlows[0].publishedAt"
as a key using lodash's get()
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.
This was a great one and I did have to add type-fest
to the editor to get the method
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.
I think we've had it in and out a few times as a dependency, we also use it in the API and I think planx-core.
flows: FlowSummary[]; | ||
setFlows: React.Dispatch<React.SetStateAction<FlowSummary[] | null>>; | ||
}) => { | ||
const [sortBy, setSortBy] = useState<SortObject>(sortArray[0]); |
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.
Naming here is a little tricky to understand - it's not super easy to work out at glance if this an object or array?
Often we can leverage the type system to avoid pseudo-Hungarian notation for variables.
A simpler approach may be to just store the sortKey as a string and then maintain a map/lookup object for reference when we need display values.
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.
I've taken the approach of slugifying the displayName and using that as a lookup key, since I had to keep the sortKey as part of an object connected with the name we use for displaying it in the select.
There's a chance here that I've Wiley Coyote'd myself and ran off a cliff when the answer was much simpler, but it met what I wanted in terms of how you can define your sort options and how they are displayed in one place, and the component handles the logic of sorting / finding them.
if (!b[sortKey][0]) return -1; | ||
|
||
const aValue = a[sortKey][0][nestedKey]; | ||
const bValue = b[sortKey][0][nestedKey]; |
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.
Thanks for the the insightful comment on how you get here 👍 This makes things easier to review and understand - much appreciated!
First off - a useEffect()
seems fine for this in order to maintain that reactivity - no real issues here.
The logic here is pretty complex and dense - I think removing the entire concept of nestedKeys
(see previous comments) would help out a lot here as there's just a single level to deal with.
f235879
to
9b7aa3f
Compare
editor.planx.uk/src/pages/Team.tsx
Outdated
<SortControl<FlowSummary> | ||
records={flows} | ||
setRecords={setFlows} | ||
sortOptions={sortArray} |
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.
Switched naming to something more generic and reusable across the comp and remove refs to flow
in the naming as well
|
||
const navigation = useNavigation(); | ||
const route = useCurrentRoute(); | ||
const selectedDisplaySlug = slugify(selectedSort.displayName); |
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.
This is probably the part I feel most unsure about, but I'm comfortable that it delivers a unifying parameter I can use to search for sortOptions
while adding it to the search params in a nice way for the user to see and read. My issue lay in the lack of readability in the key naming for deeply nested keys, and the fact displayName
could really contain any sort of spaces and the like.
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.
Yeah it's a good question - fieldName
is guaranteed to be unique, but not necessarily readable.
There's no guarantee that slugify(selectedSort.displayName)
doesn't return duplicates through right? It's unlikely but possible.
This might just be a data structure question. If we were to make sortOptions
something like this our problems go away -
sortOptions: Record<string, SortableFields<T>>
...
const sortArray: Record<string, SortableFields<FlowSummary>> = {
name: { displayName: "Name", fieldName: "name" },
lastUpdated: { displayName: "Last updated", fieldName: "updatedAt" },
lastPublished: {
displayName: "Last published",
fieldName: `publishedFlows.0.publishedAt`,
},
}
This would -
- Force a unique key
- Provide us with a map already
What do you think?
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.
Yeah this was an iteration I considered, I was thinking this is dealing with a complexity which is within the component (so not obvious when you add it to your table comp) so it is better to handle it within the component?
Imagining someone building this into their component and being confused why you have to add so many names, ultimately the same thing three times.
I can see the risk of duplicates but thought it was quite edge casey and resolvable.
But, I'm happy to integrate the change, I can see the benefits of the safety, so you don't need to try find your error in the naming.
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.
Yeah this is a fair point - lets aim to keep the complexity within the component I think 👍
Your version is clearer and simpler for consumers. The risk of duplicates is very very low!
@@ -351,6 +364,13 @@ const Team: React.FC = () => { | |||
</Box> | |||
{showAddFlowButton && <AddFlowButton flows={flows} />} | |||
</Box> | |||
{hasFeatureFlag("SORT_FLOWS") && flows && ( |
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.
We can maybe drop the flows
condition by dropping this into the next block which is already checking the more accurate teamHasFlows
.
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.
I'll add this in when I combine it with the Filters block
|
||
const navigation = useNavigation(); | ||
const route = useCurrentRoute(); | ||
const selectedDisplaySlug = slugify(selectedSort.displayName); |
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.
Yeah it's a good question - fieldName
is guaranteed to be unique, but not necessarily readable.
There's no guarantee that slugify(selectedSort.displayName)
doesn't return duplicates through right? It's unlikely but possible.
This might just be a data structure question. If we were to make sortOptions
something like this our problems go away -
sortOptions: Record<string, SortableFields<T>>
...
const sortArray: Record<string, SortableFields<FlowSummary>> = {
name: { displayName: "Name", fieldName: "name" },
lastUpdated: { displayName: "Last updated", fieldName: "updatedAt" },
lastPublished: {
displayName: "Last published",
fieldName: `publishedFlows.0.publishedAt`,
},
}
This would -
- Force a unique key
- Provide us with a map already
What do you think?
refine component and sort style style refinement on layout add status to getFlows refine type use on sort keys
5587d15
to
8c48a5b
Compare
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.
Thanks - changes here look great, and we've ended up with a solution which is much for robust and expandable - love to see it!
Not manually tested due to failing CI (due to cache issue - please run gh cache delete --all
and then re-run CI).
{ | ||
displayName: "Name", | ||
fieldName: "name", | ||
directionNames: { asc: "A - Z", desc: "Z - A" }, |
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.
Nice! When we have more use cases for this we may find we want these display values centralised within the component, and just allow consumers to define by type (e.g. direction: "alphabetical" | "date"
). This is perfect for now 👍
@DafyddLlyr manual testing is probably best with the filters included |
What does this PR do?
Working towards this EPIC: https://trello.com/c/7av8lsVr/3160-epic-as-an-editor-i-want-to-have-a-way-to-organise-my-flows-so-that-i-can-easily-find-what-im-looking-for
The main task was to add the ability to Sort flows in the Team page, the page which shows a Team's flows.
Approach
The Team page component is quite long, so I created a new component separate from this to handle the logic of sorting and also contain a Select component for users to interact with.
This new component takes in state variables
flows
andsetFlows
from the Team page component, which felt right for me so the flow list content always lives in one place but is given to other components to augment. I'd want to repeat this pattern for filtering the flows.My hope for the sort component was to provide a maintainable, scalable component for sorting the list of flows on the Team page using keys we receive from
getFlows()
contained within theFlowSummary
interface. It may need to be used across many different tables, so I have enabled a generic Type to be added when the component is used, so it will now received arecord
as an Array, a way of setting that recordsetRecord
, and options for sorting the ArraysortOptions
. This should enable the Type safety I want for this specific use case while extending it for future onesSee this as just the building blocks of the sort component I can add into Ian's work on the new Team page and filter.
What's included?
Right now it covers: Name, Last Published, and Last Updated. I added an ascending or descending toggle as well.
In order to make this work, I have extended the
getFlows()
query to pull inpublished_flows
data andstatus
data from Hasura.Future Improvement:
Important
This component is hidden using a FeatureFlag -
SORT_FLOWS