Skip to content
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

[ITG-126] Change style of checkbox icons, add tooltips, auto check subtasks of task. #51

Merged
merged 6 commits into from
Jun 6, 2022
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 10 additions & 3 deletions bode/bode/models/task/actions.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,9 @@ def edit_task(task_id, **task_data):
def is_interchangable_relation(relation):
return relation.type == RelationType.Interchangable.value and str(relation.first_task_id) == task_id

def is_subtask_relation(relation):
return relation.type == RelationType.Subtask.value and str(relation.first_task_id) == task_id

task = get_task(task_id)

for key, value in task_data.items():
Expand All @@ -47,12 +50,16 @@ def is_interchangable_relation(relation):
if related_task.status != TaskStatus.TODO.value:
continue
inter_task_data = {
"title": related_task.title,
"description": related_task.description,
"due_date": related_task.due_date,
"status": TaskStatus.INDIRECTLY_DONE.value,
}
edit_task(str(related_task.id), **inter_task_data)
if is_subtask_relation(relation):
if related_task.status == TaskStatus.DONE.value:
continue
subtask_data = {
"status": TaskStatus.DONE.value,
}
edit_task(str(related_task.id), **subtask_data)

return task

Expand Down
2 changes: 1 addition & 1 deletion bode/bode/resources/tasks/schemas.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ class TaskSchema(BaseSchema):
description = fields.String()
due_date = fields.DateTime()
status = fields.String(validate=validate.OneOf(TaskStatus.list()))
is_blocked = fields.Function(lambda task: task.status != TaskStatus.DONE.value and is_task_blocked(task.id))
is_blocked = fields.Function(lambda task: is_task_blocked(task.id))
tags = fields.List(fields.Nested(TagSchema))
relation_types = fields.List(fields.String(validate=validate.OneOf(DirectedRelationType.list())))

Expand Down
23 changes: 23 additions & 0 deletions cabra/src/assets/indirectly_done_icon.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
53 changes: 37 additions & 16 deletions cabra/src/pages/components/CheckBox.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import tw, { TwStyle, styled } from "twin.macro";

import CheckboxBlankIcon from "remixicon-react/CheckboxBlankCircleLineIcon";
import CheckboxCheckedIcon from "remixicon-react/CheckboxCircleLineIcon";
import CheckboxMultiIcon from "remixicon-react/CheckboxMultipleLineIcon";
import CheckboxIndirectlyDoneIcon from "../../assets/indirectly_done_icon.svg";
import DisabledIcon from "remixicon-react/IndeterminateCircleLineIcon";
import { InputHTMLAttributes } from "react";
import ReactTooltip from "react-tooltip";
Expand All @@ -14,6 +14,7 @@ type Props = Omit<
> & {
id: string;
status: TaskStatus;
blocked: boolean;
size?: Size;
};
type Size = "sm" | "base" | "lg";
Expand All @@ -31,7 +32,7 @@ const iconSizes: Record<Size, { size: number }> = {
base: { size: 24 },
lg: { size: 28 },
};
const Label = styled.label<{ size: Size }>`
const Label = styled.label<{ size: Size; blocked: boolean }>`
${tw`select-none`}

span {
Expand All @@ -44,15 +45,12 @@ const Label = styled.label<{ size: Size }>`
${tw`bg-slate-500 cursor-auto`}
}
input:enabled ~ span {
${({ blocked }) => blocked && tw`bg-orange-300`}
${tw`hover:opacity-60`}
}
`;

const resolveIcon = (
status: TaskStatus,
size: Size,
disabled: boolean | undefined
) => {
const resolveIcon = (status: TaskStatus, size: Size, disabled: boolean) => {
if (disabled) {
return <DisabledIcon {...iconSizes[size]} />;
}
Expand All @@ -62,14 +60,31 @@ const resolveIcon = (
case TaskStatus.TODO:
return <CheckboxBlankIcon {...iconSizes[size]} />;
case TaskStatus.INDIRECTLY_DONE:
return <CheckboxMultiIcon {...iconSizes[size]} />;
return <img src={CheckboxIndirectlyDoneIcon} {...iconSizes[size]} />;
}
};

const resolveTooltip = (status: TaskStatus, blocked: boolean) => {
if (blocked) {
if (status === TaskStatus.DONE) {
return "Task is done but is blocked by not done task.";
}
return "Task is blocked by not done task.";
}
switch (status) {
case TaskStatus.DONE:
return "Task was done.";
case TaskStatus.INDIRECTLY_DONE:
return "Task was done indirectly by other task.";
case TaskStatus.TODO:
return "Task to do.";
}
};

export default function Checkbox({
id,
size = "base",
disabled,
blocked,
status,
...props
}: Props) {
Expand All @@ -78,18 +93,24 @@ export default function Checkbox({
<Label
id={id}
size={size}
blocked={blocked}
onClick={(event) => event.stopPropagation()}
data-tip
data-for={id}
>
<StyledInput {...props} disabled={disabled} id={id} type="checkbox" />
<span>{resolveIcon(status, size, disabled)}</span>
<StyledInput
{...props}
disabled={blocked && status !== TaskStatus.DONE}
id={id}
type="checkbox"
/>
<span>
{resolveIcon(status, size, blocked && status !== TaskStatus.DONE)}
</span>
</Label>
{disabled && (
<ReactTooltip id={id} place="bottom" effect="solid">
Task is blocked by other task.
</ReactTooltip>
)}
<ReactTooltip id={id} place="bottom" effect="solid">
{resolveTooltip(status, blocked)}
</ReactTooltip>
</>
);
}
2 changes: 1 addition & 1 deletion cabra/src/pages/components/RelatedTask.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,7 @@ export default function RelatedTask({
id={task.id}
checked={task.status !== TaskStatus.TODO}
onChange={handleIsDoneChange}
disabled={task.isBlocked}
blocked={task.isBlocked}
status={task.status}
/>
</p>
Expand Down
2 changes: 1 addition & 1 deletion cabra/src/pages/components/TaskDetailsCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,7 @@ export default function TaskDetails({ id }: Props) {
checked={task.status !== TaskStatus.TODO}
id={`task-${task.id}`}
onChange={handleStatusChange}
disabled={task.isBlocked}
blocked={task.isBlocked}
status={task.status}
/>
<Link to={routeHelpers.task.edit(task.id)}>
Expand Down
2 changes: 1 addition & 1 deletion cabra/src/pages/components/TaskListItem.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ export default function TaskListItem({ task }: Props) {
checked={task.status !== TaskStatus.TODO}
id={`task-${task.id}`}
onChange={handleIsDoneChange}
disabled={task.isBlocked}
blocked={task.isBlocked}
size="sm"
status={task.status}
/>
Expand Down
3 changes: 3 additions & 0 deletions cabra/src/pages/hooks/useTaskDetails.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,9 @@ const useTask = (id: string) => {
client.invalidateQueries(
getRelatedTasks.cacheKey(task?.id, DirectedRelationType.Blocks)
);
client.invalidateQueries(
getRelatedTasks.cacheKey(task?.id, DirectedRelationType.Subtask)
);
},
});
const handleStatusChange = async () => {
Expand Down