chore: bug fixes and improvement (#3303)

* refactor: updated preloaded function for the list view quick add

* fix: resolved bug in the assignee dropdown

* chore: issue sidebar link improvement

* fix: resolved subscription store bug

* chore: updated preloaded function for the kanban layout quick add

* chore: resolved issues in the list filters and component

* chore: filter store updated

* fix: issue serializer changed

* chore: quick add preload function updated

* fix: build error

* fix: serializer changed

* fix: minor request change

* chore: resolved build issues and updated the prepopulated data in the quick add issue.

* fix: build fix and code refactor

* fix: spreadsheet layout quick add fix

* fix: issue peek overview link section updated

* fix: cycle status bug fix

* fix: serializer changes

* fix: assignee and labels listing

* chore: issue modal parent_id default value updated

* fix: cycle and module issue serializer change

* fix: cycle list serializer changed

* chore: prepopulated validation in both list and kanban for quick add and group header add issues

* chore: group header validation added

* fix: issue response payload change

* dev: make cycle and module issue create response simillar

* chore: custom control link component added

* dev: make issue create and update response simillar to list and retrieve

* fix: build error

* chore: control link component improvement

* chore: globalise issue peek overview

* chore: control link component improvement

* chore: made changes and optimised the issue peek overview root

* build-error: resolved build erros for issueId dependancy from issue detail store

* chore: peek overview link fix

* dev: update state nullable rule

---------

Co-authored-by: gurusainath <gurusainath007@gmail.com>
Co-authored-by: NarayanBavisetti <narayan3119@gmail.com>
Co-authored-by: pablohashescobar <nikhilschacko@gmail.com>
This commit is contained in:
Anmol Singh Bhatia 2024-01-05 23:37:13 +05:30 committed by sriram veeraghanta
parent 266f14d550
commit efd3ebf067
65 changed files with 630 additions and 1565 deletions

View File

@ -30,6 +30,8 @@ from plane.db.models import (
CommentReaction, CommentReaction,
IssueVote, IssueVote,
IssueRelation, IssueRelation,
State,
Project,
) )
@ -69,19 +71,16 @@ class IssueProjectLiteSerializer(BaseSerializer):
##TODO: Find a better way to write this serializer ##TODO: Find a better way to write this serializer
## Find a better approach to save manytomany? ## Find a better approach to save manytomany?
class IssueCreateSerializer(BaseSerializer): class IssueCreateSerializer(BaseSerializer):
state_detail = StateSerializer(read_only=True, source="state") # ids
created_by_detail = UserLiteSerializer(read_only=True, source="created_by") state_id = serializers.PrimaryKeyRelatedField(source="state", queryset=State.objects.all(), required=False, allow_null=True)
project_detail = ProjectLiteSerializer(read_only=True, source="project") parent_id = serializers.PrimaryKeyRelatedField(source='parent', queryset=Issue.objects.all(), required=False, allow_null=True)
workspace_detail = WorkspaceLiteSerializer(read_only=True, source="workspace") label_ids = serializers.ListField(
child=serializers.PrimaryKeyRelatedField(queryset=Label.objects.all()),
assignees = serializers.ListField(
child=serializers.PrimaryKeyRelatedField(queryset=User.objects.all()),
write_only=True, write_only=True,
required=False, required=False,
) )
assignee_ids = serializers.ListField(
labels = serializers.ListField( child=serializers.PrimaryKeyRelatedField(queryset=User.objects.all()),
child=serializers.PrimaryKeyRelatedField(queryset=Label.objects.all()),
write_only=True, write_only=True,
required=False, required=False,
) )
@ -100,8 +99,10 @@ class IssueCreateSerializer(BaseSerializer):
def to_representation(self, instance): def to_representation(self, instance):
data = super().to_representation(instance) data = super().to_representation(instance)
data['assignees'] = [str(assignee.id) for assignee in instance.assignees.all()] assignee_ids = self.initial_data.get('assignee_ids')
data['labels'] = [str(label.id) for label in instance.labels.all()] data['assignee_ids'] = assignee_ids if assignee_ids else []
label_ids = self.initial_data.get('label_ids')
data['label_ids'] = label_ids if label_ids else []
return data return data
def validate(self, data): def validate(self, data):
@ -114,8 +115,8 @@ class IssueCreateSerializer(BaseSerializer):
return data return data
def create(self, validated_data): def create(self, validated_data):
assignees = validated_data.pop("assignees", None) assignees = validated_data.pop("assignee_ids", None)
labels = validated_data.pop("labels", None) labels = validated_data.pop("label_ids", None)
project_id = self.context["project_id"] project_id = self.context["project_id"]
workspace_id = self.context["workspace_id"] workspace_id = self.context["workspace_id"]
@ -173,8 +174,8 @@ class IssueCreateSerializer(BaseSerializer):
return issue return issue
def update(self, instance, validated_data): def update(self, instance, validated_data):
assignees = validated_data.pop("assignees", None) assignees = validated_data.pop("assignee_ids", None)
labels = validated_data.pop("labels", None) labels = validated_data.pop("labels_ids", None)
# Related models # Related models
project_id = instance.project_id project_id = instance.project_id
@ -544,7 +545,7 @@ class IssueSerializer(DynamicBaseSerializer):
attachment_count = serializers.IntegerField(read_only=True) attachment_count = serializers.IntegerField(read_only=True)
link_count = serializers.IntegerField(read_only=True) link_count = serializers.IntegerField(read_only=True)
# is # is_subscribed
is_subscribed = serializers.BooleanField(read_only=True) is_subscribed = serializers.BooleanField(read_only=True)
class Meta: class Meta:

View File

@ -99,6 +99,7 @@ class BaseViewSet(TimezoneMixin, ModelViewSet, BasePaginator):
response = super().handle_exception(exc) response = super().handle_exception(exc)
return response return response
except Exception as e: except Exception as e:
print(e) if settings.DEBUG else print("Server Error")
if isinstance(e, IntegrityError): if isinstance(e, IntegrityError):
return Response( return Response(
{"error": "The payload is not valid"}, {"error": "The payload is not valid"},
@ -124,8 +125,7 @@ class BaseViewSet(TimezoneMixin, ModelViewSet, BasePaginator):
{"error": f"key {e} does not exist"}, {"error": f"key {e} does not exist"},
status=status.HTTP_400_BAD_REQUEST, status=status.HTTP_400_BAD_REQUEST,
) )
print(e) if settings.DEBUG else print("Server Error")
capture_exception(e) capture_exception(e)
return Response({"error": "Something went wrong please try again later"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR) return Response({"error": "Something went wrong please try again later"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)

View File

@ -31,6 +31,7 @@ from plane.app.serializers import (
CycleSerializer, CycleSerializer,
CycleIssueSerializer, CycleIssueSerializer,
CycleFavoriteSerializer, CycleFavoriteSerializer,
IssueSerializer,
IssueStateSerializer, IssueStateSerializer,
CycleWriteSerializer, CycleWriteSerializer,
CycleUserPropertiesSerializer, CycleUserPropertiesSerializer,
@ -46,9 +47,9 @@ from plane.db.models import (
IssueAttachment, IssueAttachment,
Label, Label,
CycleUserProperties, CycleUserProperties,
IssueSubscriber,
) )
from plane.bgtasks.issue_activites_task import issue_activity from plane.bgtasks.issue_activites_task import issue_activity
from plane.utils.grouper import group_results
from plane.utils.issue_filters import issue_filters from plane.utils.issue_filters import issue_filters
from plane.utils.analytics_plot import burndown_plot from plane.utils.analytics_plot import burndown_plot
@ -322,6 +323,8 @@ class CycleViewSet(WebhookMixin, BaseViewSet):
project_id=project_id, project_id=project_id,
owned_by=request.user, owned_by=request.user,
) )
cycle = self.get_queryset().filter(pk=serializer.data["id"]).first()
serializer = CycleSerializer(cycle)
return Response(serializer.data, status=status.HTTP_201_CREATED) return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
else: else:
@ -548,6 +551,8 @@ class CycleIssueViewSet(WebhookMixin, BaseViewSet):
.prefetch_related("labels") .prefetch_related("labels")
.order_by(order_by) .order_by(order_by)
.filter(**filters) .filter(**filters)
.annotate(cycle_id=F("issue_cycle__cycle_id"))
.annotate(module_id=F("issue_module__module_id"))
.annotate( .annotate(
link_count=IssueLink.objects.filter(issue=OuterRef("id")) link_count=IssueLink.objects.filter(issue=OuterRef("id"))
.order_by() .order_by()
@ -560,8 +565,15 @@ class CycleIssueViewSet(WebhookMixin, BaseViewSet):
.annotate(count=Func(F("id"), function="Count")) .annotate(count=Func(F("id"), function="Count"))
.values("count") .values("count")
) )
.annotate(
is_subscribed=Exists(
IssueSubscriber.objects.filter(
subscriber=self.request.user, issue_id=OuterRef("id")
)
)
)
) )
serializer = IssueStateSerializer( serializer = IssueSerializer(
issues, many=True, fields=fields if fields else None issues, many=True, fields=fields if fields else None
) )
return Response(serializer.data, status=status.HTTP_200_OK) return Response(serializer.data, status=status.HTTP_200_OK)
@ -652,8 +664,10 @@ class CycleIssueViewSet(WebhookMixin, BaseViewSet):
) )
# Return all Cycle Issues # Return all Cycle Issues
issues = self.get_queryset().values_list("issue_id", flat=True)
return Response( return Response(
CycleIssueSerializer(self.get_queryset(), many=True).data, IssueSerializer(Issue.objects.filter(pk__in=issues), many=True).data,
status=status.HTTP_200_OK, status=status.HTTP_200_OK,
) )

View File

@ -34,11 +34,11 @@ from rest_framework.parsers import MultiPartParser, FormParser
# Module imports # Module imports
from . import BaseViewSet, BaseAPIView, WebhookMixin from . import BaseViewSet, BaseAPIView, WebhookMixin
from plane.app.serializers import ( from plane.app.serializers import (
IssueCreateSerializer,
IssueActivitySerializer, IssueActivitySerializer,
IssueCommentSerializer, IssueCommentSerializer,
IssuePropertySerializer, IssuePropertySerializer,
IssueSerializer, IssueSerializer,
IssueCreateSerializer,
LabelSerializer, LabelSerializer,
IssueFlatSerializer, IssueFlatSerializer,
IssueLinkSerializer, IssueLinkSerializer,
@ -110,12 +110,7 @@ class IssueViewSet(WebhookMixin, BaseViewSet):
def get_queryset(self): def get_queryset(self):
return ( return (
Issue.issue_objects.annotate( Issue.issue_objects
sub_issues_count=Issue.issue_objects.filter(parent=OuterRef("id"))
.order_by()
.annotate(count=Func(F("id"), function="Count"))
.values("count")
)
.filter(project_id=self.kwargs.get("project_id")) .filter(project_id=self.kwargs.get("project_id"))
.filter(workspace__slug=self.kwargs.get("slug")) .filter(workspace__slug=self.kwargs.get("slug"))
.select_related("project") .select_related("project")
@ -143,13 +138,11 @@ class IssueViewSet(WebhookMixin, BaseViewSet):
.order_by() .order_by()
.annotate(count=Func(F("id"), function="Count")) .annotate(count=Func(F("id"), function="Count"))
.values("count") .values("count")
) ).annotate(
.annotate( sub_issues_count=Issue.issue_objects.filter(parent=OuterRef("id"))
is_subscribed=Exists( .order_by()
IssueSubscriber.objects.filter( .annotate(count=Func(F("id"), function="Count"))
subscriber=self.request.user, issue_id=OuterRef("id") .values("count")
)
)
) )
).distinct() ).distinct()
@ -251,16 +244,13 @@ class IssueViewSet(WebhookMixin, BaseViewSet):
current_instance=None, current_instance=None,
epoch=int(timezone.now().timestamp()), epoch=int(timezone.now().timestamp()),
) )
issue = self.get_queryset().filter(pk=serializer.data["id"]).first()
serializer = IssueSerializer(issue)
return Response(serializer.data, status=status.HTTP_201_CREATED) return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
def retrieve(self, request, slug, project_id, pk=None): def retrieve(self, request, slug, project_id, pk=None):
issue = Issue.issue_objects.annotate( issue = self.get_queryset().filter(pk=pk).first()
sub_issues_count=Issue.issue_objects.filter(parent=OuterRef("id"))
.order_by()
.annotate(count=Func(F("id"), function="Count"))
.values("count")
).get(workspace__slug=slug, project_id=project_id, pk=pk)
return Response( return Response(
IssueSerializer(issue, fields=self.fields, expand=self.expand).data, IssueSerializer(issue, fields=self.fields, expand=self.expand).data,
status=status.HTTP_200_OK, status=status.HTTP_200_OK,
@ -284,7 +274,8 @@ class IssueViewSet(WebhookMixin, BaseViewSet):
current_instance=current_instance, current_instance=current_instance,
epoch=int(timezone.now().timestamp()), epoch=int(timezone.now().timestamp()),
) )
return Response(serializer.data, status=status.HTTP_200_OK) issue = self.get_queryset().filter(pk=pk).first()
return Response(IssueSerializer(issue).data, status=status.HTTP_200_OK)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
def destroy(self, request, slug, project_id, pk=None): def destroy(self, request, slug, project_id, pk=None):
@ -719,13 +710,6 @@ class SubIssuesEndpoint(BaseAPIView):
.annotate(count=Func(F("id"), function="Count")) .annotate(count=Func(F("id"), function="Count"))
.values("count") .values("count")
) )
.annotate(
is_subscribed=Exists(
IssueSubscriber.objects.filter(
subscriber=self.request.user, issue_id=OuterRef("id")
)
)
)
.prefetch_related( .prefetch_related(
Prefetch( Prefetch(
"issue_reactions", "issue_reactions",
@ -1080,7 +1064,7 @@ class IssueArchiveViewSet(BaseViewSet):
else issue_queryset.filter(parent__isnull=True) else issue_queryset.filter(parent__isnull=True)
) )
issues = IssueLiteSerializer( issues = IssueSerializer(
issue_queryset, many=True, fields=fields if fields else None issue_queryset, many=True, fields=fields if fields else None
).data ).data
return Response(issues, status=status.HTTP_200_OK) return Response(issues, status=status.HTTP_200_OK)
@ -1163,16 +1147,6 @@ class IssueSubscriberViewSet(BaseViewSet):
project_id=project_id, project_id=project_id,
is_active=True, is_active=True,
) )
.annotate(
is_subscribed=Exists(
IssueSubscriber.objects.filter(
workspace__slug=slug,
project_id=project_id,
issue_id=issue_id,
subscriber=OuterRef("member"),
)
)
)
.select_related("member") .select_related("member")
) )
serializer = ProjectMemberLiteSerializer(members, many=True) serializer = ProjectMemberLiteSerializer(members, many=True)
@ -1613,7 +1587,7 @@ class IssueDraftViewSet(BaseViewSet):
else: else:
issue_queryset = issue_queryset.order_by(order_by_param) issue_queryset = issue_queryset.order_by(order_by_param)
issues = IssueLiteSerializer( issues = IssueSerializer(
issue_queryset, many=True, fields=fields if fields else None issue_queryset, many=True, fields=fields if fields else None
).data ).data
return Response(issues, status=status.HTTP_200_OK) return Response(issues, status=status.HTTP_200_OK)

View File

@ -20,7 +20,7 @@ from plane.app.serializers import (
ModuleIssueSerializer, ModuleIssueSerializer,
ModuleLinkSerializer, ModuleLinkSerializer,
ModuleFavoriteSerializer, ModuleFavoriteSerializer,
IssueStateSerializer, IssueSerializer,
ModuleUserPropertiesSerializer, ModuleUserPropertiesSerializer,
) )
from plane.app.permissions import ProjectEntityPermission, ProjectLitePermission from plane.app.permissions import ProjectEntityPermission, ProjectLitePermission
@ -33,6 +33,7 @@ from plane.db.models import (
ModuleFavorite, ModuleFavorite,
IssueLink, IssueLink,
IssueAttachment, IssueAttachment,
IssueSubscriber,
ModuleUserProperties, ModuleUserProperties,
) )
from plane.bgtasks.issue_activites_task import issue_activity from plane.bgtasks.issue_activites_task import issue_activity
@ -353,6 +354,8 @@ class ModuleIssueViewSet(WebhookMixin, BaseViewSet):
.prefetch_related("labels") .prefetch_related("labels")
.order_by(order_by) .order_by(order_by)
.filter(**filters) .filter(**filters)
.annotate(cycle_id=F("issue_cycle__cycle_id"))
.annotate(module_id=F("issue_module__module_id"))
.annotate( .annotate(
link_count=IssueLink.objects.filter(issue=OuterRef("id")) link_count=IssueLink.objects.filter(issue=OuterRef("id"))
.order_by() .order_by()
@ -365,8 +368,15 @@ class ModuleIssueViewSet(WebhookMixin, BaseViewSet):
.annotate(count=Func(F("id"), function="Count")) .annotate(count=Func(F("id"), function="Count"))
.values("count") .values("count")
) )
.annotate(
is_subscribed=Exists(
IssueSubscriber.objects.filter(
subscriber=self.request.user, issue_id=OuterRef("id")
)
)
)
) )
serializer = IssueStateSerializer( serializer = IssueSerializer(
issues, many=True, fields=fields if fields else None issues, many=True, fields=fields if fields else None
) )
return Response(serializer.data, status=status.HTTP_200_OK) return Response(serializer.data, status=status.HTTP_200_OK)
@ -447,8 +457,10 @@ class ModuleIssueViewSet(WebhookMixin, BaseViewSet):
epoch=int(timezone.now().timestamp()), epoch=int(timezone.now().timestamp()),
) )
issues = self.get_queryset().values_list("issue_id", flat=True)
return Response( return Response(
ModuleIssueSerializer(self.get_queryset(), many=True).data, IssueSerializer(Issue.objects.filter(pk__in=issues), many=True).data,
status=status.HTTP_200_OK, status=status.HTTP_200_OK,
) )

View File

@ -24,7 +24,7 @@ from . import BaseViewSet, BaseAPIView
from plane.app.serializers import ( from plane.app.serializers import (
GlobalViewSerializer, GlobalViewSerializer,
IssueViewSerializer, IssueViewSerializer,
IssueLiteSerializer, IssueSerializer,
IssueViewFavoriteSerializer, IssueViewFavoriteSerializer,
) )
from plane.app.permissions import ( from plane.app.permissions import (
@ -42,6 +42,7 @@ from plane.db.models import (
IssueReaction, IssueReaction,
IssueLink, IssueLink,
IssueAttachment, IssueAttachment,
IssueSubscriber,
) )
from plane.utils.issue_filters import issue_filters from plane.utils.issue_filters import issue_filters
from plane.utils.grouper import group_results from plane.utils.grouper import group_results
@ -127,6 +128,19 @@ class GlobalViewIssuesViewSet(BaseViewSet):
.annotate(count=Func(F("id"), function="Count")) .annotate(count=Func(F("id"), function="Count"))
.values("count") .values("count")
) )
.annotate(
sub_issues_count=Issue.issue_objects.filter(parent=OuterRef("id"))
.order_by()
.annotate(count=Func(F("id"), function="Count"))
.values("count")
)
.annotate(
is_subscribed=Exists(
IssueSubscriber.objects.filter(
subscriber=self.request.user, issue_id=OuterRef("id")
)
)
)
) )
# Priority Ordering # Priority Ordering
@ -185,7 +199,7 @@ class GlobalViewIssuesViewSet(BaseViewSet):
else: else:
issue_queryset = issue_queryset.order_by(order_by_param) issue_queryset = issue_queryset.order_by(order_by_param)
serializer = IssueLiteSerializer( serializer = IssueSerializer(
issue_queryset, many=True, fields=fields if fields else None issue_queryset, many=True, fields=fields if fields else None
) )
return Response(serializer.data, status=status.HTTP_200_OK) return Response(serializer.data, status=status.HTTP_200_OK)

View File

@ -1,7 +1,13 @@
import { TIssue } from "./issues"; import { TIssue } from "./issues/base";
import type { IProjectLite } from "./projects"; import type { IProjectLite } from "./projects";
export interface IInboxIssue extends TIssue { export type TInboxIssueExtended = {
completed_at: string | null;
start_date: string | null;
target_date: string | null;
};
export interface IInboxIssue extends TIssue, TInboxIssueExtended {
issue_inbox: { issue_inbox: {
duplicate_to: string | null; duplicate_to: string | null;
id: string; id: string;
@ -48,7 +54,12 @@ interface StatusDuplicate {
duplicate_to: string; duplicate_to: string;
} }
export type TInboxStatus = StatusReject | StatusSnoozed | StatusAccepted | StatusDuplicate | StatePending; export type TInboxStatus =
| StatusReject
| StatusSnoozed
| StatusAccepted
| StatusDuplicate
| StatePending;
export interface IInboxFilterOptions { export interface IInboxFilterOptions {
priority?: string[] | null; priority?: string[] | null;

View File

@ -1,8 +1,6 @@
import { ReactElement } from "react"; import { ReactElement } from "react";
import { KeyedMutator } from "swr"; import { KeyedMutator } from "swr";
import type { import type {
IState,
IUser,
ICycle, ICycle,
IModule, IModule,
IUserLite, IUserLite,
@ -12,6 +10,7 @@ import type {
Properties, Properties,
IIssueDisplayFilterOptions, IIssueDisplayFilterOptions,
IIssueReaction, IIssueReaction,
TIssue,
} from "@plane/types"; } from "@plane/types";
export interface IIssueCycle { export interface IIssueCycle {
@ -78,59 +77,6 @@ export interface IssueRelation {
relation: "blocking" | null; relation: "blocking" | null;
} }
export interface IIssue {
archived_at: string;
assignees: string[];
assignee_details: IUser[];
attachment_count: number;
attachments: any[];
issue_relations: IssueRelation[];
issue_reactions: IIssueReaction[];
related_issues: IssueRelation[];
bridge_id?: string | null;
completed_at: Date;
created_at: string;
created_by: string;
cycle: string | null;
cycle_id: string | null;
cycle_detail: ICycle | null;
description: any;
description_html: any;
description_stripped: any;
estimate_point: number | null;
id: string;
// tempId is used for optimistic updates. It is not a part of the API response.
tempId?: string;
issue_cycle: IIssueCycle | null;
issue_link: ILinkDetails[];
issue_module: IIssueModule | null;
labels: string[];
label_details: any[];
is_draft: boolean;
links_list: IIssueLink[];
link_count: number;
module: string | null;
module_id: string | null;
name: string;
parent: string | null;
parent_detail: IIssueParent | null;
priority: TIssuePriorities;
project: string;
project_detail: IProjectLite;
sequence_id: number;
sort_order: number;
sprints: string | null;
start_date: string | null;
state: string;
state_detail: IState;
sub_issues_count: number;
target_date: string | null;
updated_at: string;
updated_by: string;
workspace: string;
workspace_detail: IWorkspaceLite;
}
export interface ISubIssuesState { export interface ISubIssuesState {
backlog: number; backlog: number;
unstarted: number; unstarted: number;
@ -283,62 +229,3 @@ export interface IGroupByColumn {
export interface IIssueMap { export interface IIssueMap {
[key: string]: TIssue; [key: string]: TIssue;
} }
// new issue structure types
export type TIssue = {
id: string;
name: string;
state_id: string;
description_html: string;
sort_order: number;
completed_at: string | null;
estimate_point: number | null;
priority: TIssuePriorities;
start_date: string | null;
target_date: string | null;
sequence_id: number;
project_id: string;
parent_id: string | null;
cycle_id: string | null;
module_id: string | null;
label_ids: string[];
assignee_ids: string[];
sub_issues_count: number;
created_at: string;
updated_at: string;
created_by: string;
updated_by: string;
attachment_count: number;
link_count: number;
is_subscribed: boolean;
archived_at: boolean;
is_draft: boolean;
// tempId is used for optimistic updates. It is not a part of the API response.
tempId?: string;
// issue details
related_issues: any;
issue_reactions: any;
issue_relations: any;
issue_cycle: any;
issue_module: any;
parent_detail: any;
issue_link: any;
};
export type TIssueMap = {
[issue_id: string]: TIssue;
};
export type TLoader = "init-loader" | "mutation" | undefined;
export type TGroupedIssues = {
[group_id: string]: string[];
};
export type TSubGroupedIssues = {
[sub_grouped_id: string]: {
[group_id: string]: string[];
};
};
export type TUnGroupedIssues = string[];

View File

@ -1,32 +1,41 @@
import { TIssuePriorities } from "../issues";
// new issue structure types // new issue structure types
export type TIssue = { export type TIssue = {
id: string; id: string;
sequence_id: number;
name: string; name: string;
state_id: string;
description_html: string; description_html: string;
sort_order: number; sort_order: number;
completed_at: string | null;
estimate_point: number | null; state_id: string;
priority: TIssuePriorities; priority: TIssuePriorities;
start_date: string; label_ids: string[];
target_date: string; assignee_ids: string[];
sequence_id: number; estimate_point: number | null;
sub_issues_count: number;
attachment_count: number;
link_count: number;
project_id: string; project_id: string;
parent_id: string | null; parent_id: string | null;
cycle_id: string | null; cycle_id: string | null;
module_id: string | null; module_id: string | null;
label_ids: string[];
assignee_ids: string[];
sub_issues_count: number;
created_at: string; created_at: string;
updated_at: string; updated_at: string;
start_date: string | null;
target_date: string | null;
completed_at: string | null;
archived_at: string | null;
created_by: string; created_by: string;
updated_by: string; updated_by: string;
attachment_count: number;
link_count: number;
is_subscribed: boolean;
archived_at: boolean;
is_draft: boolean; is_draft: boolean;
is_subscribed: boolean;
// tempId is used for optimistic updates. It is not a part of the API response. // tempId is used for optimistic updates. It is not a part of the API response.
tempId?: string; tempId?: string;
}; };

View File

@ -0,0 +1,27 @@
import * as React from "react";
export type TControlLink = React.AnchorHTMLAttributes<HTMLAnchorElement> & {
href: string;
onClick: () => void;
children: React.ReactNode;
target?: string;
};
export const ControlLink: React.FC<TControlLink> = (props) => {
const { href, onClick, children, target = "_self", ...rest } = props;
const LEFT_CLICK_EVENT_CODE = 0;
const _onClick = (event: React.MouseEvent<HTMLAnchorElement, MouseEvent>) => {
const clickCondition = (event.metaKey || event.ctrlKey) && event.button === LEFT_CLICK_EVENT_CODE;
if (!clickCondition) {
event.preventDefault();
onClick();
}
};
return (
<a href={href} target={target} onClick={_onClick} {...rest}>
{children}
</a>
);
};

View File

@ -0,0 +1 @@
export * from "./control-link";

View File

@ -9,3 +9,4 @@ export * from "./progress";
export * from "./spinners"; export * from "./spinners";
export * from "./tooltip"; export * from "./tooltip";
export * from "./loader"; export * from "./loader";
export * from "./control-link";

View File

@ -116,7 +116,8 @@ export const CyclesListItem: FC<TCyclesListItem> = (props) => {
if (!cycleDetails) return null; if (!cycleDetails) return null;
// computed // computed
const cycleStatus = cycleDetails.status.toLocaleLowerCase() as TCycleGroups; // TODO: change this logic once backend fix the response
const cycleStatus = cycleDetails.status ? (cycleDetails.status.toLocaleLowerCase() as TCycleGroups) : "draft";
const isCompleted = cycleStatus === "completed"; const isCompleted = cycleStatus === "completed";
const endDate = new Date(cycleDetails.end_date ?? ""); const endDate = new Date(cycleDetails.end_date ?? "");
const startDate = new Date(cycleDetails.start_date ?? ""); const startDate = new Date(cycleDetails.start_date ?? "");

View File

@ -22,15 +22,15 @@ export const IssueAttachmentRoot: FC<TIssueAttachmentRoot> = (props) => {
const { const {
router: { workspaceSlug, projectId }, router: { workspaceSlug, projectId },
} = useApplication(); } = useApplication();
const { issueId, createAttachment, removeAttachment } = useIssueDetail(); const { peekIssue, createAttachment, removeAttachment } = useIssueDetail();
const { setToastAlert } = useToast(); const { setToastAlert } = useToast();
const handleAttachmentOperations: TAttachmentOperations = useMemo( const handleAttachmentOperations: TAttachmentOperations = useMemo(
() => ({ () => ({
create: async (data: FormData) => { create: async (data: FormData) => {
try { try {
if (!workspaceSlug || !projectId || !issueId) throw new Error("Missing required fields"); if (!workspaceSlug || !projectId || !peekIssue?.issueId) throw new Error("Missing required fields");
await createAttachment(workspaceSlug, projectId, issueId, data); await createAttachment(workspaceSlug, projectId, peekIssue?.issueId, data);
setToastAlert({ setToastAlert({
message: "The attachment has been successfully uploaded", message: "The attachment has been successfully uploaded",
type: "success", type: "success",
@ -46,8 +46,8 @@ export const IssueAttachmentRoot: FC<TIssueAttachmentRoot> = (props) => {
}, },
remove: async (attachmentId: string) => { remove: async (attachmentId: string) => {
try { try {
if (!workspaceSlug || !projectId || !issueId) throw new Error("Missing required fields"); if (!workspaceSlug || !projectId || !peekIssue?.issueId) throw new Error("Missing required fields");
await removeAttachment(workspaceSlug, projectId, issueId, attachmentId); await removeAttachment(workspaceSlug, projectId, peekIssue?.issueId, attachmentId);
setToastAlert({ setToastAlert({
message: "The attachment has been successfully removed", message: "The attachment has been successfully removed",
type: "success", type: "success",
@ -62,7 +62,7 @@ export const IssueAttachmentRoot: FC<TIssueAttachmentRoot> = (props) => {
} }
}, },
}), }),
[workspaceSlug, projectId, issueId, createAttachment, removeAttachment, setToastAlert] [workspaceSlug, projectId, peekIssue, createAttachment, removeAttachment, setToastAlert]
); );
return ( return (

View File

@ -3,7 +3,7 @@ import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
import { DragDropContext, DropResult } from "@hello-pangea/dnd"; import { DragDropContext, DropResult } from "@hello-pangea/dnd";
// components // components
import { CalendarChart, IssuePeekOverview } from "components/issues"; import { CalendarChart } from "components/issues";
// hooks // hooks
import useToast from "hooks/use-toast"; import useToast from "hooks/use-toast";
// types // types
@ -34,7 +34,7 @@ export const BaseCalendarRoot = observer((props: IBaseCalendarRoot) => {
// router // router
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId, peekIssueId, peekProjectId } = router.query; const { workspaceSlug, projectId } = router.query;
// hooks // hooks
const { setToastAlert } = useToast(); const { setToastAlert } = useToast();
@ -113,16 +113,6 @@ export const BaseCalendarRoot = observer((props: IBaseCalendarRoot) => {
/> />
</DragDropContext> </DragDropContext>
</div> </div>
{workspaceSlug && peekIssueId && peekProjectId && (
<IssuePeekOverview
workspaceSlug={workspaceSlug.toString()}
projectId={peekProjectId.toString()}
issueId={peekIssueId.toString()}
handleIssue={async (issueToUpdate) =>
await handleIssues(issueToUpdate.target_date ?? "", issueToUpdate as TIssue, EIssueActions.UPDATE)
}
/>
)}
</> </>
); );
}); });

View File

@ -97,7 +97,7 @@ export const CalendarDayTile: React.FC<Props> = observer((props) => {
formKey="target_date" formKey="target_date"
groupId={formattedDatePayload} groupId={formattedDatePayload}
prePopulatedData={{ prePopulatedData={{
target_date: renderFormattedPayloadDate(date.date), target_date: renderFormattedPayloadDate(date.date) ?? undefined,
}} }}
quickAddCallback={quickAddCallback} quickAddCallback={quickAddCallback}
viewId={viewId} viewId={viewId}

View File

@ -110,11 +110,11 @@ export const CalendarQuickAddIssueForm: React.FC<Props> = observer((props) => {
}, [errors, setToastAlert]); }, [errors, setToastAlert]);
const onSubmitHandler = async (formData: TIssue) => { const onSubmitHandler = async (formData: TIssue) => {
if (isSubmitting || !groupId || !workspaceDetail || !projectDetail || !workspaceSlug || !projectId) return; if (isSubmitting || !workspaceSlug || !projectId) return;
reset({ ...defaultValues }); reset({ ...defaultValues });
const payload = createIssuePayload(workspaceDetail, projectDetail, { const payload = createIssuePayload(projectId.toString(), {
...(prePopulatedData ?? {}), ...(prePopulatedData ?? {}),
...formData, ...formData,
}); });

View File

@ -1,10 +1,10 @@
import React, { useCallback } from "react"; import React from "react";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// hooks // hooks
import { useIssues, useUser } from "hooks/store"; import { useIssues, useUser } from "hooks/store";
// components // components
import { IssueGanttBlock, IssuePeekOverview } from "components/issues"; import { IssueGanttBlock } from "components/issues";
import { import {
GanttChartRoot, GanttChartRoot,
IBlockUpdateData, IBlockUpdateData,
@ -32,10 +32,10 @@ interface IBaseGanttRoot {
} }
export const BaseGanttRoot: React.FC<IBaseGanttRoot> = observer((props: IBaseGanttRoot) => { export const BaseGanttRoot: React.FC<IBaseGanttRoot> = observer((props: IBaseGanttRoot) => {
const { issueFiltersStore, issueStore, viewId, issueActions } = props; const { issueFiltersStore, issueStore, viewId } = props;
// router // router
const router = useRouter(); const router = useRouter();
const { workspaceSlug, peekIssueId, peekProjectId } = router.query; const { workspaceSlug } = router.query;
// store hooks // store hooks
const { const {
membership: { currentProjectRole }, membership: { currentProjectRole },
@ -57,14 +57,6 @@ export const BaseGanttRoot: React.FC<IBaseGanttRoot> = observer((props: IBaseGan
await issueStore.updateIssue(workspaceSlug.toString(), issue.project_id, issue.id, payload, viewId); await issueStore.updateIssue(workspaceSlug.toString(), issue.project_id, issue.id, payload, viewId);
}; };
const handleIssues = useCallback(
async (issue: TIssue, action: EIssueActions) => {
if (issueActions[action]) {
await issueActions[action]!(issue);
}
},
[issueActions]
);
const isAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER; const isAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER;
return ( return (
@ -92,16 +84,6 @@ export const BaseGanttRoot: React.FC<IBaseGanttRoot> = observer((props: IBaseGan
enableReorder={appliedDisplayFilters?.order_by === "sort_order" && isAllowed} enableReorder={appliedDisplayFilters?.order_by === "sort_order" && isAllowed}
/> />
</div> </div>
{workspaceSlug && peekIssueId && peekProjectId && (
<IssuePeekOverview
workspaceSlug={workspaceSlug.toString()}
projectId={peekProjectId.toString()}
issueId={peekIssueId.toString()}
handleIssue={async (issueToUpdate, action) => {
await handleIssues(issueToUpdate as TIssue, action);
}}
/>
)}
</> </>
); );
}); });

View File

@ -104,14 +104,11 @@ export const GanttInlineCreateIssueForm: React.FC<Props> = observer((props) => {
const onSubmitHandler = async (formData: TIssue) => { const onSubmitHandler = async (formData: TIssue) => {
if (isSubmitting || !workspaceSlug || !projectId) return; if (isSubmitting || !workspaceSlug || !projectId) return;
// resetting the form so that user can add another issue quickly reset({ ...defaultValues });
reset({ ...defaultValues, ...(prePopulatedData ?? {}) });
const payload = createIssuePayload(workspaceDetail!, currentProjectDetails!, { const payload = createIssuePayload(projectId.toString(), {
...(prePopulatedData ?? {}), ...(prePopulatedData ?? {}),
...formData, ...formData,
start_date: renderFormattedPayloadDate(new Date()),
target_date: renderFormattedPayloadDate(new Date(new Date().getTime() + 24 * 60 * 60 * 1000)),
}); });
try { try {

View File

@ -276,14 +276,14 @@ export const BaseKanBanRoot: React.FC<IBaseKanBanLayout> = observer((props: IBas
</DragDropContext> </DragDropContext>
</div> </div>
{workspaceSlug && peekIssueId && peekProjectId && ( {/* {workspaceSlug && peekIssueId && peekProjectId && (
<IssuePeekOverview <IssuePeekOverview
workspaceSlug={workspaceSlug.toString()} workspaceSlug={workspaceSlug.toString()}
projectId={peekProjectId.toString()} projectId={peekProjectId.toString()}
issueId={peekIssueId.toString()} issueId={peekIssueId.toString()}
handleIssue={async (issueToUpdate) => await handleIssues(issueToUpdate as TIssue, EIssueActions.UPDATE)} handleIssue={async (issueToUpdate) => await handleIssues(issueToUpdate as TIssue, EIssueActions.UPDATE)}
/> />
)} )} */}
</> </>
); );
}); });

View File

@ -79,6 +79,8 @@ const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
const verticalAlignPosition = (_list: IGroupByColumn) => kanBanToggle?.groupByHeaderMinMax.includes(_list.id); const verticalAlignPosition = (_list: IGroupByColumn) => kanBanToggle?.groupByHeaderMinMax.includes(_list.id);
const isGroupByCreatedBy = group_by === "created_by";
return ( return (
<div className="relative flex h-full w-full gap-3"> <div className="relative flex h-full w-full gap-3">
{list && {list &&
@ -100,7 +102,7 @@ const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
issuePayload={_list.payload} issuePayload={_list.payload}
disableIssueCreation={disableIssueCreation} disableIssueCreation={disableIssueCreation || isGroupByCreatedBy}
currentStore={currentStore} currentStore={currentStore}
addIssuesToView={addIssuesToView} addIssuesToView={addIssuesToView}
/> />

View File

@ -9,6 +9,8 @@ import {
TUnGroupedIssues, TUnGroupedIssues,
} from "@plane/types"; } from "@plane/types";
import { EIssueActions } from "../types"; import { EIssueActions } from "../types";
// hooks
import { useProjectState } from "hooks/store";
//components //components
import { KanBanQuickAddIssueForm, KanbanIssueBlocksList } from "."; import { KanBanQuickAddIssueForm, KanbanIssueBlocksList } from ".";
@ -56,6 +58,33 @@ export const KanbanGroup = (props: IKanbanGroup) => {
viewId, viewId,
} = props; } = props;
const projectState = useProjectState();
const prePopulateQuickAddData = (groupByKey: string | null, value: string) => {
const defaultState = projectState.projectStates?.find((state) => state.default);
let preloadedData: object = { state_id: defaultState?.id };
if (groupByKey) {
if (groupByKey === "state") {
preloadedData = { ...preloadedData, state_id: value };
} else if (groupByKey === "priority") {
preloadedData = { ...preloadedData, priority: value };
} else if (groupByKey === "labels" && value != "None") {
preloadedData = { ...preloadedData, label_ids: [value] };
} else if (groupByKey === "assignees" && value != "None") {
preloadedData = { ...preloadedData, assignee_ids: [value] };
} else if (groupByKey === "created_by") {
preloadedData = { ...preloadedData };
} else {
preloadedData = { ...preloadedData, [groupByKey]: value };
}
}
return preloadedData;
};
const isGroupByCreatedBy = group_by === "created_by";
return ( return (
<div className={`${verticalPosition ? `min-h-[150px] w-[0px] overflow-hidden` : `w-full transition-all`}`}> <div className={`${verticalPosition ? `min-h-[150px] w-[0px] overflow-hidden` : `w-full transition-all`}`}>
<Droppable droppableId={`${groupId}__${sub_group_id}`}> <Droppable droppableId={`${groupId}__${sub_group_id}`}>
@ -87,13 +116,13 @@ export const KanbanGroup = (props: IKanbanGroup) => {
</Droppable> </Droppable>
<div className="sticky bottom-0 z-[0] w-full flex-shrink-0 bg-custom-background-90 py-1"> <div className="sticky bottom-0 z-[0] w-full flex-shrink-0 bg-custom-background-90 py-1">
{enableQuickIssueCreate && !disableIssueCreation && ( {enableQuickIssueCreate && !disableIssueCreation && !isGroupByCreatedBy && (
<KanBanQuickAddIssueForm <KanBanQuickAddIssueForm
formKey="name" formKey="name"
groupId={groupId} groupId={groupId}
subGroupId={sub_group_id} subGroupId={sub_group_id}
prePopulatedData={{ prePopulatedData={{
...(group_by && { [group_by]: groupId }), ...(group_by && prePopulateQuickAddData(group_by, groupId)),
...(sub_group_by && sub_group_id !== "null" && { [sub_group_by]: sub_group_id }), ...(sub_group_by && sub_group_id !== "null" && { [sub_group_by]: sub_group_id }),
}} }}
quickAddCallback={quickAddCallback} quickAddCallback={quickAddCallback}

View File

@ -4,7 +4,7 @@ import { useForm } from "react-hook-form";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
import { PlusIcon } from "lucide-react"; import { PlusIcon } from "lucide-react";
// hooks // hooks
import { useProject, useWorkspace } from "hooks/store"; import { useProject } from "hooks/store";
import useToast from "hooks/use-toast"; import useToast from "hooks/use-toast";
import useKeypress from "hooks/use-keypress"; import useKeypress from "hooks/use-keypress";
import useOutsideClickDetector from "hooks/use-outside-click-detector"; import useOutsideClickDetector from "hooks/use-outside-click-detector";
@ -59,10 +59,8 @@ export const KanBanQuickAddIssueForm: React.FC<IKanBanQuickAddIssueForm> = obser
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId } = router.query; const { workspaceSlug, projectId } = router.query;
// store hooks // store hooks
const { getWorkspaceBySlug } = useWorkspace();
const { getProjectById } = useProject(); const { getProjectById } = useProject();
const workspaceDetail = workspaceSlug ? getWorkspaceBySlug(workspaceSlug.toString()) : null;
const projectDetail = projectId ? getProjectById(projectId.toString()) : null; const projectDetail = projectId ? getProjectById(projectId.toString()) : null;
const ref = useRef<HTMLFormElement>(null); const ref = useRef<HTMLFormElement>(null);
@ -87,11 +85,11 @@ export const KanBanQuickAddIssueForm: React.FC<IKanBanQuickAddIssueForm> = obser
}, [isOpen, reset]); }, [isOpen, reset]);
const onSubmitHandler = async (formData: TIssue) => { const onSubmitHandler = async (formData: TIssue) => {
if (isSubmitting || !groupId || !workspaceDetail || !projectDetail || !workspaceSlug || !projectId) return; if (isSubmitting || !workspaceSlug || !projectId) return;
reset({ ...defaultValues }); reset({ ...defaultValues });
const payload = createIssuePayload(workspaceDetail, projectDetail, { const payload = createIssuePayload(projectId.toString(), {
...(prePopulatedData ?? {}), ...(prePopulatedData ?? {}),
...formData, ...formData,
}); });
@ -143,33 +141,6 @@ export const KanBanQuickAddIssueForm: React.FC<IKanBanQuickAddIssueForm> = obser
<span className="text-sm font-medium text-custom-primary-100">New Issue</span> <span className="text-sm font-medium text-custom-primary-100">New Issue</span>
</div> </div>
)} )}
{/* {isOpen && (
<form
ref={ref}
onSubmit={handleSubmit(onSubmitHandler)}
className="flex flex-col border-[0.5px] border-custom-border-100 justify-between gap-1.5 group/card relative select-none px-3.5 py-3 h-[118px] mb-3 mx-1.5 rounded bg-custom-background-300 shadow-custom-shadow-sm"
>
<Inputs register={register} setFocus={setFocus} projectDetails={projectDetails} />
</form>
)}
{isOpen && (
<p className="text-xs ml-3 italic mb-2 text-custom-text-200">
Press {"'"}Enter{"'"} to add another issue
</p>
)}
{!isOpen && (
<button
type="button"
className="flex items-center gap-x-[6px] text-custom-primary-100 px-2 py-3 rounded-md"
onClick={() => setIsOpen(true)}
>
<PlusIcon className="h-3.5 w-3.5 stroke-2" />
<span className="text-sm font-medium text-custom-primary-100">New Issue</span>
</button>
)} */}
</div> </div>
); );
}); });

View File

@ -1,13 +1,13 @@
import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { IssueProperties } from "../properties/all-properties"; import { IssueProperties } from "../properties/all-properties";
// hooks
import { useApplication, useIssueDetail, useProject } from "hooks/store";
// ui // ui
import { Spinner, Tooltip } from "@plane/ui"; import { Spinner, Tooltip, ControlLink } from "@plane/ui";
// types // types
import { TIssue, IIssueDisplayProperties, TIssueMap } from "@plane/types"; import { TIssue, IIssueDisplayProperties, TIssueMap } from "@plane/types";
import { EIssueActions } from "../types"; import { EIssueActions } from "../types";
import { useProject } from "hooks/store";
interface IssueBlockProps { interface IssueBlockProps {
issueId: string; issueId: string;
@ -20,27 +20,29 @@ interface IssueBlockProps {
export const IssueBlock: React.FC<IssueBlockProps> = observer((props: IssueBlockProps) => { export const IssueBlock: React.FC<IssueBlockProps> = observer((props: IssueBlockProps) => {
const { issuesMap, issueId, handleIssues, quickActions, displayProperties, canEditProperties } = props; const { issuesMap, issueId, handleIssues, quickActions, displayProperties, canEditProperties } = props;
// router // hooks
const router = useRouter(); const {
router: { workspaceSlug, projectId },
} = useApplication();
const { getProjectById } = useProject();
const { setPeekIssue } = useIssueDetail();
const updateIssue = (issueToUpdate: TIssue) => { const updateIssue = (issueToUpdate: TIssue) => {
handleIssues(issueToUpdate, EIssueActions.UPDATE); handleIssues(issueToUpdate, EIssueActions.UPDATE);
}; };
const handleIssuePeekOverview = (issue: TIssue) =>
workspaceSlug &&
issue &&
issue.project_id &&
issue.id &&
setPeekIssue({ workspaceSlug, projectId: issue.project_id, issueId: issue.id });
const issue = issuesMap[issueId]; const issue = issuesMap[issueId];
if (!issue) return null; if (!issue) return null;
const handleIssuePeekOverview = () => {
const { query } = router;
router.push({
pathname: router.pathname,
query: { ...query, peekIssueId: issue?.id, peekProjectId: issue?.project_id },
});
};
const canEditIssueProperties = canEditProperties(issue.project_id); const canEditIssueProperties = canEditProperties(issue.project_id);
const { getProjectById } = useProject();
const projectDetails = getProjectById(issue.project_id); const projectDetails = getProjectById(issue.project_id);
return ( return (
@ -55,14 +57,17 @@ export const IssueBlock: React.FC<IssueBlockProps> = observer((props: IssueBlock
{issue?.tempId !== undefined && ( {issue?.tempId !== undefined && (
<div className="absolute left-0 top-0 z-[99999] h-full w-full animate-pulse bg-custom-background-100/20" /> <div className="absolute left-0 top-0 z-[99999] h-full w-full animate-pulse bg-custom-background-100/20" />
)} )}
<Tooltip tooltipHeading="Title" tooltipContent={issue.name}>
<div <ControlLink
className="line-clamp-1 w-full cursor-pointer text-sm font-medium text-custom-text-100" href={`/${workspaceSlug}/projects/${projectId}/issues/${issueId}`}
onClick={handleIssuePeekOverview} target="_blank"
> onClick={() => handleIssuePeekOverview(issue)}
{issue.name} className="w-full line-clamp-1 cursor-pointer text-sm font-medium text-custom-text-100"
</div> >
</Tooltip> <Tooltip tooltipHeading="Title" tooltipContent={issue.name}>
<span>{issue.name}</span>
</Tooltip>
</ControlLink>
<div className="ml-auto flex flex-shrink-0 items-center gap-2"> <div className="ml-auto flex flex-shrink-0 items-center gap-2">
{!issue?.tempId ? ( {!issue?.tempId ? (

View File

@ -21,7 +21,6 @@ export interface IGroupByList {
issueIds: TGroupedIssues | TUnGroupedIssues | any; issueIds: TGroupedIssues | TUnGroupedIssues | any;
issuesMap: TIssueMap; issuesMap: TIssueMap;
group_by: string | null; group_by: string | null;
is_list?: boolean;
handleIssues: (issue: TIssue, action: EIssueActions) => Promise<void>; handleIssues: (issue: TIssue, action: EIssueActions) => Promise<void>;
quickActions: (issue: TIssue) => React.ReactNode; quickActions: (issue: TIssue) => React.ReactNode;
displayProperties: IIssueDisplayProperties | undefined; displayProperties: IIssueDisplayProperties | undefined;
@ -45,7 +44,6 @@ const GroupByList: React.FC<IGroupByList> = (props) => {
issueIds, issueIds,
issuesMap, issuesMap,
group_by, group_by,
is_list = false,
handleIssues, handleIssues,
quickActions, quickActions,
displayProperties, displayProperties,
@ -70,11 +68,27 @@ const GroupByList: React.FC<IGroupByList> = (props) => {
const prePopulateQuickAddData = (groupByKey: string | null, value: any) => { const prePopulateQuickAddData = (groupByKey: string | null, value: any) => {
const defaultState = projectState.projectStates?.find((state) => state.default); const defaultState = projectState.projectStates?.find((state) => state.default);
if (groupByKey === null) return { state_id: defaultState?.id }; let preloadedData: object = { state_id: defaultState?.id };
else {
if (groupByKey === "state") return { state: groupByKey === "state" ? value : defaultState?.id }; if (groupByKey === null) {
else return { state_id: defaultState?.id, [groupByKey]: value }; preloadedData = { ...preloadedData };
} else {
if (groupByKey === "state") {
preloadedData = { ...preloadedData, state_id: value };
} else if (groupByKey === "priority") {
preloadedData = { ...preloadedData, priority: value };
} else if (groupByKey === "labels" && value != "None") {
preloadedData = { ...preloadedData, label_ids: [value] };
} else if (groupByKey === "assignees" && value != "None") {
preloadedData = { ...preloadedData, assignee_ids: [value] };
} else if (groupByKey === "created_by") {
preloadedData = { ...preloadedData };
} else {
preloadedData = { ...preloadedData, [groupByKey]: value };
}
} }
return preloadedData;
}; };
const validateEmptyIssueGroups = (issues: TIssue[]) => { const validateEmptyIssueGroups = (issues: TIssue[]) => {
@ -83,6 +97,10 @@ const GroupByList: React.FC<IGroupByList> = (props) => {
return true; return true;
}; };
const is_list = group_by === null ? true : false;
const isGroupByCreatedBy = group_by === "created_by";
return ( return (
<div className="relative h-full w-full"> <div className="relative h-full w-full">
{list && {list &&
@ -97,7 +115,7 @@ const GroupByList: React.FC<IGroupByList> = (props) => {
title={_list.name || ""} title={_list.name || ""}
count={is_list ? issueIds?.length || 0 : issueIds?.[_list.id]?.length || 0} count={is_list ? issueIds?.length || 0 : issueIds?.[_list.id]?.length || 0}
issuePayload={_list.payload} issuePayload={_list.payload}
disableIssueCreation={disableIssueCreation} disableIssueCreation={disableIssueCreation || isGroupByCreatedBy}
currentStore={currentStore} currentStore={currentStore}
addIssuesToView={addIssuesToView} addIssuesToView={addIssuesToView}
/> />
@ -114,7 +132,7 @@ const GroupByList: React.FC<IGroupByList> = (props) => {
/> />
)} )}
{enableIssueQuickAdd && !disableIssueCreation && ( {enableIssueQuickAdd && !disableIssueCreation && !isGroupByCreatedBy && (
<div className="sticky bottom-0 z-[1] w-full flex-shrink-0"> <div className="sticky bottom-0 z-[1] w-full flex-shrink-0">
<ListQuickAddIssueForm <ListQuickAddIssueForm
prePopulatedData={prePopulateQuickAddData(group_by, _list.id)} prePopulatedData={prePopulateQuickAddData(group_by, _list.id)}

View File

@ -62,9 +62,10 @@ export const ListQuickAddIssueForm: FC<IListQuickAddIssueForm> = observer((props
// router // router
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId } = router.query; const { workspaceSlug, projectId } = router.query;
// store hooks // hooks
const { currentWorkspace } = useWorkspace(); const { getProjectById } = useProject();
const { currentProjectDetails } = useProject();
const projectDetail = (projectId && getProjectById(projectId.toString())) || undefined;
const ref = useRef<HTMLFormElement>(null); const ref = useRef<HTMLFormElement>(null);
@ -88,11 +89,11 @@ export const ListQuickAddIssueForm: FC<IListQuickAddIssueForm> = observer((props
}, [isOpen, reset]); }, [isOpen, reset]);
const onSubmitHandler = async (formData: TIssue) => { const onSubmitHandler = async (formData: TIssue) => {
if (isSubmitting || !currentWorkspace || !currentProjectDetails || !workspaceSlug || !projectId) return; if (isSubmitting || !workspaceSlug || !projectId) return;
reset({ ...defaultValues }); reset({ ...defaultValues });
const payload = createIssuePayload(currentWorkspace, currentProjectDetails, { const payload = createIssuePayload(projectId.toString(), {
...(prePopulatedData ?? {}), ...(prePopulatedData ?? {}),
...formData, ...formData,
}); });
@ -127,12 +128,7 @@ export const ListQuickAddIssueForm: FC<IListQuickAddIssueForm> = observer((props
onSubmit={handleSubmit(onSubmitHandler)} onSubmit={handleSubmit(onSubmitHandler)}
className="flex w-full items-center gap-x-3 border-[0.5px] border-t-0 border-custom-border-100 bg-custom-background-100 px-3" className="flex w-full items-center gap-x-3 border-[0.5px] border-t-0 border-custom-border-100 bg-custom-background-100 px-3"
> >
<Inputs <Inputs formKey={"name"} register={register} setFocus={setFocus} projectDetail={projectDetail ?? null} />
formKey={"name"}
register={register}
setFocus={setFocus}
projectDetail={currentProjectDetails ?? null}
/>
</form> </form>
<div className="px-3 py-2 text-xs italic text-custom-text-200">{`Press 'Enter' to add another issue`}</div> <div className="px-3 py-2 text-xs italic text-custom-text-200">{`Press 'Enter' to add another issue`}</div>
</div> </div>

View File

@ -141,8 +141,8 @@ export const IssueProperties: React.FC<IIssueProperties> = observer((props) => {
onChange={handleAssignee} onChange={handleAssignee}
disabled={isReadOnly} disabled={isReadOnly}
multiple multiple
buttonVariant={issue.assignee_ids.length > 0 ? "transparent-without-text" : "border-without-text"} buttonVariant={issue.assignee_ids?.length > 0 ? "transparent-without-text" : "border-without-text"}
buttonClassName={issue.assignee_ids.length > 0 ? "hover:bg-transparent px-0" : ""} buttonClassName={issue.assignee_ids?.length > 0 ? "hover:bg-transparent px-0" : ""}
/> />
</div> </div>
</WithDisplayPropertiesHOC> </WithDisplayPropertiesHOC>

View File

@ -1,4 +1,4 @@
import { useRouter } from "next/router"; import { FC } from "react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
import useSWR from "swr"; import useSWR from "swr";
// components // components
@ -10,21 +10,24 @@ import {
ProjectAppliedFiltersRoot, ProjectAppliedFiltersRoot,
ProjectSpreadsheetLayout, ProjectSpreadsheetLayout,
ProjectEmptyState, ProjectEmptyState,
IssuePeekOverview,
} from "components/issues"; } from "components/issues";
// ui
import { Spinner } from "@plane/ui"; import { Spinner } from "@plane/ui";
import { useIssues } from "hooks/store/use-issues";
import { EIssuesStoreType } from "constants/issue";
// hooks // hooks
import { useApplication, useIssues } from "hooks/store";
// constants
import { EIssuesStoreType } from "constants/issue";
export const ProjectLayoutRoot: React.FC = observer(() => { export const ProjectLayoutRoot: FC = observer(() => {
// router // hooks
const router = useRouter(); const {
const { workspaceSlug, projectId } = router.query as { workspaceSlug: string; projectId: string }; router: { workspaceSlug, projectId },
} = useApplication();
const { issues, issuesFilter } = useIssues(EIssuesStoreType.PROJECT); const { issues, issuesFilter } = useIssues(EIssuesStoreType.PROJECT);
useSWR( useSWR(
workspaceSlug && projectId ? `PROJECT_ISSUES_V3_${workspaceSlug}_${projectId}` : null, workspaceSlug && projectId ? `PROJECT_ISSUES_${workspaceSlug}_${projectId}` : null,
async () => { async () => {
if (workspaceSlug && projectId) { if (workspaceSlug && projectId) {
await issuesFilter?.fetchFilters(workspaceSlug, projectId); await issuesFilter?.fetchFilters(workspaceSlug, projectId);
@ -40,28 +43,35 @@ export const ProjectLayoutRoot: React.FC = observer(() => {
<div className="relative flex h-full w-full flex-col overflow-hidden"> <div className="relative flex h-full w-full flex-col overflow-hidden">
<ProjectAppliedFiltersRoot /> <ProjectAppliedFiltersRoot />
{issues?.loader === "init-loader" || !issues?.groupedIssueIds ? ( {issues?.loader === "init-loader" ? (
<div className="flex h-full w-full items-center justify-center"> <div className="flex h-full w-full items-center justify-center">
<Spinner /> <Spinner />
</div> </div>
) : ( ) : (
<> <>
{(issues?.groupedIssueIds ?? {}).length == 0 ? ( {!issues?.groupedIssueIds ? (
<ProjectEmptyState /> <div className="relative h-full w-full overflow-y-auto">
) : ( <ProjectEmptyState />
<div className="relative h-full w-full overflow-auto bg-custom-background-90">
{activeLayout === "list" ? (
<ListLayout />
) : activeLayout === "kanban" ? (
<KanBanLayout />
) : activeLayout === "calendar" ? (
<CalendarLayout />
) : activeLayout === "gantt_chart" ? (
<GanttLayout />
) : activeLayout === "spreadsheet" ? (
<ProjectSpreadsheetLayout />
) : null}
</div> </div>
) : (
<>
<div className="relative h-full w-full overflow-auto bg-custom-background-90">
{activeLayout === "list" ? (
<ListLayout />
) : activeLayout === "kanban" ? (
<KanBanLayout />
) : activeLayout === "calendar" ? (
<CalendarLayout />
) : activeLayout === "gantt_chart" ? (
<GanttLayout />
) : activeLayout === "spreadsheet" ? (
<ProjectSpreadsheetLayout />
) : null}
</div>
{/* peek overview */}
<IssuePeekOverview />
</>
)} )}
</> </>
)} )}

View File

@ -29,12 +29,15 @@ export const ProjectViewLayoutRoot: React.FC = observer(() => {
issuesFilter: { issueFilters, fetchFilters }, issuesFilter: { issueFilters, fetchFilters },
} = useIssues(EIssuesStoreType.PROJECT_VIEW); } = useIssues(EIssuesStoreType.PROJECT_VIEW);
useSWR(workspaceSlug && projectId && viewId ? `PROJECT_ISSUES_V3_${workspaceSlug}_${projectId}` : null, async () => { useSWR(
if (workspaceSlug && projectId && viewId) { workspaceSlug && projectId && viewId ? `PROJECT_VIEW_ISSUES_${workspaceSlug}_${projectId}` : null,
await fetchFilters(workspaceSlug, projectId, viewId); async () => {
await fetchIssues(workspaceSlug, projectId, groupedIssueIds ? "mutation" : "init-loader"); if (workspaceSlug && projectId && viewId) {
await fetchFilters(workspaceSlug, projectId, viewId);
await fetchIssues(workspaceSlug, projectId, groupedIssueIds ? "mutation" : "init-loader");
}
} }
}); );
const activeLayout = issueFilters?.displayFilters?.layout; const activeLayout = issueFilters?.displayFilters?.layout;

View File

@ -32,7 +32,7 @@ export const SpreadsheetAssigneeColumn: React.FC<Props> = ({ issueId, onChange,
disabled={disabled} disabled={disabled}
multiple multiple
placeholder="Assignees" placeholder="Assignees"
buttonVariant={issueDetail.assignee_ids.length > 0 ? "transparent-without-text" : "transparent-with-text"} buttonVariant={issueDetail.assignee_ids?.length > 0 ? "transparent-without-text" : "transparent-with-text"}
buttonClassName="text-left" buttonClassName="text-left"
buttonContainerClassName="w-full" buttonContainerClassName="w-full"
/> />

View File

@ -1,4 +1,5 @@
import { useEffect, useState, useRef } from "react"; import { useEffect, useState, useRef } from "react";
import { useRouter } from "next/router";
import { useForm } from "react-hook-form"; import { useForm } from "react-hook-form";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
import { PlusIcon } from "lucide-react"; import { PlusIcon } from "lucide-react";
@ -55,6 +56,10 @@ const Inputs = (props: any) => {
export const SpreadsheetQuickAddIssueForm: React.FC<Props> = observer((props) => { export const SpreadsheetQuickAddIssueForm: React.FC<Props> = observer((props) => {
const { formKey, prePopulatedData, quickAddCallback, viewId } = props; const { formKey, prePopulatedData, quickAddCallback, viewId } = props;
// router
const router = useRouter();
const { workspaceSlug, projectId } = router.query;
// store hooks // store hooks
const { currentWorkspace } = useWorkspace(); const { currentWorkspace } = useWorkspace();
const { currentProjectDetails } = useProject(); const { currentProjectDetails } = useProject();
@ -148,7 +153,7 @@ export const SpreadsheetQuickAddIssueForm: React.FC<Props> = observer((props) =>
reset({ ...defaultValues }); reset({ ...defaultValues });
const payload = createIssuePayload(currentWorkspace, currentProjectDetails, { const payload = createIssuePayload(currentProjectDetails.id, {
...(prePopulatedData ?? {}), ...(prePopulatedData ?? {}),
...formData, ...formData,
}); });

View File

@ -1,13 +1,7 @@
import React, { useEffect, useRef, useState } from "react"; import React, { useEffect, useRef, useState } from "react";
import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { import { SpreadsheetColumnsList, SpreadsheetIssuesColumn, SpreadsheetQuickAddIssueForm } from "components/issues";
IssuePeekOverview,
SpreadsheetColumnsList,
SpreadsheetIssuesColumn,
SpreadsheetQuickAddIssueForm,
} from "components/issues";
import { Spinner, LayersIcon } from "@plane/ui"; import { Spinner, LayersIcon } from "@plane/ui";
// types // types
import { TIssue, IIssueDisplayFilterOptions, IIssueDisplayProperties, IIssueLabel, IState } from "@plane/types"; import { TIssue, IIssueDisplayFilterOptions, IIssueDisplayProperties, IIssueLabel, IState } from "@plane/types";
@ -56,9 +50,6 @@ export const SpreadsheetView: React.FC<Props> = observer((props) => {
const [isScrolled, setIsScrolled] = useState(false); const [isScrolled, setIsScrolled] = useState(false);
// refs // refs
const containerRef = useRef<HTMLDivElement | null>(null); const containerRef = useRef<HTMLDivElement | null>(null);
// router
const router = useRouter();
const { workspaceSlug, peekIssueId, peekProjectId } = router.query;
const handleScroll = () => { const handleScroll = () => {
if (!containerRef.current) return; if (!containerRef.current) return;
@ -186,14 +177,6 @@ export const SpreadsheetView: React.FC<Props> = observer((props) => {
))} */} ))} */}
</div> </div>
</div> </div>
{workspaceSlug && peekIssueId && peekProjectId && (
<IssuePeekOverview
workspaceSlug={workspaceSlug.toString()}
projectId={peekProjectId.toString()}
issueId={peekIssueId.toString()}
handleIssue={async (issueToUpdate: any) => await handleIssues(issueToUpdate, EIssueActions.UPDATE)}
/>
)}
</div> </div>
); );
}); });

View File

@ -49,7 +49,7 @@ const getProjectColumns = (project: IProjectStore): IGroupByColumn[] | undefined
id: project.id, id: project.id,
name: project.name, name: project.name,
Icon: <div className="w-6 h-6">{renderEmoji(project.emoji || "")}</div>, Icon: <div className="w-6 h-6">{renderEmoji(project.emoji || "")}</div>,
payload: { project: project.id }, payload: { project_id: project.id },
}; };
}) as any; }) as any;
}; };
@ -66,7 +66,7 @@ const getStateColumns = (projectState: IStateStore): IGroupByColumn[] | undefine
<StateGroupIcon stateGroup={state.group} color={state.color} width="14" height="14" /> <StateGroupIcon stateGroup={state.group} color={state.color} width="14" height="14" />
</div> </div>
), ),
payload: { state: state.id }, payload: { state_id: state.id },
})) as any; })) as any;
}; };
@ -111,7 +111,7 @@ const getLabelsColumns = (projectLabel: ILabelRootStore) => {
Icon: ( Icon: (
<div className="w-[12px] h-[12px] rounded-full" style={{ backgroundColor: label.color ? label.color : "#666" }} /> <div className="w-[12px] h-[12px] rounded-full" style={{ backgroundColor: label.color ? label.color : "#666" }} />
), ),
payload: { labels: [label.id] }, payload: label?.id === "None" ? {} : { label_ids: [label.id] },
})); }));
}; };
@ -123,17 +123,17 @@ const getAssigneeColumns = (member: IMemberRootStore) => {
if (!projectMemberIds) return; if (!projectMemberIds) return;
const assigneeColumns = projectMemberIds.map((memberId) => { const assigneeColumns: any = projectMemberIds.map((memberId) => {
const member = getUserDetails(memberId); const member = getUserDetails(memberId);
return { return {
id: memberId, id: memberId,
name: member?.display_name || "", name: member?.display_name || "",
Icon: <Avatar name={member?.display_name} src={member?.avatar} size="md" />, Icon: <Avatar name={member?.display_name} src={member?.avatar} size="md" />,
payload: { assignees: [memberId] }, payload: { assignee_ids: [memberId] },
}; };
}); });
assigneeColumns.push({ id: "None", name: "None", Icon: <Avatar size="md" />, payload: { assignees: [""] } }); assigneeColumns.push({ id: "None", name: "None", Icon: <Avatar size="md" />, payload: {} });
return assigneeColumns; return assigneeColumns;
}; };
@ -152,7 +152,7 @@ const getCreatedByColumns = (member: IMemberRootStore) => {
id: memberId, id: memberId,
name: member?.display_name || "", name: member?.display_name || "",
Icon: <Avatar name={member?.display_name} src={member?.avatar} size="md" />, Icon: <Avatar name={member?.display_name} src={member?.avatar} size="md" />,
payload: { assignees: [memberId] }, payload: {},
}; };
}); });
}; };

View File

@ -1,5 +1,6 @@
import { FC, useState } from "react"; import { FC, useState } from "react";
// hooks // hooks
import useToast from "hooks/use-toast";
import { useIssueDetail } from "hooks/store"; import { useIssueDetail } from "hooks/store";
// ui // ui
import { ExternalLinkIcon, Tooltip } from "@plane/ui"; import { ExternalLinkIcon, Tooltip } from "@plane/ui";
@ -9,6 +10,7 @@ import { Pencil, Trash2, LinkIcon } from "lucide-react";
import { IssueLinkCreateUpdateModal, TLinkOperationsModal } from "./create-update-link-modal"; import { IssueLinkCreateUpdateModal, TLinkOperationsModal } from "./create-update-link-modal";
// helpers // helpers
import { calculateTimeAgo } from "helpers/date-time.helper"; import { calculateTimeAgo } from "helpers/date-time.helper";
import { copyTextToClipboard } from "helpers/string.helper";
export type TIssueLinkDetail = { export type TIssueLinkDetail = {
linkId: string; linkId: string;
@ -23,6 +25,8 @@ export const IssueLinkDetail: FC<TIssueLinkDetail> = (props) => {
const { const {
link: { getLinkById }, link: { getLinkById },
} = useIssueDetail(); } = useIssueDetail();
const { setToastAlert } = useToast();
// state // state
const [isIssueLinkModalOpen, setIsIssueLinkModalOpen] = useState(false); const [isIssueLinkModalOpen, setIsIssueLinkModalOpen] = useState(false);
const toggleIssueLinkModal = (modalToggle: boolean) => setIsIssueLinkModalOpen(modalToggle); const toggleIssueLinkModal = (modalToggle: boolean) => setIsIssueLinkModalOpen(modalToggle);
@ -40,18 +44,23 @@ export const IssueLinkDetail: FC<TIssueLinkDetail> = (props) => {
/> />
<div className="relative flex flex-col rounded-md bg-custom-background-90 p-2.5"> <div className="relative flex flex-col rounded-md bg-custom-background-90 p-2.5">
<div className="flex w-full items-start justify-between gap-2"> <div
className="flex w-full items-start justify-between gap-2 cursor-pointer"
onClick={() => {
copyTextToClipboard(linkDetail.url);
setToastAlert({
type: "success",
title: "Link copied!",
message: "Link copied to clipboard",
});
}}
>
<div className="flex items-start gap-2 truncate"> <div className="flex items-start gap-2 truncate">
<span className="py-1"> <span className="py-1">
<LinkIcon className="h-3 w-3 flex-shrink-0" /> <LinkIcon className="h-3 w-3 flex-shrink-0" />
</span> </span>
<Tooltip tooltipContent={linkDetail.title && linkDetail.title !== "" ? linkDetail.title : linkDetail.url}> <Tooltip tooltipContent={linkDetail.title && linkDetail.title !== "" ? linkDetail.title : linkDetail.url}>
<span <span className="truncate text-xs">
className="cursor-pointer truncate text-xs"
// onClick={() =>
// copyToClipboard(linkDetail.title && linkDetail.title !== "" ? linkDetail.title : linkDetail.url)
// }
>
{linkDetail.title && linkDetail.title !== "" ? linkDetail.title : linkDetail.url} {linkDetail.title && linkDetail.title !== "" ? linkDetail.title : linkDetail.url}
</span> </span>
</Tooltip> </Tooltip>

View File

@ -27,7 +27,7 @@ export const IssueLinkRoot: FC<TIssueLinkRoot> = (props) => {
const { const {
router: { workspaceSlug, projectId }, router: { workspaceSlug, projectId },
} = useApplication(); } = useApplication();
const { issueId, createLink, updateLink, removeLink } = useIssueDetail(); const { peekIssue, createLink, updateLink, removeLink } = useIssueDetail();
// state // state
const [isIssueLinkModalOpen, setIsIssueLinkModalOpen] = useState(false); const [isIssueLinkModalOpen, setIsIssueLinkModalOpen] = useState(false);
const toggleIssueLinkModal = (modalToggle: boolean) => setIsIssueLinkModalOpen(modalToggle); const toggleIssueLinkModal = (modalToggle: boolean) => setIsIssueLinkModalOpen(modalToggle);
@ -38,8 +38,8 @@ export const IssueLinkRoot: FC<TIssueLinkRoot> = (props) => {
() => ({ () => ({
create: async (data: Partial<TIssueLink>) => { create: async (data: Partial<TIssueLink>) => {
try { try {
if (!workspaceSlug || !projectId || !issueId) throw new Error("Missing required fields"); if (!workspaceSlug || !projectId || !peekIssue?.issueId) throw new Error("Missing required fields");
await createLink(workspaceSlug, projectId, issueId, data); await createLink(workspaceSlug, projectId, peekIssue?.issueId, data);
setToastAlert({ setToastAlert({
message: "The link has been successfully created", message: "The link has been successfully created",
type: "success", type: "success",
@ -56,8 +56,8 @@ export const IssueLinkRoot: FC<TIssueLinkRoot> = (props) => {
}, },
update: async (linkId: string, data: Partial<TIssueLink>) => { update: async (linkId: string, data: Partial<TIssueLink>) => {
try { try {
if (!workspaceSlug || !projectId || !issueId) throw new Error("Missing required fields"); if (!workspaceSlug || !projectId || !peekIssue?.issueId) throw new Error("Missing required fields");
await updateLink(workspaceSlug, projectId, issueId, linkId, data); await updateLink(workspaceSlug, projectId, peekIssue?.issueId, linkId, data);
setToastAlert({ setToastAlert({
message: "The link has been successfully updated", message: "The link has been successfully updated",
type: "success", type: "success",
@ -74,8 +74,8 @@ export const IssueLinkRoot: FC<TIssueLinkRoot> = (props) => {
}, },
remove: async (linkId: string) => { remove: async (linkId: string) => {
try { try {
if (!workspaceSlug || !projectId || !issueId) throw new Error("Missing required fields"); if (!workspaceSlug || !projectId || !peekIssue?.issueId) throw new Error("Missing required fields");
await removeLink(workspaceSlug, projectId, issueId, linkId); await removeLink(workspaceSlug, projectId, peekIssue?.issueId, linkId);
setToastAlert({ setToastAlert({
message: "The link has been successfully removed", message: "The link has been successfully removed",
type: "success", type: "success",
@ -91,7 +91,7 @@ export const IssueLinkRoot: FC<TIssueLinkRoot> = (props) => {
} }
}, },
}), }),
[workspaceSlug, projectId, issueId, createLink, updateLink, removeLink, setToastAlert] [workspaceSlug, projectId, peekIssue, createLink, updateLink, removeLink, setToastAlert]
); );
return ( return (

View File

@ -6,11 +6,17 @@ import { CalendarDays, Link2, Plus, Signal, Tag, Triangle, LayoutPanelTop } from
import { useIssueDetail, useProject, useUser } from "hooks/store"; import { useIssueDetail, useProject, useUser } from "hooks/store";
// ui icons // ui icons
import { DiceIcon, DoubleCircleIcon, UserGroupIcon, ContrastIcon } from "@plane/ui"; import { DiceIcon, DoubleCircleIcon, UserGroupIcon, ContrastIcon } from "@plane/ui";
import { SidebarCycleSelect, SidebarLabelSelect, SidebarModuleSelect, SidebarParentSelect } from "components/issues"; import {
IssueLinkRoot,
SidebarCycleSelect,
SidebarLabelSelect,
SidebarModuleSelect,
SidebarParentSelect,
} from "components/issues";
import { EstimateDropdown, PriorityDropdown, ProjectMemberDropdown, StateDropdown } from "components/dropdowns"; import { EstimateDropdown, PriorityDropdown, ProjectMemberDropdown, StateDropdown } from "components/dropdowns";
// components // components
import { CustomDatePicker } from "components/ui"; import { CustomDatePicker } from "components/ui";
import { LinkModal, LinksList } from "components/core"; import { LinkModal } from "components/core";
// types // types
import { TIssue, TIssuePriorities, ILinkDetails, IIssueLink } from "@plane/types"; import { TIssue, TIssuePriorities, ILinkDetails, IIssueLink } from "@plane/types";
// constants // constants
@ -39,6 +45,9 @@ export const PeekOverviewProperties: FC<IPeekOverviewProperties> = observer((pro
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId } = router.query; const { workspaceSlug, projectId } = router.query;
const uneditable = currentProjectRole ? [5, 10].includes(currentProjectRole) : false;
const isAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER;
const handleState = (_state: string) => { const handleState = (_state: string) => {
issueUpdate({ ...issue, state_id: _state }); issueUpdate({ ...issue, state_id: _state });
}; };
@ -274,42 +283,8 @@ export const PeekOverviewProperties: FC<IPeekOverviewProperties> = observer((pro
<span className="border-t border-custom-border-200" /> <span className="border-t border-custom-border-200" />
<div className="flex w-full flex-col gap-5 pt-5"> <div className="flex w-full flex-col gap-5 pt-5">
<div className="flex w-full flex-col gap-2"> <div className="flex flex-col gap-3">
<div className="flex w-80 items-center gap-2"> <IssueLinkRoot uneditable={uneditable} isAllowed={isAllowed} />
<div className="flex w-40 items-center gap-2 text-sm">
<Link2 className="h-4 w-4 flex-shrink-0" />
<p>Links</p>
</div>
<div>
{!disableUserActions && (
<button
type="button"
className={`flex ${
disableUserActions ? "cursor-not-allowed" : "cursor-pointer hover:bg-custom-background-90"
} items-center gap-1 rounded-2xl border border-custom-border-100 px-2 py-0.5 text-xs text-custom-text-300 hover:text-custom-text-200`}
onClick={() => toggleIssueLinkModal(true)}
disabled={false}
>
<Plus className="h-3 w-3" /> New
</button>
)}
</div>
</div>
<div className="flex flex-col gap-3">
{issue?.issue_link && issue.issue_link.length > 0 ? (
<LinksList
links={issue.issue_link}
handleDeleteLink={issueLinkDelete}
handleEditLink={handleEditLink}
userAuth={{
isGuest: currentProjectRole === EUserProjectRoles.GUEST,
isViewer: currentProjectRole === EUserProjectRoles.VIEWER,
isMember: currentProjectRole === EUserProjectRoles.MEMBER,
isOwner: currentProjectRole === EUserProjectRoles.ADMIN,
}}
/>
) : null}
</div>
</div> </div>
</div> </div>
</div> </div>

View File

@ -1,4 +1,5 @@
import { FC, Fragment, ReactNode, useCallback, useEffect } from "react"; import { FC, Fragment, useEffect, useState } from "react";
// router
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// hooks // hooks
@ -13,33 +14,28 @@ import { TIssue, IIssueLink } from "@plane/types";
// constants // constants
import { EUserProjectRoles } from "constants/project"; import { EUserProjectRoles } from "constants/project";
import { EIssuesStoreType } from "constants/issue"; import { EIssuesStoreType } from "constants/issue";
import { EIssueActions } from "../issue-layouts/types";
interface IIssuePeekOverview { interface IIssuePeekOverview {
workspaceSlug: string;
projectId: string;
issueId: string;
handleIssue: (issue: Partial<TIssue>, action: EIssueActions) => void;
isArchived?: boolean; isArchived?: boolean;
children?: ReactNode;
} }
export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => { export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
const { workspaceSlug, projectId, issueId, handleIssue, children, isArchived = false } = props; const { isArchived = false } = props;
// router // router
const router = useRouter(); const router = useRouter();
const { peekIssueId } = router.query; // hooks
// FIXME const { currentProjectDetails } = useProject();
// store hooks const { setToastAlert } = useToast();
// const {
// archivedIssueDetail: {
// getIssue: getArchivedIssue,
// loader: archivedIssueLoader,
// fetchPeekIssueDetails: fetchArchivedPeekIssueDetails,
// },
// } = useMobxStore();
const { const {
membership: { currentProjectRole },
} = useUser();
const {
issues: { removeIssue: removeArchivedIssue },
} = useIssues(EIssuesStoreType.ARCHIVED);
const {
peekIssue,
updateIssue,
removeIssue,
createComment, createComment,
updateComment, updateComment,
removeComment, removeComment,
@ -53,37 +49,38 @@ export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
updateLink, updateLink,
removeLink, removeLink,
issue: { getIssueById, fetchIssue }, issue: { getIssueById, fetchIssue },
// loader,
setIssueId,
fetchActivities, fetchActivities,
} = useIssueDetail(); } = useIssueDetail();
const { // state
issues: { removeIssue }, const [loader, setLoader] = useState(false);
} = useIssues(EIssuesStoreType.ARCHIVED);
const {
membership: { currentProjectRole },
} = useUser();
const { currentProjectDetails } = useProject();
const { setToastAlert } = useToast();
const fetchIssueDetail = useCallback(async () => {
if (workspaceSlug && projectId && peekIssueId) {
//if (isArchived) await fetchArchivedPeekIssueDetails(workspaceSlug, projectId, peekIssueId as string);
//else
await fetchIssue(workspaceSlug, projectId, peekIssueId.toString());
}
}, [fetchIssue, workspaceSlug, projectId, peekIssueId]);
useEffect(() => { useEffect(() => {
fetchIssueDetail(); if (peekIssue) {
}, [workspaceSlug, projectId, peekIssueId, fetchIssueDetail]); setLoader(true);
fetchIssue(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId).finally(() => {
setLoader(false);
});
}
}, [peekIssue, fetchIssue]);
if (!peekIssue) return <></>;
const issue = getIssueById(peekIssue.issueId) || undefined;
const redirectToIssueDetail = () => {
router.push({
pathname: `/${peekIssue.workspaceSlug}/projects/${peekIssue.projectId}/${
isArchived ? "archived-issues" : "issues"
}/${peekIssue.issueId}`,
});
};
const handleCopyText = (e: React.MouseEvent<HTMLButtonElement>) => { const handleCopyText = (e: React.MouseEvent<HTMLButtonElement>) => {
e.stopPropagation(); e.stopPropagation();
e.preventDefault(); e.preventDefault();
copyUrlToClipboard( copyUrlToClipboard(
`${workspaceSlug}/projects/${projectId}/${isArchived ? "archived-issues" : "issues"}/${peekIssueId}` `${peekIssue.workspaceSlug}/projects/${peekIssue.projectId}/${isArchived ? "archived-issues" : "issues"}/${
peekIssue.issueId
}`
).then(() => { ).then(() => {
setToastAlert({ setToastAlert({
type: "success", type: "success",
@ -93,101 +90,81 @@ export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
}); });
}; };
const redirectToIssueDetail = () => {
router.push({
pathname: `/${workspaceSlug}/projects/${projectId}/${isArchived ? "archived-issues" : "issues"}/${issueId}`,
});
};
// const issue = isArchived ? getArchivedIssue : getIssue;
// const isLoading = isArchived ? archivedIssueLoader : loader;
const issue = getIssueById(issueId);
const isLoading = false;
const issueUpdate = async (_data: Partial<TIssue>) => { const issueUpdate = async (_data: Partial<TIssue>) => {
if (handleIssue) { if (!issue) return;
await handleIssue(_data, EIssueActions.UPDATE); await updateIssue(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, _data);
fetchActivities(workspaceSlug, projectId, issueId); fetchActivities(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId);
} };
const issueDelete = async () => {
if (!issue) return;
if (isArchived) await removeArchivedIssue(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId);
else await removeIssue(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId);
}; };
const issueReactionCreate = (reaction: string) => createReaction(workspaceSlug, projectId, issueId, reaction); const issueReactionCreate = (reaction: string) =>
createReaction(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, reaction);
const issueReactionRemove = (reaction: string) =>
removeReaction(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, reaction);
const issueReactionRemove = (reaction: string) => removeReaction(workspaceSlug, projectId, issueId, reaction); const issueCommentCreate = (comment: any) =>
createComment(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, comment);
const issueCommentCreate = (comment: any) => createComment(workspaceSlug, projectId, issueId, comment); const issueCommentUpdate = (comment: any) =>
updateComment(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, comment?.id, comment);
const issueCommentUpdate = (comment: any) => updateComment(workspaceSlug, projectId, issueId, comment?.id, comment); const issueCommentRemove = (commentId: string) =>
removeComment(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, commentId);
const issueCommentRemove = (commentId: string) => removeComment(workspaceSlug, projectId, issueId, commentId);
const issueCommentReactionCreate = (commentId: string, reaction: string) => const issueCommentReactionCreate = (commentId: string, reaction: string) =>
createCommentReaction(workspaceSlug, projectId, commentId, reaction); createCommentReaction(peekIssue.workspaceSlug, peekIssue.projectId, commentId, reaction);
const issueCommentReactionRemove = (commentId: string, reaction: string) => const issueCommentReactionRemove = (commentId: string, reaction: string) =>
removeCommentReaction(workspaceSlug, projectId, commentId, reaction); removeCommentReaction(peekIssue.workspaceSlug, peekIssue.projectId, commentId, reaction);
const issueSubscriptionCreate = () => createSubscription(workspaceSlug, projectId, issueId); const issueSubscriptionCreate = () =>
createSubscription(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId);
const issueSubscriptionRemove = () => removeSubscription(workspaceSlug, projectId, issueId); const issueSubscriptionRemove = () =>
removeSubscription(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId);
const issueLinkCreate = (formData: IIssueLink) => createLink(workspaceSlug, projectId, issueId, formData);
const issueLinkCreate = (formData: IIssueLink) =>
createLink(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, formData);
const issueLinkUpdate = (formData: IIssueLink, linkId: string) => const issueLinkUpdate = (formData: IIssueLink, linkId: string) =>
updateLink(workspaceSlug, projectId, issueId, linkId, formData); updateLink(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, linkId, formData);
const issueLinkDelete = (linkId: string) =>
const issueLinkDelete = (linkId: string) => removeLink(workspaceSlug, projectId, issueId, linkId); removeLink(peekIssue.workspaceSlug, peekIssue.projectId, peekIssue.issueId, linkId);
const handleDeleteIssue = async () => {
if (!issue) return;
if (isArchived) await removeIssue(workspaceSlug, projectId, issue?.id);
// FIXME else delete...
const { query } = router;
if (query.peekIssueId) {
setIssueId(undefined);
delete query.peekIssueId;
delete query.peekProjectId;
router.push({
pathname: router.pathname,
query: { ...query },
});
}
};
const userRole = currentProjectRole ?? EUserProjectRoles.GUEST; const userRole = currentProjectRole ?? EUserProjectRoles.GUEST;
const isLoading = !issue || loader ? true : false;
return ( return (
<Fragment> <Fragment>
<IssueView {isLoading ? (
workspaceSlug={workspaceSlug} <></> // TODO: show the spinner
projectId={projectId} ) : (
issueId={issueId} <IssueView
issue={issue} workspaceSlug={peekIssue.workspaceSlug}
isLoading={isLoading} projectId={peekIssue.projectId}
isArchived={isArchived} issueId={peekIssue.issueId}
handleCopyText={handleCopyText} issue={issue}
redirectToIssueDetail={redirectToIssueDetail} isLoading={isLoading}
issueUpdate={issueUpdate} isArchived={isArchived}
issueReactionCreate={issueReactionCreate} handleCopyText={handleCopyText}
issueReactionRemove={issueReactionRemove} redirectToIssueDetail={redirectToIssueDetail}
issueCommentCreate={issueCommentCreate} issueUpdate={issueUpdate}
issueCommentUpdate={issueCommentUpdate} issueReactionCreate={issueReactionCreate}
issueCommentRemove={issueCommentRemove} issueReactionRemove={issueReactionRemove}
issueCommentReactionCreate={issueCommentReactionCreate} issueCommentCreate={issueCommentCreate}
issueCommentReactionRemove={issueCommentReactionRemove} issueCommentUpdate={issueCommentUpdate}
issueSubscriptionCreate={issueSubscriptionCreate} issueCommentRemove={issueCommentRemove}
issueSubscriptionRemove={issueSubscriptionRemove} issueCommentReactionCreate={issueCommentReactionCreate}
issueLinkCreate={issueLinkCreate} issueCommentReactionRemove={issueCommentReactionRemove}
issueLinkUpdate={issueLinkUpdate} issueSubscriptionCreate={issueSubscriptionCreate}
issueLinkDelete={issueLinkDelete} issueSubscriptionRemove={issueSubscriptionRemove}
handleDeleteIssue={handleDeleteIssue} issueLinkCreate={issueLinkCreate}
disableUserActions={[5, 10].includes(userRole)} issueLinkUpdate={issueLinkUpdate}
showCommentAccessSpecifier={currentProjectDetails?.is_deployed} issueLinkDelete={issueLinkDelete}
> handleDeleteIssue={issueDelete}
{children} disableUserActions={[5, 10].includes(userRole)}
</IssueView> showCommentAccessSpecifier={currentProjectDetails?.is_deployed}
/>
)}
</Fragment> </Fragment>
); );
}); });

View File

@ -1,7 +1,5 @@
import { FC, ReactNode, useRef, useState } from "react"; import { FC, useRef, useState } from "react";
import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
import useSWR from "swr";
import { MoveRight, MoveDiagonal, Bell, Link2, Trash2 } from "lucide-react"; import { MoveRight, MoveDiagonal, Bell, Link2, Trash2 } from "lucide-react";
// hooks // hooks
import { useIssueDetail, useUser } from "hooks/store"; import { useIssueDetail, useUser } from "hooks/store";
@ -43,7 +41,6 @@ interface IIssueView {
issueLinkUpdate: (formData: IIssueLink, linkId: string) => Promise<ILinkDetails>; issueLinkUpdate: (formData: IIssueLink, linkId: string) => Promise<ILinkDetails>;
issueLinkDelete: (linkId: string) => Promise<void>; issueLinkDelete: (linkId: string) => Promise<void>;
handleDeleteIssue: () => Promise<void>; handleDeleteIssue: () => Promise<void>;
children: ReactNode;
disableUserActions?: boolean; disableUserActions?: boolean;
showCommentAccessSpecifier?: boolean; showCommentAccessSpecifier?: boolean;
} }
@ -92,7 +89,6 @@ export const IssueView: FC<IIssueView> = observer((props) => {
issueLinkUpdate, issueLinkUpdate,
issueLinkDelete, issueLinkDelete,
handleDeleteIssue, handleDeleteIssue,
children,
disableUserActions = false, disableUserActions = false,
showCommentAccessSpecifier = false, showCommentAccessSpecifier = false,
} = props; } = props;
@ -101,58 +97,19 @@ export const IssueView: FC<IIssueView> = observer((props) => {
const [isSubmitting, setIsSubmitting] = useState<"submitting" | "submitted" | "saved">("saved"); const [isSubmitting, setIsSubmitting] = useState<"submitting" | "submitted" | "saved">("saved");
// ref // ref
const issuePeekOverviewRef = useRef<HTMLDivElement>(null); const issuePeekOverviewRef = useRef<HTMLDivElement>(null);
// router
const router = useRouter();
const { peekIssueId } = router.query;
// store hooks // store hooks
const { const {
fetchSubscriptions,
activity, activity,
reaction, reaction,
subscription, subscription,
setIssueId, setPeekIssue,
isAnyModalOpen, isAnyModalOpen,
isDeleteIssueModalOpen, isDeleteIssueModalOpen,
toggleDeleteIssueModal, toggleDeleteIssueModal,
} = useIssueDetail(); } = useIssueDetail();
const { currentUser } = useUser(); const { currentUser } = useUser();
const updateRoutePeekId = () => { const removeRoutePeekId = () => setPeekIssue(undefined);
if (issueId != peekIssueId) {
setIssueId(issueId);
const { query } = router;
router.push({
pathname: router.pathname,
query: { ...query, peekIssueId: issueId, peekProjectId: projectId },
});
}
};
const removeRoutePeekId = () => {
const { query } = router;
if (query.peekIssueId) {
setIssueId(undefined);
delete query.peekIssueId;
delete query.peekProjectId;
router.push({
pathname: router.pathname,
query: { ...query },
});
}
};
useSWR(
workspaceSlug && projectId && issueId && peekIssueId && issueId === peekIssueId
? `ISSUE_PEEK_OVERVIEW_SUBSCRIPTION_${workspaceSlug}_${projectId}_${peekIssueId}`
: null,
async () => {
if (workspaceSlug && projectId && issueId && peekIssueId && issueId === peekIssueId) {
await fetchSubscriptions(workspaceSlug, projectId, issueId);
}
}
);
const issueReactions = reaction.getReactionsByIssueId(issueId) || []; const issueReactions = reaction.getReactionsByIssueId(issueId) || [];
const issueActivity = activity.getActivitiesByIssueId(issueId); const issueActivity = activity.getActivitiesByIssueId(issueId);
@ -172,6 +129,7 @@ export const IssueView: FC<IIssueView> = observer((props) => {
onSubmit={handleDeleteIssue} onSubmit={handleDeleteIssue}
/> />
)} )}
{issue && isArchived && ( {issue && isArchived && (
<DeleteArchivedIssueModal <DeleteArchivedIssueModal
data={issue} data={issue}
@ -180,14 +138,9 @@ export const IssueView: FC<IIssueView> = observer((props) => {
onSubmit={handleDeleteIssue} onSubmit={handleDeleteIssue}
/> />
)} )}
<div className="w-full truncate !text-base">
{children && (
<div onClick={updateRoutePeekId} className="w-full cursor-pointer">
{children}
</div>
)}
{issueId === peekIssueId && ( <div className="w-full truncate !text-base">
{issueId && (
<div <div
ref={issuePeekOverviewRef} ref={issuePeekOverviewRef}
className={`fixed z-20 flex flex-col overflow-hidden rounded border border-custom-border-200 bg-custom-background-100 transition-all duration-300 className={`fixed z-20 flex flex-col overflow-hidden rounded border border-custom-border-200 bg-custom-background-100 transition-all duration-300
@ -248,7 +201,7 @@ export const IssueView: FC<IIssueView> = observer((props) => {
<div className="flex items-center gap-4"> <div className="flex items-center gap-4">
{issue?.created_by !== currentUser?.id && {issue?.created_by !== currentUser?.id &&
!issue?.assignee_ids.includes(currentUser?.id ?? "") && !issue?.assignee_ids.includes(currentUser?.id ?? "") &&
!router.pathname.includes("[archivedIssueId]") && ( !issue?.archived_at && (
<Button <Button
size="sm" size="sm"
prependIcon={<Bell className="h-3 w-3" />} prependIcon={<Bell className="h-3 w-3" />}

View File

@ -2,7 +2,7 @@ import React, { useState } from "react";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import useSWR, { mutate } from "swr"; import useSWR, { mutate } from "swr";
// hooks // hooks
import { useIssues } from "hooks/store"; import { useCycle, useIssues } from "hooks/store";
// services // services
import { CycleService } from "services/cycle.service"; import { CycleService } from "services/cycle.service";
// ui // ui
@ -32,6 +32,7 @@ export const SidebarCycleSelect: React.FC<Props> = (props) => {
const { const {
issues: { removeIssueFromCycle, addIssueToCycle }, issues: { removeIssueFromCycle, addIssueToCycle },
} = useIssues(EIssuesStoreType.CYCLE); } = useIssues(EIssuesStoreType.CYCLE);
const { getCycleById } = useCycle();
const [isUpdating, setIsUpdating] = useState(false); const [isUpdating, setIsUpdating] = useState(false);
@ -87,17 +88,17 @@ export const SidebarCycleSelect: React.FC<Props> = (props) => {
), ),
})); }));
const issueCycle = issueDetail?.issue_cycle; const issueCycle = (issueDetail && issueDetail.cycle_id && getCycleById(issueDetail.cycle_id)) || undefined;
const disableSelect = disabled || isUpdating; const disableSelect = disabled || isUpdating;
return ( return (
<div className="flex items-center gap-1"> <div className="flex items-center gap-1">
<CustomSearchSelect <CustomSearchSelect
value={issueCycle?.cycle_detail.id} value={issueDetail?.cycle_id}
onChange={(value: any) => { onChange={(value: any) => {
value === issueCycle?.cycle_detail.id value === issueDetail?.cycle_id
? handleRemoveIssueFromCycle(issueCycle?.cycle ?? "") ? handleRemoveIssueFromCycle(issueDetail?.cycle_id ?? "")
: handleCycleChange : handleCycleChange
? handleCycleChange(value) ? handleCycleChange(value)
: handleCycleStoreChange(value); : handleCycleStoreChange(value);
@ -105,7 +106,7 @@ export const SidebarCycleSelect: React.FC<Props> = (props) => {
options={options} options={options}
customButton={ customButton={
<div> <div>
<Tooltip position="left" tooltipContent={`${issueCycle ? issueCycle.cycle_detail.name : "No cycle"}`}> <Tooltip position="left" tooltipContent={`${issueCycle ? issueCycle?.name : "No cycle"}`}>
<button <button
type="button" type="button"
className={`flex w-full items-center rounded bg-custom-background-80 px-2.5 py-0.5 text-xs ${ className={`flex w-full items-center rounded bg-custom-background-80 px-2.5 py-0.5 text-xs ${
@ -118,7 +119,7 @@ export const SidebarCycleSelect: React.FC<Props> = (props) => {
}`} }`}
> >
<span className="flex-shrink-0">{issueCycle && <ContrastIcon className="h-3.5 w-3.5" />}</span> <span className="flex-shrink-0">{issueCycle && <ContrastIcon className="h-3.5 w-3.5" />}</span>
<span className="truncate">{issueCycle ? issueCycle.cycle_detail.name : "No cycle"}</span> <span className="truncate">{issueCycle ? issueCycle?.name : "No cycle"}</span>
</span> </span>
</button> </button>
</Tooltip> </Tooltip>

View File

@ -81,17 +81,17 @@ export const SidebarModuleSelect: React.FC<Props> = observer((props) => {
}); });
// derived values // derived values
const issueModule = issueDetail?.issue_module; const issueModule = (issueDetail && issueDetail?.module_id && getModuleById(issueDetail.module_id)) || undefined;
const selectedModule = issueModule?.module ? getModuleById(issueModule?.module) : null;
const disableSelect = disabled || isUpdating; const disableSelect = disabled || isUpdating;
return ( return (
<div className="flex items-center gap-1"> <div className="flex items-center gap-1">
<CustomSearchSelect <CustomSearchSelect
value={issueModule?.module_detail.id} value={issueDetail?.module_id}
onChange={(value: any) => { onChange={(value: any) => {
value === issueModule?.module_detail.id value === issueDetail?.module_id
? handleRemoveIssueFromModule(issueModule?.id ?? "", issueModule?.module ?? "") ? handleRemoveIssueFromModule(issueModule?.id ?? "", issueDetail?.module_id ?? "")
: handleModuleChange : handleModuleChange
? handleModuleChange(value) ? handleModuleChange(value)
: handleModuleStoreChange(value); : handleModuleStoreChange(value);
@ -99,7 +99,7 @@ export const SidebarModuleSelect: React.FC<Props> = observer((props) => {
options={options} options={options}
customButton={ customButton={
<div> <div>
<Tooltip position="left" tooltipContent={`${selectedModule?.name ?? "No module"}`}> <Tooltip position="left" tooltipContent={`${issueModule?.name ?? "No module"}`}>
<button <button
type="button" type="button"
className={`flex w-full items-center rounded bg-custom-background-80 px-2.5 py-0.5 text-xs ${ className={`flex w-full items-center rounded bg-custom-background-80 px-2.5 py-0.5 text-xs ${
@ -112,7 +112,7 @@ export const SidebarModuleSelect: React.FC<Props> = observer((props) => {
}`} }`}
> >
<span className="flex-shrink-0">{issueModule && <DiceIcon className="h-3.5 w-3.5" />}</span> <span className="flex-shrink-0">{issueModule && <DiceIcon className="h-3.5 w-3.5" />}</span>
<span className="truncate">{selectedModule?.name ?? "No module"}</span> <span className="truncate">{issueModule?.name ?? "No module"}</span>
</span> </span>
</button> </button>
</Tooltip> </Tooltip>

View File

@ -2,13 +2,14 @@ import React, { useState } from "react";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
// hooks // hooks
import { useIssueDetail, useProject } from "hooks/store"; import { useIssueDetail, useIssues, useProject } from "hooks/store";
// components // components
import { ParentIssuesListModal } from "components/issues"; import { ParentIssuesListModal } from "components/issues";
// icons // icons
import { X } from "lucide-react"; import { X } from "lucide-react";
// types // types
import { TIssue, ISearchIssueResponse } from "@plane/types"; import { TIssue, ISearchIssueResponse } from "@plane/types";
import { observer } from "mobx-react-lite";
type Props = { type Props = {
onChange: (value: string) => void; onChange: (value: string) => void;
@ -16,7 +17,7 @@ type Props = {
disabled?: boolean; disabled?: boolean;
}; };
export const SidebarParentSelect: React.FC<Props> = ({ onChange, issueDetails, disabled = false }) => { export const SidebarParentSelect: React.FC<Props> = observer(({ onChange, issueDetails, disabled = false }) => {
const [selectedParentIssue, setSelectedParentIssue] = useState<ISearchIssueResponse | null>(null); const [selectedParentIssue, setSelectedParentIssue] = useState<ISearchIssueResponse | null>(null);
const { isParentIssueModalOpen, toggleParentIssueModal } = useIssueDetail(); const { isParentIssueModalOpen, toggleParentIssueModal } = useIssueDetail();
@ -26,6 +27,7 @@ export const SidebarParentSelect: React.FC<Props> = ({ onChange, issueDetails, d
// hooks // hooks
const { getProjectById } = useProject(); const { getProjectById } = useProject();
const { issueMap } = useIssues();
return ( return (
<> <>
@ -56,7 +58,7 @@ export const SidebarParentSelect: React.FC<Props> = ({ onChange, issueDetails, d
{selectedParentIssue && issueDetails?.parent_id ? ( {selectedParentIssue && issueDetails?.parent_id ? (
`${selectedParentIssue.project__identifier}-${selectedParentIssue.sequence_id}` `${selectedParentIssue.project__identifier}-${selectedParentIssue.sequence_id}`
) : !selectedParentIssue && issueDetails?.parent_id ? ( ) : !selectedParentIssue && issueDetails?.parent_id ? (
`${getProjectById(issueDetails.parent_id)?.identifier}-${issueDetails.parent_detail?.sequence_id}` `${getProjectById(issueDetails.parent_id)?.identifier}-${issueMap[issueDetails.parent_id]?.sequence_id}`
) : ( ) : (
<span className="text-custom-text-200">Select issue</span> <span className="text-custom-text-200">Select issue</span>
)} )}
@ -64,4 +66,4 @@ export const SidebarParentSelect: React.FC<Props> = ({ onChange, issueDetails, d
</button> </button>
</> </>
); );
}; });

View File

@ -1,10 +1,8 @@
import { useRouter } from "next/router";
import React from "react"; import React from "react";
import { ChevronDown, ChevronRight, X, Pencil, Trash, Link as LinkIcon, Loader } from "lucide-react"; import { ChevronDown, ChevronRight, X, Pencil, Trash, Link as LinkIcon, Loader } from "lucide-react";
// components // components
import { SubIssuesRootList } from "./issues-list"; import { SubIssuesRootList } from "./issues-list";
import { IssueProperty } from "./properties"; import { IssueProperty } from "./properties";
import { IssuePeekOverview } from "components/issues";
// ui // ui
import { CustomMenu, Tooltip } from "@plane/ui"; import { CustomMenu, Tooltip } from "@plane/ui";
// types // types
@ -42,7 +40,6 @@ export const SubIssues: React.FC<ISubIssues> = ({
projectId, projectId,
parentIssue, parentIssue,
issueId, issueId,
handleIssue,
spacingLeft = 0, spacingLeft = 0,
user, user,
editable, editable,
@ -53,9 +50,6 @@ export const SubIssues: React.FC<ISubIssues> = ({
handleIssueCrudOperation, handleIssueCrudOperation,
handleUpdateIssue, handleUpdateIssue,
}) => { }) => {
const router = useRouter();
const { peekProjectId, peekIssueId } = router.query;
const { const {
issue: { getIssueById }, issue: { getIssueById },
} = useIssueDetail(); } = useIssueDetail();
@ -68,25 +62,8 @@ export const SubIssues: React.FC<ISubIssues> = ({
(issue?.project_id && getProjectStates(issue?.project_id)?.find((state) => issue?.state_id == state.id)) || (issue?.project_id && getProjectStates(issue?.project_id)?.find((state) => issue?.state_id == state.id)) ||
undefined; undefined;
const handleIssuePeekOverview = () => {
const { query } = router;
router.push({
pathname: router.pathname,
query: { ...query, peekIssueId: issue?.id, peekProjectId: issue?.project_id },
});
};
return ( return (
<> <>
{workspaceSlug && peekProjectId && peekIssueId && peekIssueId === issue?.id && (
<IssuePeekOverview
workspaceSlug={workspaceSlug}
projectId={peekProjectId.toString()}
issueId={peekIssueId.toString()}
handleIssue={async (issueToUpdate) => await handleUpdateIssue(issue, { ...issue, ...issueToUpdate })}
/>
)}
<div> <div>
{issue && ( {issue && (
<div <div
@ -116,7 +93,7 @@ export const SubIssues: React.FC<ISubIssues> = ({
)} )}
</div> </div>
<div className="flex w-full cursor-pointer items-center gap-2" onClick={handleIssuePeekOverview}> <div className="flex w-full cursor-pointer items-center gap-2">
<div <div
className="h-[6px] w-[6px] flex-shrink-0 rounded-full" className="h-[6px] w-[6px] flex-shrink-0 rounded-full"
style={{ style={{

View File

@ -2,15 +2,7 @@ import { v4 as uuidv4 } from "uuid";
// helpers // helpers
import { orderArrayBy } from "helpers/array.helper"; import { orderArrayBy } from "helpers/array.helper";
// types // types
import { import { TIssue, TIssueGroupByOptions, TIssueLayouts, TIssueOrderByOptions, TIssueParams } from "@plane/types";
TIssue,
TIssueGroupByOptions,
TIssueLayouts,
TIssueOrderByOptions,
TIssueParams,
IProject,
IWorkspace,
} from "@plane/types";
// constants // constants
import { ISSUE_DISPLAY_FILTERS_BY_LAYOUT } from "constants/issue"; import { ISSUE_DISPLAY_FILTERS_BY_LAYOUT } from "constants/issue";
@ -123,76 +115,21 @@ export const handleIssueQueryParamsByLayout = (
* *
* @description create a full issue payload with some default values. This function also parse the form field * @description create a full issue payload with some default values. This function also parse the form field
* like assignees, labels, etc. and add them to the payload * like assignees, labels, etc. and add them to the payload
* @param workspaceDetail workspace detail to be added in the issue payload * @param projectId project id to be added in the issue payload
* @param projectDetail project detail to be added in the issue payload
* @param formData partial issue data from the form. This will override the default values * @param formData partial issue data from the form. This will override the default values
* @returns full issue payload with some default values * @returns full issue payload with some default values
*/ */
export const createIssuePayload: (projectId: string, formData: Partial<TIssue>) => TIssue = (
export const createIssuePayload: ( projectId: string,
workspaceDetail: IWorkspace,
projectDetail: IProject,
formData: Partial<TIssue> formData: Partial<TIssue>
) => TIssue = (workspaceDetail: IWorkspace, projectDetail: IProject, formData: Partial<TIssue>) => { ) => {
const payload = { const payload: TIssue = {
archived_at: null,
assignee_details: [],
attachment_count: 0,
attachments: [],
issue_relations: [],
related_issues: [],
bridge_id: null,
completed_at: new Date(),
created_at: "",
created_by: "",
cycle: null,
cycle_id: null,
cycle_detail: null,
description: {},
description_html: "",
description_stripped: "",
estimate_point: null,
issue_cycle: null,
issue_link: [],
issue_module: null,
label_details: [],
is_draft: false,
links_list: [],
link_count: 0,
module: null,
module_id: null,
name: "",
parent: null,
parent_detail: null,
priority: "none",
project: projectDetail.id,
project_detail: projectDetail,
sequence_id: 0,
sort_order: 0,
sprints: null,
start_date: null,
state: projectDetail.default_state,
state_detail: {} as any,
sub_issues_count: 0,
target_date: null,
updated_at: "",
updated_by: "",
workspace: workspaceDetail.id,
workspace_detail: workspaceDetail,
id: uuidv4(), id: uuidv4(),
project_id: projectId,
// tempId is used for optimistic updates. It is not a part of the API response.
tempId: uuidv4(), tempId: uuidv4(),
// to be overridden by the form data // to be overridden by the form data
...formData, ...formData,
assignee_ids: Array.isArray(formData.assignee_ids)
? formData.assignee_ids
: formData.assignee_ids && formData.assignee_ids !== "none" && formData.assignee_ids !== null
? [formData.assignee_ids]
: [],
label_ids: Array.isArray(formData.label_ids)
? formData.label_ids
: formData.label_ids && formData.label_ids !== "none" && formData.label_ids !== null
? [formData.label_ids]
: [],
} as TIssue; } as TIssue;
return payload; return payload;

View File

@ -47,7 +47,7 @@ export const AppProvider: FC<IAppProvider> = observer((props) => {
<InstanceLayout> <InstanceLayout>
<StoreWrapper> <StoreWrapper>
<CrispWrapper user={currentUser}> <CrispWrapper user={currentUser}>
<PosthogWrapper {/* <PosthogWrapper
user={currentUser} user={currentUser}
workspaceRole={currentWorkspaceRole} workspaceRole={currentWorkspaceRole}
projectRole={currentProjectRole} projectRole={currentProjectRole}
@ -55,7 +55,8 @@ export const AppProvider: FC<IAppProvider> = observer((props) => {
posthogHost={envConfig?.posthog_host || null} posthogHost={envConfig?.posthog_host || null}
> >
<SWRConfig value={SWR_CONFIG}>{children}</SWRConfig> <SWRConfig value={SWR_CONFIG}>{children}</SWRConfig>
</PosthogWrapper> </PosthogWrapper> */}
<SWRConfig value={SWR_CONFIG}>{children}</SWRConfig>
</CrispWrapper> </CrispWrapper>
</StoreWrapper> </StoreWrapper>
</InstanceLayout> </InstanceLayout>

View File

@ -30,8 +30,8 @@ const defaultValues: Partial<TIssue> = {
state_id: "", state_id: "",
priority: "low", priority: "low",
target_date: new Date().toString(), target_date: new Date().toString(),
issue_cycle: null, cycle_id: null,
issue_module: null, module_id: null,
}; };
// services // services

View File

@ -26,8 +26,8 @@ const defaultValues: Partial<TIssue> = {
// description: "", // description: "",
description_html: "", description_html: "",
estimate_point: null, estimate_point: null,
issue_cycle: null, cycle_id: null,
issue_module: null, module_id: null,
name: "", name: "",
priority: "low", priority: "low",
start_date: undefined, start_date: undefined,
@ -43,7 +43,7 @@ const IssueDetailsPage: NextPageWithLayout = observer(() => {
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId, issueId: routeIssueId } = router.query; const { workspaceSlug, projectId, issueId: routeIssueId } = router.query;
const { issueId, fetchIssue } = useIssueDetail(); const { peekIssue, fetchIssue } = useIssueDetail();
useEffect(() => { useEffect(() => {
if (!workspaceSlug || !projectId || !routeIssueId) return; if (!workspaceSlug || !projectId || !routeIssueId) return;
fetchIssue(workspaceSlug as string, projectId as string, routeIssueId as string); fetchIssue(workspaceSlug as string, projectId as string, routeIssueId as string);
@ -54,9 +54,9 @@ const IssueDetailsPage: NextPageWithLayout = observer(() => {
mutate: mutateIssueDetails, mutate: mutateIssueDetails,
error, error,
} = useSWR( } = useSWR(
workspaceSlug && projectId && issueId ? ISSUE_DETAILS(issueId as string) : null, workspaceSlug && projectId && peekIssue?.issueId ? ISSUE_DETAILS(peekIssue?.issueId as string) : null,
workspaceSlug && projectId && issueId workspaceSlug && projectId && peekIssue?.issueId
? () => issueService.retrieve(workspaceSlug as string, projectId as string, issueId as string) ? () => issueService.retrieve(workspaceSlug as string, projectId as string, peekIssue?.issueId as string)
: null : null
); );
@ -66,10 +66,10 @@ const IssueDetailsPage: NextPageWithLayout = observer(() => {
const submitChanges = useCallback( const submitChanges = useCallback(
async (formData: Partial<TIssue>) => { async (formData: Partial<TIssue>) => {
if (!workspaceSlug || !projectId || !issueId) return; if (!workspaceSlug || !projectId || !peekIssue?.issueId) return;
mutate<TIssue>( mutate<TIssue>(
ISSUE_DETAILS(issueId as string), ISSUE_DETAILS(peekIssue?.issueId as string),
(prevData) => { (prevData) => {
if (!prevData) return prevData; if (!prevData) return prevData;
@ -85,30 +85,30 @@ const IssueDetailsPage: NextPageWithLayout = observer(() => {
...formData, ...formData,
}; };
delete payload.related_issues; // delete payload.related_issues;
delete payload.issue_relations; // delete payload.issue_relations;
await issueService await issueService
.patchIssue(workspaceSlug as string, projectId as string, issueId as string, payload) .patchIssue(workspaceSlug as string, projectId as string, peekIssue?.issueId as string, payload)
.then(() => { .then(() => {
mutateIssueDetails(); mutateIssueDetails();
mutate(PROJECT_ISSUES_ACTIVITY(issueId as string)); mutate(PROJECT_ISSUES_ACTIVITY(peekIssue?.issueId as string));
}) })
.catch((e) => { .catch((e) => {
console.error(e); console.error(e);
}); });
}, },
[workspaceSlug, issueId, projectId, mutateIssueDetails] [workspaceSlug, peekIssue?.issueId, projectId, mutateIssueDetails]
); );
useEffect(() => { useEffect(() => {
if (!issueDetails) return; if (!issueDetails) return;
mutate(PROJECT_ISSUES_ACTIVITY(issueId as string)); mutate(PROJECT_ISSUES_ACTIVITY(peekIssue?.issueId as string));
reset({ reset({
...issueDetails, ...issueDetails,
}); });
}, [issueDetails, reset, issueId]); }, [issueDetails, reset, peekIssue?.issueId]);
return ( return (
<> <>
@ -123,7 +123,7 @@ const IssueDetailsPage: NextPageWithLayout = observer(() => {
onClick: () => router.push(`/${workspaceSlug}/projects/${projectId}/issues`), onClick: () => router.push(`/${workspaceSlug}/projects/${projectId}/issues`),
}} }}
/> />
) : issueDetails && projectId && issueId ? ( ) : issueDetails && projectId && peekIssue?.issueId ? (
<div className="flex h-full overflow-hidden"> <div className="flex h-full overflow-hidden">
<div className="h-full w-2/3 space-y-5 divide-y-2 divide-custom-border-300 overflow-y-auto p-5"> <div className="h-full w-2/3 space-y-5 divide-y-2 divide-custom-border-300 overflow-y-auto p-5">
<IssueMainContent issueDetails={issueDetails} submitChanges={submitChanges} /> <IssueMainContent issueDetails={issueDetails} submitChanges={submitChanges} />

View File

@ -18,7 +18,6 @@ import { AppLayout } from "layouts/app-layout";
// components // components
import { GptAssistantPopover } from "components/core"; import { GptAssistantPopover } from "components/core";
import { PageDetailsHeader } from "components/headers/page-details"; import { PageDetailsHeader } from "components/headers/page-details";
import { IssuePeekOverview } from "components/issues/peek-overview";
import { EmptyState } from "components/common"; import { EmptyState } from "components/common";
// ui // ui
import { DocumentEditorWithRef, DocumentReadOnlyEditorWithRef } from "@plane/document-editor"; import { DocumentEditorWithRef, DocumentReadOnlyEditorWithRef } from "@plane/document-editor";
@ -49,7 +48,7 @@ const PageDetailsPage: NextPageWithLayout = observer(() => {
const editorRef = useRef<any>(null); const editorRef = useRef<any>(null);
// router // router
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId, pageId, peekIssueId } = router.query; const { workspaceSlug, projectId, pageId } = router.query;
// store hooks // store hooks
const { const {
issues: { updateIssue }, issues: { updateIssue },
@ -108,12 +107,6 @@ const PageDetailsPage: NextPageWithLayout = observer(() => {
} }
); );
const handleUpdateIssue = (issueId: string, data: Partial<TIssue>) => {
if (!workspaceSlug || !projectId || !currentUser) return;
updateIssue(workspaceSlug.toString(), projectId.toString(), issueId, data);
};
const fetchIssue = async (issueId: string) => { const fetchIssue = async (issueId: string) => {
const issue = await issueService.retrieve(workspaceSlug as string, projectId as string, issueId as string); const issue = await issueService.retrieve(workspaceSlug as string, projectId as string, issueId as string);
return issue as TIssue; return issue as TIssue;
@ -523,17 +516,6 @@ const PageDetailsPage: NextPageWithLayout = observer(() => {
)} )}
</div> </div>
)} )}
<IssuePeekOverview
workspaceSlug={workspaceSlug as string}
projectId={projectId as string}
issueId={peekIssueId ? (peekIssueId as string) : ""}
isArchived={false}
handleIssue={(issueToUpdate) => {
if (peekIssueId && typeof peekIssueId === "string") {
handleUpdateIssue(peekIssueId, issueToUpdate);
}
}}
/>
</div> </div>
</div> </div>
) : ( ) : (

View File

@ -158,16 +158,23 @@ export class ArchivedIssuesFilter extends IssueFilterHelperStore implements IArc
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -145,16 +145,23 @@ export class CycleIssuesFilter extends IssueFilterHelperStore implements ICycleI
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -142,16 +142,23 @@ export class DraftIssuesFilter extends IssueFilterHelperStore implements IDraftI
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -76,7 +76,10 @@ export class IssueHelperStore implements TIssueHelperStore {
const state_group = const state_group =
this.rootStore?.stateDetails?.find((_state) => _state.id === _issue?.state_id)?.group || "None"; this.rootStore?.stateDetails?.find((_state) => _state.id === _issue?.state_id)?.group || "None";
groupArray = [state_group]; groupArray = [state_group];
} else groupArray = this.getGroupArray(get(_issue, ISSUE_FILTER_DEFAULT_DATA[groupBy]), isCalendarIssues); } else {
const groupValue = get(_issue, ISSUE_FILTER_DEFAULT_DATA[groupBy]);
groupArray = groupValue !== undefined ? this.getGroupArray(groupValue, isCalendarIssues) : [];
}
for (const group of groupArray) { for (const group of groupArray) {
if (group && _issues[group]) _issues[group].push(_issue.id); if (group && _issues[group]) _issues[group].push(_issue.id);
@ -116,8 +119,10 @@ export class IssueHelperStore implements TIssueHelperStore {
subGroupArray = [state_group]; subGroupArray = [state_group];
groupArray = [state_group]; groupArray = [state_group];
} else { } else {
subGroupArray = this.getGroupArray(get(_issue, ISSUE_FILTER_DEFAULT_DATA[subGroupBy])); const subGroupValue = get(_issue, ISSUE_FILTER_DEFAULT_DATA[subGroupBy]);
groupArray = this.getGroupArray(get(_issue, ISSUE_FILTER_DEFAULT_DATA[groupBy])); const groupValue = get(_issue, ISSUE_FILTER_DEFAULT_DATA[groupBy]);
subGroupArray = subGroupValue != undefined ? this.getGroupArray(subGroupValue) : [];
groupArray = groupValue != undefined ? this.getGroupArray(groupValue) : [];
} }
for (const subGroup of subGroupArray) { for (const subGroup of subGroupArray) {

View File

@ -49,7 +49,7 @@ export class IssueActivityStore implements IIssueActivityStore {
// computed // computed
get issueActivities() { get issueActivities() {
const issueId = this.rootIssueDetailStore.issueId; const issueId = this.rootIssueDetailStore.peekIssue?.issueId;
if (!issueId) return undefined; if (!issueId) return undefined;
return this.activities[issueId] ?? undefined; return this.activities[issueId] ?? undefined;
} }

View File

@ -62,7 +62,7 @@ export class IssueAttachmentStore implements IIssueAttachmentStore {
// computed // computed
get issueAttachments() { get issueAttachments() {
const issueId = this.rootIssueDetailStore.issueId; const issueId = this.rootIssueDetailStore.peekIssue?.issueId;
if (!issueId) return undefined; if (!issueId) return undefined;
return this.attachments[issueId] ?? undefined; return this.attachments[issueId] ?? undefined;
} }

View File

@ -60,7 +60,7 @@ export class IssueLinkStore implements IIssueLinkStore {
// computed // computed
get issueLinks() { get issueLinks() {
const issueId = this.rootIssueDetailStore.issueId; const issueId = this.rootIssueDetailStore.peekIssue?.issueId;
if (!issueId) return undefined; if (!issueId) return undefined;
return this.links[issueId] ?? undefined; return this.links[issueId] ?? undefined;
} }

View File

@ -53,7 +53,7 @@ export class IssueReactionStore implements IIssueReactionStore {
// computed // computed
get issueReactions() { get issueReactions() {
const issueId = this.rootIssueDetailStore.issueId; const issueId = this.rootIssueDetailStore.peekIssue?.issueId;
if (!issueId) return undefined; if (!issueId) return undefined;
return this.reactions[issueId] ?? undefined; return this.reactions[issueId] ?? undefined;
} }

View File

@ -68,7 +68,7 @@ export class IssueRelationStore implements IIssueRelationStore {
// computed // computed
get issueRelations() { get issueRelations() {
const issueId = this.rootIssueDetailStore.issueId; const issueId = this.rootIssueDetailStore.peekIssue?.issueId;
if (!issueId) return undefined; if (!issueId) return undefined;
return this.relationMap?.[issueId] ?? undefined; return this.relationMap?.[issueId] ?? undefined;
} }

View File

@ -18,6 +18,12 @@ import { IIssueRelationStore, IssueRelationStore, IIssueRelationStoreActions } f
import { TIssue, IIssueActivity, TIssueLink, TIssueRelationTypes } from "@plane/types"; import { TIssue, IIssueActivity, TIssueLink, TIssueRelationTypes } from "@plane/types";
export type TPeekIssue = {
workspaceSlug: string;
projectId: string;
issueId: string;
};
export interface IIssueDetail export interface IIssueDetail
extends IIssueStoreActions, extends IIssueStoreActions,
IIssueReactionStoreActions, IIssueReactionStoreActions,
@ -30,14 +36,14 @@ export interface IIssueDetail
IIssueAttachmentStoreActions, IIssueAttachmentStoreActions,
IIssueRelationStoreActions { IIssueRelationStoreActions {
// observables // observables
issueId: string | undefined; peekIssue: TPeekIssue | undefined;
isIssueLinkModalOpen: boolean; isIssueLinkModalOpen: boolean;
isParentIssueModalOpen: boolean; isParentIssueModalOpen: boolean;
isDeleteIssueModalOpen: boolean; isDeleteIssueModalOpen: boolean;
// computed // computed
isAnyModalOpen: boolean; isAnyModalOpen: boolean;
// actions // actions
setIssueId: (issueId: string | undefined) => void; setPeekIssue: (peekIssue: TPeekIssue | undefined) => void;
toggleIssueLinkModal: (value: boolean) => void; toggleIssueLinkModal: (value: boolean) => void;
toggleParentIssueModal: (value: boolean) => void; toggleParentIssueModal: (value: boolean) => void;
toggleDeleteIssueModal: (value: boolean) => void; toggleDeleteIssueModal: (value: boolean) => void;
@ -57,7 +63,7 @@ export interface IIssueDetail
export class IssueDetail implements IIssueDetail { export class IssueDetail implements IIssueDetail {
// observables // observables
issueId: string | undefined = undefined; peekIssue: TPeekIssue | undefined = undefined;
isIssueLinkModalOpen: boolean = false; isIssueLinkModalOpen: boolean = false;
isParentIssueModalOpen: boolean = false; isParentIssueModalOpen: boolean = false;
isDeleteIssueModalOpen: boolean = false; isDeleteIssueModalOpen: boolean = false;
@ -77,14 +83,14 @@ export class IssueDetail implements IIssueDetail {
constructor(rootStore: IIssueRootStore) { constructor(rootStore: IIssueRootStore) {
makeObservable(this, { makeObservable(this, {
// observables // observables
issueId: observable.ref, peekIssue: observable,
isIssueLinkModalOpen: observable.ref, isIssueLinkModalOpen: observable.ref,
isParentIssueModalOpen: observable.ref, isParentIssueModalOpen: observable.ref,
isDeleteIssueModalOpen: observable.ref, isDeleteIssueModalOpen: observable.ref,
// computed // computed
isAnyModalOpen: computed, isAnyModalOpen: computed,
// action // action
setIssueId: action, setPeekIssue: action,
toggleIssueLinkModal: action, toggleIssueLinkModal: action,
toggleParentIssueModal: action, toggleParentIssueModal: action,
toggleDeleteIssueModal: action, toggleDeleteIssueModal: action,
@ -110,16 +116,14 @@ export class IssueDetail implements IIssueDetail {
} }
// actions // actions
setIssueId = (issueId: string | undefined) => (this.issueId = issueId); setPeekIssue = (peekIssue: TPeekIssue | undefined) => (this.peekIssue = peekIssue);
toggleIssueLinkModal = (value: boolean) => (this.isIssueLinkModalOpen = value); toggleIssueLinkModal = (value: boolean) => (this.isIssueLinkModalOpen = value);
toggleParentIssueModal = (value: boolean) => (this.isParentIssueModalOpen = value); toggleParentIssueModal = (value: boolean) => (this.isParentIssueModalOpen = value);
toggleDeleteIssueModal = (value: boolean) => (this.isDeleteIssueModalOpen = value); toggleDeleteIssueModal = (value: boolean) => (this.isDeleteIssueModalOpen = value);
// issue // issue
fetchIssue = async (workspaceSlug: string, projectId: string, issueId: string) => { fetchIssue = async (workspaceSlug: string, projectId: string, issueId: string) =>
this.issueId = issueId; this.issue.fetchIssue(workspaceSlug, projectId, issueId);
return this.issue.fetchIssue(workspaceSlug, projectId, issueId);
};
updateIssue = async (workspaceSlug: string, projectId: string, issueId: string, data: Partial<TIssue>) => updateIssue = async (workspaceSlug: string, projectId: string, issueId: string, data: Partial<TIssue>) =>
this.issue.updateIssue(workspaceSlug, projectId, issueId, data); this.issue.updateIssue(workspaceSlug, projectId, issueId, data);
removeIssue = async (workspaceSlug: string, projectId: string, issueId: string) => removeIssue = async (workspaceSlug: string, projectId: string, issueId: string) =>

View File

@ -46,7 +46,7 @@ export class IssueSubscriptionStore implements IIssueSubscriptionStore {
if (!issueId) return undefined; if (!issueId) return undefined;
const currentUserId = this.rootIssueDetail.rootIssueStore.currentUserId; const currentUserId = this.rootIssueDetail.rootIssueStore.currentUserId;
if (!currentUserId) return undefined; if (!currentUserId) return undefined;
return this.subscriptionMap[issueId][currentUserId] ?? undefined; return this.subscriptionMap[issueId]?.[currentUserId] ?? undefined;
}; };
fetchSubscriptions = async (workspaceSlug: string, projectId: string, issueId: string) => { fetchSubscriptions = async (workspaceSlug: string, projectId: string, issueId: string) => {

View File

@ -1,757 +0,0 @@
import { observable, action, makeObservable, runInAction, computed, autorun } from "mobx";
// services
import { IssueService, IssueReactionService, IssueCommentService } from "services/issue";
import { NotificationService } from "services/notification.service";
// types
import { IIssueRootStore } from "./root.store";
import type { TIssue, IIssueActivity, IIssueLink, ILinkDetails } from "@plane/types";
// constants
import { groupReactionEmojis } from "constants/issue";
import { RootStore } from "store/root.store";
export interface IIssueDetailStore {
loader: boolean;
error: any | null;
peekId: string | null;
issues: {
[issueId: string]: TIssue;
};
issueReactions: {
[issueId: string]: any;
};
issueActivity: {
[issueId: string]: IIssueActivity[];
};
issueCommentReactions: {
[issueId: string]: {
[comment_id: string]: any;
};
};
issueSubscription: {
[issueId: string]: any;
};
setPeekId: (issueId: string | null) => void;
// computed
getIssue: TIssue | null;
getIssueReactions: any | null;
getIssueActivity: IIssueActivity[] | null;
getIssueCommentReactions: any | null;
getIssueSubscription: any | null;
// fetch issue details
fetchIssueDetails: (workspaceSlug: string, projectId: string, issueId: string) => Promise<TIssue>;
// deleting issue
deleteIssue: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
fetchPeekIssueDetails: (workspaceSlug: string, projectId: string, issueId: string) => Promise<TIssue>;
fetchIssueReactions: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
createIssueReaction: (workspaceSlug: string, projectId: string, issueId: string, reaction: string) => Promise<void>;
removeIssueReaction: (workspaceSlug: string, projectId: string, issueId: string, reaction: string) => Promise<void>;
createIssueLink: (
workspaceSlug: string,
projectId: string,
issueId: string,
data: IIssueLink
) => Promise<ILinkDetails>;
updateIssueLink: (
workspaceSlug: string,
projectId: string,
issueId: string,
linkId: string,
data: IIssueLink
) => Promise<ILinkDetails>;
deleteIssueLink: (workspaceSlug: string, projectId: string, issueId: string, linkId: string) => Promise<void>;
fetchIssueActivity: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
createIssueComment: (workspaceSlug: string, projectId: string, issueId: string, data: any) => Promise<void>;
updateIssueComment: (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
data: any
) => Promise<void>;
removeIssueComment: (workspaceSlug: string, projectId: string, issueId: string, commentId: string) => Promise<void>;
fetchIssueCommentReactions: (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string
) => Promise<void>;
creationIssueCommentReaction: (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
reaction: string
) => Promise<void>;
removeIssueCommentReaction: (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
reaction: string
) => Promise<void>;
fetchIssueSubscription: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
createIssueSubscription: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
removeIssueSubscription: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
}
export class IssueDetailStore implements IIssueDetailStore {
loader: boolean = false;
error: any | null = null;
peekId: string | null = null;
issues: {
[issueId: string]: TIssue;
} = {};
issueReactions: {
[issueId: string]: any;
} = {};
issueActivity: {
[issueId: string]: IIssueActivity[];
} = {};
issueCommentReactions: {
[issueId: string]: any;
} = {};
issueSubscription: {
[issueId: string]: any;
} = {};
// root store
issueRootStore;
rootStore;
// service
issueService;
issueReactionService;
issueCommentService;
notificationService;
constructor(_issueRootStore: IIssueRootStore, _rootStore: RootStore) {
makeObservable(this, {
// observable
loader: observable.ref,
error: observable.ref,
peekId: observable.ref,
issues: observable.ref,
issueReactions: observable.ref,
issueActivity: observable.ref,
issueCommentReactions: observable.ref,
issueSubscription: observable.ref,
getIssue: computed,
getIssueReactions: computed,
getIssueActivity: computed,
getIssueCommentReactions: computed,
getIssueSubscription: computed,
setPeekId: action,
fetchIssueDetails: action,
deleteIssue: action,
fetchPeekIssueDetails: action,
fetchIssueReactions: action,
createIssueReaction: action,
removeIssueReaction: action,
createIssueLink: action,
updateIssueLink: action,
deleteIssueLink: action,
fetchIssueActivity: action,
createIssueComment: action,
updateIssueComment: action,
removeIssueComment: action,
fetchIssueCommentReactions: action,
creationIssueCommentReaction: action,
removeIssueCommentReaction: action,
fetchIssueSubscription: action,
createIssueSubscription: action,
removeIssueSubscription: action,
});
this.issueRootStore = _issueRootStore;
this.rootStore = _rootStore;
this.issueService = new IssueService();
this.issueReactionService = new IssueReactionService();
this.issueCommentService = new IssueCommentService();
this.notificationService = new NotificationService();
autorun(() => {
const projectId = this.rootStore?.app.router.projectId;
const peekId = this.peekId;
if (!projectId || !peekId) return;
// FIXME: uncomment and fix
// const issue = this.issueRootStore.projectIssues.issues?.[projectId]?.[peekId];
// if (issue && issue.id)
// runInAction(() => {
// this.issues = {
// ...this.issues,
// [issue.id]: {
// ...this.issues[issue.id],
// ...issue,
// },
// };
// });
});
}
get getIssue() {
if (!this.peekId) return null;
const _issue = this.issues[this.peekId];
return _issue || null;
}
get getIssueReactions() {
if (!this.peekId) return null;
const _reactions = this.issueReactions[this.peekId];
return _reactions || null;
}
get getIssueActivity() {
if (!this.peekId) return null;
const activity = this.issueActivity[this.peekId];
return activity || null;
}
get getIssueCommentReactions() {
if (!this.peekId) return null;
const _commentReactions = this.issueCommentReactions[this.peekId];
return _commentReactions || null;
}
get getIssueSubscription() {
if (!this.peekId) return null;
const _commentSubscription = this.issueSubscription[this.peekId];
return _commentSubscription || null;
}
setPeekId = (issueId: string | null) => (this.peekId = issueId);
fetchIssueDetails = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
this.loader = true;
this.error = null;
this.peekId = issueId;
const issueDetailsResponse = await this.issueService.retrieve(workspaceSlug, projectId, issueId);
runInAction(() => {
this.loader = false;
this.error = null;
this.issues = {
...this.issues,
[issueId]: issueDetailsResponse,
};
});
return issueDetailsResponse;
} catch (error) {
runInAction(() => {
this.loader = false;
this.error = error;
});
throw error;
}
};
deleteIssue = async (workspaceSlug: string, projectId: string, issueId: string) => {
const newIssues = { ...this.issues };
delete newIssues[issueId];
try {
runInAction(() => {
this.loader = true;
this.error = null;
this.issues = newIssues;
});
const user = this.rootStore.user.currentUser;
if (!user) return;
const response = await this.issueService.deleteIssue(workspaceSlug, projectId, issueId);
runInAction(() => {
this.loader = false;
this.error = null;
});
return response;
} catch (error) {
this.fetchIssueDetails(workspaceSlug, projectId, issueId);
runInAction(() => {
this.loader = false;
this.error = error;
});
return error;
}
};
fetchPeekIssueDetails = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
this.loader = true;
this.error = null;
this.peekId = issueId;
const issueDetailsResponse = await this.issueService.retrieve(workspaceSlug, projectId, issueId);
await this.fetchIssueReactions(workspaceSlug, projectId, issueId);
await this.fetchIssueActivity(workspaceSlug, projectId, issueId);
runInAction(() => {
this.loader = false;
this.error = null;
this.issues = {
...this.issues,
[issueId]: issueDetailsResponse,
};
});
return issueDetailsResponse;
} catch (error) {
runInAction(() => {
this.loader = false;
this.error = error;
});
throw error;
}
};
// reactions
fetchIssueReactions = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const _reactions = await this.issueReactionService.listIssueReactions(workspaceSlug, projectId, issueId);
const _issueReactions = {
...this.issueReactions,
[issueId]: groupReactionEmojis(_reactions),
};
runInAction(() => {
this.issueReactions = _issueReactions;
});
} catch (error) {
console.warn("error creating the issue reaction", error);
throw error;
}
};
createIssueReaction = async (workspaceSlug: string, projectId: string, issueId: string, reaction: string) => {
let _currentReactions = this.getIssueReactions;
try {
const _reaction = await this.issueReactionService.createIssueReaction(workspaceSlug, projectId, issueId, {
reaction,
});
_currentReactions = {
..._currentReactions,
[reaction]: [..._currentReactions[reaction], { ..._reaction }],
};
runInAction(() => {
this.issueReactions = {
...this.issueReactions,
[issueId]: _currentReactions,
};
});
} catch (error) {
runInAction(() => {
this.issueReactions = {
...this.issueReactions,
[issueId]: _currentReactions,
};
});
console.warn("error creating the issue reaction", error);
throw error;
}
};
removeIssueReaction = async (workspaceSlug: string, projectId: string, issueId: string, reaction: string) => {
let _currentReactions = this.getIssueReactions;
try {
const user = this.rootStore.user.currentUser;
if (user) {
_currentReactions = {
..._currentReactions,
[reaction]: [..._currentReactions[reaction].filter((r: any) => r.actor !== user.id)],
};
runInAction(() => {
this.issueReactions = {
...this.issueReactions,
[issueId]: _currentReactions,
};
});
await this.issueReactionService.deleteIssueReaction(workspaceSlug, projectId, issueId, reaction);
}
} catch (error) {
runInAction(() => {
this.issueReactions = {
...this.issueReactions,
[issueId]: _currentReactions,
};
});
console.warn("error removing the issue reaction", error);
throw error;
}
};
fetchIssueActivity = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const issueActivityResponse = await this.issueService.getIssueActivities(workspaceSlug, projectId, issueId);
const _issueComments = {
...this.issueActivity,
[issueId]: [...issueActivityResponse],
};
runInAction(() => {
this.issueActivity = _issueComments;
});
} catch (error) {
console.warn("error creating the issue comment", error);
throw error;
}
};
// comments
createIssueComment = async (workspaceSlug: string, projectId: string, issueId: string, data: any) => {
try {
const _issueCommentResponse = await this.issueCommentService.createIssueComment(
workspaceSlug,
projectId,
issueId,
data
);
const _issueComments = {
...this.issueActivity,
[issueId]: [...this.issueActivity[issueId], _issueCommentResponse],
};
runInAction(() => {
this.issueActivity = _issueComments;
});
} catch (error) {
console.warn("error creating the issue comment", error);
throw error;
}
};
updateIssueComment = async (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
data: any
) => {
try {
const _issueCommentResponse = await this.issueCommentService.patchIssueComment(
workspaceSlug,
projectId,
issueId,
commentId,
data
);
const _issueComments = {
...this.issueActivity,
[issueId]: this.issueActivity[issueId].map((comment) =>
comment.id === commentId ? _issueCommentResponse : comment
),
};
runInAction(() => {
this.issueActivity = _issueComments;
});
} catch (error) {
console.warn("error updating the issue comment", error);
throw error;
}
};
removeIssueComment = async (workspaceSlug: string, projectId: string, issueId: string, commentId: string) => {
try {
const _issueComments = {
...this.issueActivity,
[issueId]: this.issueActivity[issueId]?.filter((comment) => comment.id != commentId),
};
await this.issueCommentService.deleteIssueComment(workspaceSlug, projectId, issueId, commentId);
runInAction(() => {
this.issueActivity = _issueComments;
});
} catch (error) {
console.warn("error removing the issue comment", error);
throw error;
}
};
// comment reactions
fetchIssueCommentReactions = async (workspaceSlug: string, projectId: string, issueId: string, commentId: string) => {
try {
const _reactions = await this.issueReactionService.listIssueCommentReactions(workspaceSlug, projectId, commentId);
const _issueCommentReactions = {
...this.issueCommentReactions,
[issueId]: {
...this.issueCommentReactions[issueId],
[commentId]: groupReactionEmojis(_reactions),
},
};
runInAction(() => {
this.issueCommentReactions = _issueCommentReactions;
});
} catch (error) {
console.warn("error removing the issue comment", error);
throw error;
}
};
creationIssueCommentReaction = async (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
reaction: string
) => {
let _currentReactions = this.getIssueCommentReactions;
_currentReactions = _currentReactions && commentId ? _currentReactions?.[commentId] : null;
try {
const _reaction = await this.issueReactionService.createIssueCommentReaction(
workspaceSlug,
projectId,
commentId,
{
reaction,
}
);
_currentReactions = {
..._currentReactions,
[reaction]: [..._currentReactions?.[reaction], { ..._reaction }],
};
const _issueCommentReactions = {
...this.issueCommentReactions,
[issueId]: {
...this.issueCommentReactions[issueId],
[commentId]: _currentReactions,
},
};
runInAction(() => {
this.issueCommentReactions = _issueCommentReactions;
});
} catch (error) {
console.warn("error removing the issue comment", error);
throw error;
}
};
removeIssueCommentReaction = async (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
reaction: string
) => {
let _currentReactions = this.getIssueCommentReactions;
_currentReactions = _currentReactions && commentId ? _currentReactions?.[commentId] : null;
try {
const user = this.rootStore.user.currentUser;
if (user) {
_currentReactions = {
..._currentReactions,
[reaction]: [..._currentReactions?.[reaction].filter((r: any) => r.actor !== user.id)],
};
const _issueCommentReactions = {
...this.issueCommentReactions,
[issueId]: {
...this.issueCommentReactions[issueId],
[commentId]: _currentReactions,
},
};
runInAction(() => {
this.issueCommentReactions = _issueCommentReactions;
});
await this.issueReactionService.deleteIssueCommentReaction(workspaceSlug, projectId, commentId, reaction);
}
} catch (error) {
console.warn("error removing the issue comment", error);
throw error;
}
};
createIssueLink = async (workspaceSlug: string, projectId: string, issueId: string, data: IIssueLink) => {
try {
const response = await this.issueService.createIssueLink(workspaceSlug, projectId, issueId, data);
runInAction(() => {
this.issues = {
...this.issues,
[issueId]: {
...this.issues[issueId],
issue_link: [response, ...this.issues[issueId].issue_link],
},
};
});
return response;
} catch (error) {
console.error("Failed to create link in store", error);
this.fetchIssueDetails(workspaceSlug, projectId, issueId);
runInAction(() => {
this.error = error;
});
throw error;
}
};
updateIssueLink = async (
workspaceSlug: string,
projectId: string,
issueId: string,
linkId: string,
data: IIssueLink
) => {
try {
const response = await this.issueService.updateIssueLink(workspaceSlug, projectId, issueId, linkId, data);
runInAction(() => {
this.issues = {
...this.issues,
[issueId]: {
...this.issues[issueId],
issue_link: this.issues[issueId].issue_link.map((link: any) => (link.id === linkId ? response : link)),
},
};
});
return response;
} catch (error) {
console.error("Failed to update link in issue store", error);
this.fetchIssueDetails(workspaceSlug, projectId, issueId);
runInAction(() => {
this.error = error;
});
throw error;
}
};
deleteIssueLink = async (workspaceSlug: string, projectId: string, issueId: string, linkId: string) => {
try {
runInAction(() => {
this.issues = {
...this.issues,
[issueId]: {
...this.issues[issueId],
issue_link: this.issues[issueId].issue_link.filter((link: any) => link.id !== linkId),
},
};
});
await this.issueService.deleteIssueLink(workspaceSlug, projectId, issueId, linkId);
} catch (error) {
console.error("Failed to delete link in issue store", error);
runInAction(() => {
this.error = error;
});
throw error;
}
};
// subscriptions
fetchIssueSubscription = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const _subscription = await this.notificationService.getIssueNotificationSubscriptionStatus(
workspaceSlug,
projectId,
issueId
);
const _issueSubscription = {
...this.issueSubscription,
[issueId]: _subscription,
};
runInAction(() => {
this.issueSubscription = _issueSubscription;
});
} catch (error) {
console.warn("error fetching the issue subscription", error);
throw error;
}
};
createIssueSubscription = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
await this.notificationService.subscribeToIssueNotifications(workspaceSlug, projectId, issueId);
const _issueSubscription = {
...this.issueSubscription,
[issueId]: { subscribed: true },
};
runInAction(() => {
this.issueSubscription = _issueSubscription;
});
} catch (error) {
console.warn("error creating the issue subscription", error);
throw error;
}
};
removeIssueSubscription = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const _issueSubscription = {
...this.issueSubscription,
[issueId]: { subscribed: false },
};
runInAction(() => {
this.issueSubscription = _issueSubscription;
});
await this.notificationService.unsubscribeFromIssueNotifications(workspaceSlug, projectId, issueId);
} catch (error) {
console.warn("error removing the issue subscription", error);
throw error;
}
};
}

View File

@ -40,7 +40,7 @@ export class IssueKanBanViewStore implements IIssueKanBanViewStore {
get canUserDragDrop() { get canUserDragDrop() {
return true; return true;
if (this.rootStore.issueDetail.issueId) return false; if (this.rootStore.issueDetail.peekIssue?.issueId) return false;
// FIXME: uncomment and fix // FIXME: uncomment and fix
// if ( // if (
// this.rootStore?.issueFilter?.userDisplayFilters?.order_by && // this.rootStore?.issueFilter?.userDisplayFilters?.order_by &&

View File

@ -145,16 +145,23 @@ export class ModuleIssuesFilter extends IssueFilterHelperStore implements IModul
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -150,16 +150,23 @@ export class ProfileIssuesFilter extends IssueFilterHelperStore implements IProf
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -146,16 +146,23 @@ export class ProjectViewIssuesFilter extends IssueFilterHelperStore implements I
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -142,16 +142,23 @@ export class ProjectIssuesFilter extends IssueFilterHelperStore implements IProj
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {

View File

@ -157,16 +157,23 @@ export class WorkspaceIssuesFilter extends IssueFilterHelperStore implements IWo
_filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters }; _filters.displayFilters = { ..._filters.displayFilters, ...updatedDisplayFilters };
// set sub_group_by to null if group_by is set to null // set sub_group_by to null if group_by is set to null
if (_filters.displayFilters.group_by === null) _filters.displayFilters.sub_group_by = null; if (_filters.displayFilters.group_by === null) {
_filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same // set sub_group_by to null if layout is switched to kanban group_by and sub_group_by are same
if ( if (
_filters.displayFilters.layout === "kanban" && _filters.displayFilters.layout === "kanban" &&
_filters.displayFilters.group_by === _filters.displayFilters.sub_group_by _filters.displayFilters.group_by === _filters.displayFilters.sub_group_by
) ) {
_filters.displayFilters.sub_group_by = null; _filters.displayFilters.sub_group_by = null;
updatedDisplayFilters.sub_group_by = null;
}
// set group_by to state if layout is switched to kanban and group_by is null // set group_by to state if layout is switched to kanban and group_by is null
if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) if (_filters.displayFilters.layout === "kanban" && _filters.displayFilters.group_by === null) {
_filters.displayFilters.group_by = "state"; _filters.displayFilters.group_by = "state";
updatedDisplayFilters.group_by = "state";
}
runInAction(() => { runInAction(() => {
Object.keys(updatedDisplayFilters).forEach((_key) => { Object.keys(updatedDisplayFilters).forEach((_key) => {