mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
Merge branch 'develop' of github.com:makeplane/plane into feat/mobx-global-views
This commit is contained in:
commit
f63a04c1ab
@ -562,7 +562,7 @@ class IssueSerializer(DynamicBaseSerializer):
|
|||||||
state_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
state_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
||||||
parent_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
parent_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
||||||
cycle_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
cycle_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
||||||
module_id = serializers.PrimaryKeyRelatedField(read_only=True)
|
module_ids = serializers.SerializerMethodField()
|
||||||
|
|
||||||
# Many to many
|
# Many to many
|
||||||
label_ids = serializers.PrimaryKeyRelatedField(
|
label_ids = serializers.PrimaryKeyRelatedField(
|
||||||
@ -597,7 +597,7 @@ class IssueSerializer(DynamicBaseSerializer):
|
|||||||
"project_id",
|
"project_id",
|
||||||
"parent_id",
|
"parent_id",
|
||||||
"cycle_id",
|
"cycle_id",
|
||||||
"module_id",
|
"module_ids",
|
||||||
"label_ids",
|
"label_ids",
|
||||||
"assignee_ids",
|
"assignee_ids",
|
||||||
"sub_issues_count",
|
"sub_issues_count",
|
||||||
@ -613,6 +613,10 @@ class IssueSerializer(DynamicBaseSerializer):
|
|||||||
]
|
]
|
||||||
read_only_fields = fields
|
read_only_fields = fields
|
||||||
|
|
||||||
|
def get_module_ids(self, obj):
|
||||||
|
# Access the prefetched modules and extract module IDs
|
||||||
|
return [module for module in obj.issue_module.values_list("module_id", flat=True)]
|
||||||
|
|
||||||
|
|
||||||
class IssueLiteSerializer(DynamicBaseSerializer):
|
class IssueLiteSerializer(DynamicBaseSerializer):
|
||||||
workspace_detail = WorkspaceLiteSerializer(
|
workspace_detail = WorkspaceLiteSerializer(
|
||||||
|
@ -35,17 +35,26 @@ urlpatterns = [
|
|||||||
name="project-modules",
|
name="project-modules",
|
||||||
),
|
),
|
||||||
path(
|
path(
|
||||||
"workspaces/<str:slug>/projects/<uuid:project_id>/modules/<uuid:module_id>/module-issues/",
|
"workspaces/<str:slug>/projects/<uuid:project_id>/issues/<uuid:issue_id>/modules/",
|
||||||
ModuleIssueViewSet.as_view(
|
ModuleIssueViewSet.as_view(
|
||||||
{
|
{
|
||||||
|
"post": "create_issue_modules",
|
||||||
|
}
|
||||||
|
),
|
||||||
|
name="issue-module",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"workspaces/<str:slug>/projects/<uuid:project_id>/modules/<uuid:module_id>/issues/",
|
||||||
|
ModuleIssueViewSet.as_view(
|
||||||
|
{
|
||||||
|
"post": "create_module_issues",
|
||||||
"get": "list",
|
"get": "list",
|
||||||
"post": "create",
|
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
name="project-module-issues",
|
name="project-module-issues",
|
||||||
),
|
),
|
||||||
path(
|
path(
|
||||||
"workspaces/<str:slug>/projects/<uuid:project_id>/modules/<uuid:module_id>/module-issues/<uuid:issue_id>/",
|
"workspaces/<str:slug>/projects/<uuid:project_id>/modules/<uuid:module_id>/issues/<uuid:issue_id>/",
|
||||||
ModuleIssueViewSet.as_view(
|
ModuleIssueViewSet.as_view(
|
||||||
{
|
{
|
||||||
"get": "retrieve",
|
"get": "retrieve",
|
||||||
|
@ -599,16 +599,11 @@ class CycleIssueViewSet(WebhookMixin, BaseViewSet):
|
|||||||
)
|
)
|
||||||
.filter(project_id=project_id)
|
.filter(project_id=project_id)
|
||||||
.filter(workspace__slug=slug)
|
.filter(workspace__slug=slug)
|
||||||
.select_related("project")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.select_related("workspace")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.select_related("state")
|
|
||||||
.select_related("parent")
|
|
||||||
.prefetch_related("assignees")
|
|
||||||
.prefetch_related("labels")
|
|
||||||
.order_by(order_by)
|
.order_by(order_by)
|
||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
|
@ -100,7 +100,7 @@ def dashboard_assigned_issues(self, request, slug):
|
|||||||
)
|
)
|
||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.select_related("workspace", "project", "state", "parent")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.prefetch_related("assignees", "labels")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.prefetch_related(
|
.prefetch_related(
|
||||||
Prefetch(
|
Prefetch(
|
||||||
"issue_relation",
|
"issue_relation",
|
||||||
@ -110,7 +110,6 @@ def dashboard_assigned_issues(self, request, slug):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
@ -221,9 +220,8 @@ def dashboard_created_issues(self, request, slug):
|
|||||||
)
|
)
|
||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.select_related("workspace", "project", "state", "parent")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.prefetch_related("assignees", "labels")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
|
@ -95,7 +95,7 @@ class InboxIssueViewSet(BaseViewSet):
|
|||||||
issue_inbox__inbox_id=self.kwargs.get("inbox_id")
|
issue_inbox__inbox_id=self.kwargs.get("inbox_id")
|
||||||
)
|
)
|
||||||
.select_related("workspace", "project", "state", "parent")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.prefetch_related("labels", "assignees")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.prefetch_related(
|
.prefetch_related(
|
||||||
Prefetch(
|
Prefetch(
|
||||||
"issue_inbox",
|
"issue_inbox",
|
||||||
@ -105,7 +105,6 @@ class InboxIssueViewSet(BaseViewSet):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
|
@ -112,12 +112,8 @@ class IssueViewSet(WebhookMixin, BaseViewSet):
|
|||||||
project_id=self.kwargs.get("project_id")
|
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("workspace", "project", "state", "parent")
|
||||||
.select_related("workspace")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.select_related("state")
|
|
||||||
.select_related("parent")
|
|
||||||
.prefetch_related("assignees")
|
|
||||||
.prefetch_related("labels")
|
|
||||||
.prefetch_related(
|
.prefetch_related(
|
||||||
Prefetch(
|
Prefetch(
|
||||||
"issue_reactions",
|
"issue_reactions",
|
||||||
@ -125,7 +121,6 @@ class IssueViewSet(WebhookMixin, BaseViewSet):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
@ -1087,12 +1082,31 @@ class IssueArchiveViewSet(BaseViewSet):
|
|||||||
.filter(archived_at__isnull=False)
|
.filter(archived_at__isnull=False)
|
||||||
.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("workspace", "project", "state", "parent")
|
||||||
.select_related("workspace")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.select_related("state")
|
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
||||||
.select_related("parent")
|
.annotate(
|
||||||
.prefetch_related("assignees")
|
link_count=IssueLink.objects.filter(issue=OuterRef("id"))
|
||||||
.prefetch_related("labels")
|
.order_by()
|
||||||
|
.annotate(count=Func(F("id"), function="Count"))
|
||||||
|
.values("count")
|
||||||
|
)
|
||||||
|
.annotate(
|
||||||
|
attachment_count=IssueAttachment.objects.filter(
|
||||||
|
issue=OuterRef("id")
|
||||||
|
)
|
||||||
|
.order_by()
|
||||||
|
.annotate(count=Func(F("id"), function="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")
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@method_decorator(gzip_page)
|
@method_decorator(gzip_page)
|
||||||
@ -1120,22 +1134,6 @@ class IssueArchiveViewSet(BaseViewSet):
|
|||||||
issue_queryset = (
|
issue_queryset = (
|
||||||
self.get_queryset()
|
self.get_queryset()
|
||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
|
||||||
.annotate(module_id=F("issue_module__module_id"))
|
|
||||||
.annotate(
|
|
||||||
link_count=IssueLink.objects.filter(issue=OuterRef("id"))
|
|
||||||
.order_by()
|
|
||||||
.annotate(count=Func(F("id"), function="Count"))
|
|
||||||
.values("count")
|
|
||||||
)
|
|
||||||
.annotate(
|
|
||||||
attachment_count=IssueAttachment.objects.filter(
|
|
||||||
issue=OuterRef("id")
|
|
||||||
)
|
|
||||||
.order_by()
|
|
||||||
.annotate(count=Func(F("id"), function="Count"))
|
|
||||||
.values("count")
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Priority Ordering
|
# Priority Ordering
|
||||||
@ -1681,18 +1679,37 @@ class IssueDraftViewSet(BaseViewSet):
|
|||||||
.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"))
|
||||||
.filter(is_draft=True)
|
.filter(is_draft=True)
|
||||||
.select_related("project")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.select_related("workspace")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.select_related("state")
|
|
||||||
.select_related("parent")
|
|
||||||
.prefetch_related("assignees")
|
|
||||||
.prefetch_related("labels")
|
|
||||||
.prefetch_related(
|
.prefetch_related(
|
||||||
Prefetch(
|
Prefetch(
|
||||||
"issue_reactions",
|
"issue_reactions",
|
||||||
queryset=IssueReaction.objects.select_related("actor"),
|
queryset=IssueReaction.objects.select_related("actor"),
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
||||||
|
.annotate(
|
||||||
|
link_count=IssueLink.objects.filter(issue=OuterRef("id"))
|
||||||
|
.order_by()
|
||||||
|
.annotate(count=Func(F("id"), function="Count"))
|
||||||
|
.values("count")
|
||||||
|
)
|
||||||
|
.annotate(
|
||||||
|
attachment_count=IssueAttachment.objects.filter(
|
||||||
|
issue=OuterRef("id")
|
||||||
|
)
|
||||||
|
.order_by()
|
||||||
|
.annotate(count=Func(F("id"), function="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")
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@method_decorator(gzip_page)
|
@method_decorator(gzip_page)
|
||||||
@ -1719,22 +1736,6 @@ class IssueDraftViewSet(BaseViewSet):
|
|||||||
issue_queryset = (
|
issue_queryset = (
|
||||||
self.get_queryset()
|
self.get_queryset()
|
||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
|
||||||
.annotate(module_id=F("issue_module__module_id"))
|
|
||||||
.annotate(
|
|
||||||
link_count=IssueLink.objects.filter(issue=OuterRef("id"))
|
|
||||||
.order_by()
|
|
||||||
.annotate(count=Func(F("id"), function="Count"))
|
|
||||||
.values("count")
|
|
||||||
)
|
|
||||||
.annotate(
|
|
||||||
attachment_count=IssueAttachment.objects.filter(
|
|
||||||
issue=OuterRef("id")
|
|
||||||
)
|
|
||||||
.order_by()
|
|
||||||
.annotate(count=Func(F("id"), function="Count"))
|
|
||||||
.values("count")
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Priority Ordering
|
# Priority Ordering
|
||||||
|
@ -7,6 +7,8 @@ from django.db.models import Prefetch, F, OuterRef, Func, Exists, Count, Q
|
|||||||
from django.core import serializers
|
from django.core import serializers
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
from django.views.decorators.gzip import gzip_page
|
from django.views.decorators.gzip import gzip_page
|
||||||
|
from django.core.serializers.json import DjangoJSONEncoder
|
||||||
|
|
||||||
|
|
||||||
# Third party imports
|
# Third party imports
|
||||||
from rest_framework.response import Response
|
from rest_framework.response import Response
|
||||||
@ -296,23 +298,20 @@ class ModuleViewSet(WebhookMixin, BaseViewSet):
|
|||||||
"issue", flat=True
|
"issue", flat=True
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
_ = [
|
||||||
issue_activity.delay(
|
issue_activity.delay(
|
||||||
type="module.activity.deleted",
|
type="module.activity.deleted",
|
||||||
requested_data=json.dumps(
|
requested_data=json.dumps({"module_id": str(pk)}),
|
||||||
{
|
|
||||||
"module_id": str(pk),
|
|
||||||
"module_name": str(module.name),
|
|
||||||
"issues": [str(issue_id) for issue_id in module_issues],
|
|
||||||
}
|
|
||||||
),
|
|
||||||
actor_id=str(request.user.id),
|
actor_id=str(request.user.id),
|
||||||
issue_id=str(pk),
|
issue_id=str(issue),
|
||||||
project_id=str(project_id),
|
project_id=project_id,
|
||||||
current_instance=None,
|
current_instance=json.dumps({"module_name": str(module.name)}),
|
||||||
epoch=int(timezone.now().timestamp()),
|
epoch=int(timezone.now().timestamp()),
|
||||||
notification=True,
|
notification=True,
|
||||||
origin=request.META.get("HTTP_ORIGIN"),
|
origin=request.META.get("HTTP_ORIGIN"),
|
||||||
)
|
)
|
||||||
|
for issue in module_issues
|
||||||
|
]
|
||||||
module.delete()
|
module.delete()
|
||||||
return Response(status=status.HTTP_204_NO_CONTENT)
|
return Response(status=status.HTTP_204_NO_CONTENT)
|
||||||
|
|
||||||
@ -332,62 +331,18 @@ class ModuleIssueViewSet(WebhookMixin, BaseViewSet):
|
|||||||
ProjectEntityPermission,
|
ProjectEntityPermission,
|
||||||
]
|
]
|
||||||
|
|
||||||
def get_queryset(self):
|
|
||||||
return self.filter_queryset(
|
|
||||||
super()
|
|
||||||
.get_queryset()
|
|
||||||
.annotate(
|
|
||||||
sub_issues_count=Issue.issue_objects.filter(
|
|
||||||
parent=OuterRef("issue")
|
|
||||||
)
|
|
||||||
.order_by()
|
|
||||||
.annotate(count=Func(F("id"), function="Count"))
|
|
||||||
.values("count")
|
|
||||||
)
|
|
||||||
.filter(workspace__slug=self.kwargs.get("slug"))
|
|
||||||
.filter(project_id=self.kwargs.get("project_id"))
|
|
||||||
.filter(module_id=self.kwargs.get("module_id"))
|
|
||||||
.filter(project__project_projectmember__member=self.request.user)
|
|
||||||
.select_related("project")
|
|
||||||
.select_related("workspace")
|
|
||||||
.select_related("module")
|
|
||||||
.select_related("issue", "issue__state", "issue__project")
|
|
||||||
.prefetch_related("issue__assignees", "issue__labels")
|
|
||||||
.prefetch_related("module__members")
|
|
||||||
.distinct()
|
|
||||||
)
|
|
||||||
|
|
||||||
@method_decorator(gzip_page)
|
def get_queryset(self):
|
||||||
def list(self, request, slug, project_id, module_id):
|
return (
|
||||||
fields = [
|
Issue.objects.filter(
|
||||||
field
|
project_id=self.kwargs.get("project_id"),
|
||||||
for field in request.GET.get("fields", "").split(",")
|
workspace__slug=self.kwargs.get("slug"),
|
||||||
if field
|
issue_module__module_id=self.kwargs.get("module_id")
|
||||||
]
|
|
||||||
order_by = request.GET.get("order_by", "created_at")
|
|
||||||
filters = issue_filters(request.query_params, "GET")
|
|
||||||
issues = (
|
|
||||||
Issue.issue_objects.filter(issue_module__module_id=module_id)
|
|
||||||
.annotate(
|
|
||||||
sub_issues_count=Issue.issue_objects.filter(
|
|
||||||
parent=OuterRef("id")
|
|
||||||
)
|
)
|
||||||
.order_by()
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.annotate(count=Func(F("id"), function="Count"))
|
.prefetch_related("labels", "assignees")
|
||||||
.values("count")
|
.prefetch_related('issue_module__module')
|
||||||
)
|
|
||||||
.filter(project_id=project_id)
|
|
||||||
.filter(workspace__slug=slug)
|
|
||||||
.select_related("project")
|
|
||||||
.select_related("workspace")
|
|
||||||
.select_related("state")
|
|
||||||
.select_related("parent")
|
|
||||||
.prefetch_related("assignees")
|
|
||||||
.prefetch_related("labels")
|
|
||||||
.order_by(order_by)
|
|
||||||
.filter(**filters)
|
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
@ -403,106 +358,119 @@ class ModuleIssueViewSet(WebhookMixin, BaseViewSet):
|
|||||||
.values("count")
|
.values("count")
|
||||||
)
|
)
|
||||||
.annotate(
|
.annotate(
|
||||||
is_subscribed=Exists(
|
sub_issues_count=Issue.issue_objects.filter(
|
||||||
IssueSubscriber.objects.filter(
|
parent=OuterRef("id")
|
||||||
subscriber=self.request.user, issue_id=OuterRef("id")
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
.order_by()
|
||||||
|
.annotate(count=Func(F("id"), function="Count"))
|
||||||
|
.values("count")
|
||||||
)
|
)
|
||||||
|
).distinct()
|
||||||
|
|
||||||
|
@method_decorator(gzip_page)
|
||||||
|
def list(self, request, slug, project_id, module_id):
|
||||||
|
fields = [
|
||||||
|
field
|
||||||
|
for field in request.GET.get("fields", "").split(",")
|
||||||
|
if field
|
||||||
|
]
|
||||||
|
filters = issue_filters(request.query_params, "GET")
|
||||||
|
issue_queryset = self.get_queryset().filter(**filters)
|
||||||
serializer = IssueSerializer(
|
serializer = IssueSerializer(
|
||||||
issues, 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)
|
||||||
|
|
||||||
def create(self, request, slug, project_id, module_id):
|
# create multiple issues inside a module
|
||||||
|
def create_module_issues(self, request, slug, project_id, module_id):
|
||||||
issues = request.data.get("issues", [])
|
issues = request.data.get("issues", [])
|
||||||
if not len(issues):
|
if not len(issues):
|
||||||
return Response(
|
return Response(
|
||||||
{"error": "Issues are required"},
|
{"error": "Issues are required"},
|
||||||
status=status.HTTP_400_BAD_REQUEST,
|
status=status.HTTP_400_BAD_REQUEST,
|
||||||
)
|
)
|
||||||
module = Module.objects.get(
|
project = Project.objects.get(pk=project_id)
|
||||||
workspace__slug=slug, project_id=project_id, pk=module_id
|
_ = ModuleIssue.objects.bulk_create(
|
||||||
)
|
[
|
||||||
|
|
||||||
module_issues = list(ModuleIssue.objects.filter(issue_id__in=issues))
|
|
||||||
|
|
||||||
update_module_issue_activity = []
|
|
||||||
records_to_update = []
|
|
||||||
record_to_create = []
|
|
||||||
|
|
||||||
for issue in issues:
|
|
||||||
module_issue = [
|
|
||||||
module_issue
|
|
||||||
for module_issue in module_issues
|
|
||||||
if str(module_issue.issue_id) in issues
|
|
||||||
]
|
|
||||||
|
|
||||||
if len(module_issue):
|
|
||||||
if module_issue[0].module_id != module_id:
|
|
||||||
update_module_issue_activity.append(
|
|
||||||
{
|
|
||||||
"old_module_id": str(module_issue[0].module_id),
|
|
||||||
"new_module_id": str(module_id),
|
|
||||||
"issue_id": str(module_issue[0].issue_id),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
module_issue[0].module_id = module_id
|
|
||||||
records_to_update.append(module_issue[0])
|
|
||||||
else:
|
|
||||||
record_to_create.append(
|
|
||||||
ModuleIssue(
|
ModuleIssue(
|
||||||
module=module,
|
issue_id=str(issue),
|
||||||
issue_id=issue,
|
module_id=module_id,
|
||||||
project_id=project_id,
|
project_id=project_id,
|
||||||
workspace=module.workspace,
|
workspace_id=project.workspace_id,
|
||||||
created_by=request.user,
|
created_by=request.user,
|
||||||
updated_by=request.user,
|
updated_by=request.user,
|
||||||
)
|
)
|
||||||
)
|
for issue in issues
|
||||||
|
],
|
||||||
ModuleIssue.objects.bulk_create(
|
|
||||||
record_to_create,
|
|
||||||
batch_size=10,
|
batch_size=10,
|
||||||
ignore_conflicts=True,
|
ignore_conflicts=True,
|
||||||
)
|
)
|
||||||
|
# Bulk Update the activity
|
||||||
ModuleIssue.objects.bulk_update(
|
_ = [
|
||||||
records_to_update,
|
|
||||||
["module"],
|
|
||||||
batch_size=10,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Capture Issue Activity
|
|
||||||
issue_activity.delay(
|
issue_activity.delay(
|
||||||
type="module.activity.created",
|
type="module.activity.created",
|
||||||
requested_data=json.dumps({"modules_list": issues}),
|
requested_data=json.dumps({"module_id": str(module_id)}),
|
||||||
actor_id=str(self.request.user.id),
|
actor_id=str(request.user.id),
|
||||||
issue_id=None,
|
issue_id=str(issue),
|
||||||
project_id=str(self.kwargs.get("project_id", None)),
|
project_id=project_id,
|
||||||
current_instance=json.dumps(
|
current_instance=None,
|
||||||
{
|
|
||||||
"updated_module_issues": update_module_issue_activity,
|
|
||||||
"created_module_issues": serializers.serialize(
|
|
||||||
"json", record_to_create
|
|
||||||
),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
epoch=int(timezone.now().timestamp()),
|
epoch=int(timezone.now().timestamp()),
|
||||||
notification=True,
|
notification=True,
|
||||||
origin=request.META.get("HTTP_ORIGIN"),
|
origin=request.META.get("HTTP_ORIGIN"),
|
||||||
)
|
)
|
||||||
|
for issue in issues
|
||||||
|
]
|
||||||
|
issues = (self.get_queryset().filter(pk__in=issues))
|
||||||
|
serializer = IssueSerializer(issues , many=True)
|
||||||
|
return Response(serializer.data, status=status.HTTP_201_CREATED)
|
||||||
|
|
||||||
issues = self.get_queryset().values_list("issue_id", flat=True)
|
|
||||||
|
|
||||||
|
# create multiple module inside an issue
|
||||||
|
def create_issue_modules(self, request, slug, project_id, issue_id):
|
||||||
|
modules = request.data.get("modules", [])
|
||||||
|
if not len(modules):
|
||||||
return Response(
|
return Response(
|
||||||
IssueSerializer(
|
{"error": "Modules are required"},
|
||||||
Issue.objects.filter(pk__in=issues), many=True
|
status=status.HTTP_400_BAD_REQUEST,
|
||||||
).data,
|
|
||||||
status=status.HTTP_200_OK,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
project = Project.objects.get(pk=project_id)
|
||||||
|
_ = ModuleIssue.objects.bulk_create(
|
||||||
|
[
|
||||||
|
ModuleIssue(
|
||||||
|
issue_id=issue_id,
|
||||||
|
module_id=module,
|
||||||
|
project_id=project_id,
|
||||||
|
workspace_id=project.workspace_id,
|
||||||
|
created_by=request.user,
|
||||||
|
updated_by=request.user,
|
||||||
|
)
|
||||||
|
for module in modules
|
||||||
|
],
|
||||||
|
batch_size=10,
|
||||||
|
ignore_conflicts=True,
|
||||||
|
)
|
||||||
|
# Bulk Update the activity
|
||||||
|
_ = [
|
||||||
|
issue_activity.delay(
|
||||||
|
type="module.activity.created",
|
||||||
|
requested_data=json.dumps({"module_id": module}),
|
||||||
|
actor_id=str(request.user.id),
|
||||||
|
issue_id=issue_id,
|
||||||
|
project_id=project_id,
|
||||||
|
current_instance=None,
|
||||||
|
epoch=int(timezone.now().timestamp()),
|
||||||
|
notification=True,
|
||||||
|
origin=request.META.get("HTTP_ORIGIN"),
|
||||||
|
)
|
||||||
|
for module in modules
|
||||||
|
]
|
||||||
|
|
||||||
|
issue = (self.get_queryset().filter(pk=issue_id).first())
|
||||||
|
serializer = IssueSerializer(issue)
|
||||||
|
return Response(serializer.data, status=status.HTTP_201_CREATED)
|
||||||
|
|
||||||
|
|
||||||
def destroy(self, request, slug, project_id, module_id, issue_id):
|
def destroy(self, request, slug, project_id, module_id, issue_id):
|
||||||
module_issue = ModuleIssue.objects.get(
|
module_issue = ModuleIssue.objects.get(
|
||||||
workspace__slug=slug,
|
workspace__slug=slug,
|
||||||
@ -512,16 +480,11 @@ class ModuleIssueViewSet(WebhookMixin, BaseViewSet):
|
|||||||
)
|
)
|
||||||
issue_activity.delay(
|
issue_activity.delay(
|
||||||
type="module.activity.deleted",
|
type="module.activity.deleted",
|
||||||
requested_data=json.dumps(
|
requested_data=json.dumps({"module_id": str(module_id)}),
|
||||||
{
|
|
||||||
"module_id": str(module_id),
|
|
||||||
"issues": [str(issue_id)],
|
|
||||||
}
|
|
||||||
),
|
|
||||||
actor_id=str(request.user.id),
|
actor_id=str(request.user.id),
|
||||||
issue_id=str(issue_id),
|
issue_id=str(issue_id),
|
||||||
project_id=str(project_id),
|
project_id=str(project_id),
|
||||||
current_instance=None,
|
current_instance=json.dumps({"module_name": module_issue.module.name}),
|
||||||
epoch=int(timezone.now().timestamp()),
|
epoch=int(timezone.now().timestamp()),
|
||||||
notification=True,
|
notification=True,
|
||||||
origin=request.META.get("HTTP_ORIGIN"),
|
origin=request.META.get("HTTP_ORIGIN"),
|
||||||
|
@ -228,7 +228,7 @@ class IssueSearchEndpoint(BaseAPIView):
|
|||||||
parent = request.query_params.get("parent", "false")
|
parent = request.query_params.get("parent", "false")
|
||||||
issue_relation = request.query_params.get("issue_relation", "false")
|
issue_relation = request.query_params.get("issue_relation", "false")
|
||||||
cycle = request.query_params.get("cycle", "false")
|
cycle = request.query_params.get("cycle", "false")
|
||||||
module = request.query_params.get("module", "false")
|
module = request.query_params.get("module", False)
|
||||||
sub_issue = request.query_params.get("sub_issue", "false")
|
sub_issue = request.query_params.get("sub_issue", "false")
|
||||||
|
|
||||||
issue_id = request.query_params.get("issue_id", False)
|
issue_id = request.query_params.get("issue_id", False)
|
||||||
@ -269,8 +269,8 @@ class IssueSearchEndpoint(BaseAPIView):
|
|||||||
if cycle == "true":
|
if cycle == "true":
|
||||||
issues = issues.exclude(issue_cycle__isnull=False)
|
issues = issues.exclude(issue_cycle__isnull=False)
|
||||||
|
|
||||||
if module == "true":
|
if module:
|
||||||
issues = issues.exclude(issue_module__isnull=False)
|
issues = issues.exclude(issue_module__module=module)
|
||||||
|
|
||||||
return Response(
|
return Response(
|
||||||
issues.values(
|
issues.values(
|
||||||
|
@ -87,12 +87,8 @@ class GlobalViewIssuesViewSet(BaseViewSet):
|
|||||||
.values("count")
|
.values("count")
|
||||||
)
|
)
|
||||||
.filter(workspace__slug=self.kwargs.get("slug"))
|
.filter(workspace__slug=self.kwargs.get("slug"))
|
||||||
.select_related("project")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.select_related("workspace")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.select_related("state")
|
|
||||||
.select_related("parent")
|
|
||||||
.prefetch_related("assignees")
|
|
||||||
.prefetch_related("labels")
|
|
||||||
.prefetch_related(
|
.prefetch_related(
|
||||||
Prefetch(
|
Prefetch(
|
||||||
"issue_reactions",
|
"issue_reactions",
|
||||||
@ -127,7 +123,6 @@ class GlobalViewIssuesViewSet(BaseViewSet):
|
|||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.filter(project__project_projectmember__member=self.request.user)
|
.filter(project__project_projectmember__member=self.request.user)
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
@ -150,13 +145,6 @@ class GlobalViewIssuesViewSet(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")
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Priority Ordering
|
# Priority Ordering
|
||||||
|
@ -1346,9 +1346,8 @@ class WorkspaceUserProfileIssuesEndpoint(BaseAPIView):
|
|||||||
)
|
)
|
||||||
.filter(**filters)
|
.filter(**filters)
|
||||||
.select_related("workspace", "project", "state", "parent")
|
.select_related("workspace", "project", "state", "parent")
|
||||||
.prefetch_related("assignees", "labels")
|
.prefetch_related("assignees", "labels", "issue_module__module")
|
||||||
.annotate(cycle_id=F("issue_cycle__cycle_id"))
|
.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()
|
||||||
|
@ -148,10 +148,12 @@ def send_email_notification(
|
|||||||
template_data = []
|
template_data = []
|
||||||
total_changes = 0
|
total_changes = 0
|
||||||
comments = []
|
comments = []
|
||||||
|
actors_involved = []
|
||||||
for actor_id, changes in data.items():
|
for actor_id, changes in data.items():
|
||||||
actor = User.objects.get(pk=actor_id)
|
actor = User.objects.get(pk=actor_id)
|
||||||
total_changes = total_changes + len(changes)
|
total_changes = total_changes + len(changes)
|
||||||
comment = changes.pop("comment", False)
|
comment = changes.pop("comment", False)
|
||||||
|
actors_involved.append(actor_id)
|
||||||
if comment:
|
if comment:
|
||||||
comments.append(
|
comments.append(
|
||||||
{
|
{
|
||||||
@ -191,6 +193,7 @@ def send_email_notification(
|
|||||||
context = {
|
context = {
|
||||||
"data": template_data,
|
"data": template_data,
|
||||||
"summary": summary,
|
"summary": summary,
|
||||||
|
"actors_involved": len(set(actors_involved)),
|
||||||
"issue": {
|
"issue": {
|
||||||
"issue_identifier": f"{str(issue.project.identifier)}-{str(issue.sequence_id)}",
|
"issue_identifier": f"{str(issue.project.identifier)}-{str(issue.sequence_id)}",
|
||||||
"name": issue.name,
|
"name": issue.name,
|
||||||
@ -200,6 +203,9 @@ def send_email_notification(
|
|||||||
"email": receiver.email,
|
"email": receiver.email,
|
||||||
},
|
},
|
||||||
"issue_url": f"{base_api}/{str(issue.project.workspace.slug)}/projects/{str(issue.project.id)}/issues/{str(issue.id)}",
|
"issue_url": f"{base_api}/{str(issue.project.workspace.slug)}/projects/{str(issue.project.id)}/issues/{str(issue.id)}",
|
||||||
|
"project_url": f"{base_api}/{str(issue.project.workspace.slug)}/projects/{str(issue.project.id)}/issues/",
|
||||||
|
"workspace":str(issue.project.workspace.slug),
|
||||||
|
"project": str(issue.project.name),
|
||||||
"user_preference": f"{base_api}/profile/preferences/email",
|
"user_preference": f"{base_api}/profile/preferences/email",
|
||||||
"comments": comments,
|
"comments": comments,
|
||||||
}
|
}
|
||||||
|
@ -30,6 +30,7 @@ from plane.app.serializers import IssueActivitySerializer
|
|||||||
from plane.bgtasks.notification_task import notifications
|
from plane.bgtasks.notification_task import notifications
|
||||||
from plane.settings.redis import redis_instance
|
from plane.settings.redis import redis_instance
|
||||||
|
|
||||||
|
|
||||||
# Track Changes in name
|
# Track Changes in name
|
||||||
def track_name(
|
def track_name(
|
||||||
requested_data,
|
requested_data,
|
||||||
@ -852,58 +853,14 @@ def create_module_issue_activity(
|
|||||||
requested_data = (
|
requested_data = (
|
||||||
json.loads(requested_data) if requested_data is not None else None
|
json.loads(requested_data) if requested_data is not None else None
|
||||||
)
|
)
|
||||||
current_instance = (
|
module = Module.objects.filter(pk=requested_data.get("module_id")).first()
|
||||||
json.loads(current_instance) if current_instance is not None else None
|
issue = Issue.objects.filter(pk=issue_id).first()
|
||||||
)
|
|
||||||
|
|
||||||
# Updated Records:
|
|
||||||
updated_records = current_instance.get("updated_module_issues", [])
|
|
||||||
created_records = json.loads(
|
|
||||||
current_instance.get("created_module_issues", [])
|
|
||||||
)
|
|
||||||
|
|
||||||
for updated_record in updated_records:
|
|
||||||
old_module = Module.objects.filter(
|
|
||||||
pk=updated_record.get("old_module_id", None)
|
|
||||||
).first()
|
|
||||||
new_module = Module.objects.filter(
|
|
||||||
pk=updated_record.get("new_module_id", None)
|
|
||||||
).first()
|
|
||||||
issue = Issue.objects.filter(pk=updated_record.get("issue_id")).first()
|
|
||||||
if issue:
|
|
||||||
issue.updated_at = timezone.now()
|
|
||||||
issue.save(update_fields=["updated_at"])
|
|
||||||
|
|
||||||
issue_activities.append(
|
|
||||||
IssueActivity(
|
|
||||||
issue_id=updated_record.get("issue_id"),
|
|
||||||
actor_id=actor_id,
|
|
||||||
verb="updated",
|
|
||||||
old_value=old_module.name,
|
|
||||||
new_value=new_module.name,
|
|
||||||
field="modules",
|
|
||||||
project_id=project_id,
|
|
||||||
workspace_id=workspace_id,
|
|
||||||
comment=f"updated module to ",
|
|
||||||
old_identifier=old_module.id,
|
|
||||||
new_identifier=new_module.id,
|
|
||||||
epoch=epoch,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
for created_record in created_records:
|
|
||||||
module = Module.objects.filter(
|
|
||||||
pk=created_record.get("fields").get("module")
|
|
||||||
).first()
|
|
||||||
issue = Issue.objects.filter(
|
|
||||||
pk=created_record.get("fields").get("issue")
|
|
||||||
).first()
|
|
||||||
if issue:
|
if issue:
|
||||||
issue.updated_at = timezone.now()
|
issue.updated_at = timezone.now()
|
||||||
issue.save(update_fields=["updated_at"])
|
issue.save(update_fields=["updated_at"])
|
||||||
issue_activities.append(
|
issue_activities.append(
|
||||||
IssueActivity(
|
IssueActivity(
|
||||||
issue_id=created_record.get("fields").get("issue"),
|
issue_id=issue_id,
|
||||||
actor_id=actor_id,
|
actor_id=actor_id,
|
||||||
verb="created",
|
verb="created",
|
||||||
old_value="",
|
old_value="",
|
||||||
@ -912,7 +869,7 @@ def create_module_issue_activity(
|
|||||||
project_id=project_id,
|
project_id=project_id,
|
||||||
workspace_id=workspace_id,
|
workspace_id=workspace_id,
|
||||||
comment=f"added module {module.name}",
|
comment=f"added module {module.name}",
|
||||||
new_identifier=module.id,
|
new_identifier=requested_data.get("module_id"),
|
||||||
epoch=epoch,
|
epoch=epoch,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -934,29 +891,23 @@ def delete_module_issue_activity(
|
|||||||
current_instance = (
|
current_instance = (
|
||||||
json.loads(current_instance) if current_instance is not None else None
|
json.loads(current_instance) if current_instance is not None else None
|
||||||
)
|
)
|
||||||
|
module_name = current_instance.get("module_name")
|
||||||
module_id = requested_data.get("module_id", "")
|
current_issue = Issue.objects.filter(pk=issue_id).first()
|
||||||
module_name = requested_data.get("module_name", "")
|
if current_issue:
|
||||||
module = Module.objects.filter(pk=module_id).first()
|
|
||||||
issues = requested_data.get("issues")
|
|
||||||
|
|
||||||
for issue in issues:
|
|
||||||
current_issue = Issue.objects.filter(pk=issue).first()
|
|
||||||
if issue:
|
|
||||||
current_issue.updated_at = timezone.now()
|
current_issue.updated_at = timezone.now()
|
||||||
current_issue.save(update_fields=["updated_at"])
|
current_issue.save(update_fields=["updated_at"])
|
||||||
issue_activities.append(
|
issue_activities.append(
|
||||||
IssueActivity(
|
IssueActivity(
|
||||||
issue_id=issue,
|
issue_id=issue_id,
|
||||||
actor_id=actor_id,
|
actor_id=actor_id,
|
||||||
verb="deleted",
|
verb="deleted",
|
||||||
old_value=module.name if module is not None else module_name,
|
old_value=module_name,
|
||||||
new_value="",
|
new_value="",
|
||||||
field="modules",
|
field="modules",
|
||||||
project_id=project_id,
|
project_id=project_id,
|
||||||
workspace_id=workspace_id,
|
workspace_id=workspace_id,
|
||||||
comment=f"removed this issue from {module.name if module is not None else module_name}",
|
comment=f"removed this issue from {module_name}",
|
||||||
old_identifier=module_id if module_id is not None else None,
|
old_identifier=requested_data.get("module_id") if requested_data.get("module_id") is not None else None,
|
||||||
epoch=epoch,
|
epoch=epoch,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -1649,7 +1600,6 @@ def issue_activity(
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
capture_exception(e)
|
capture_exception(e)
|
||||||
|
|
||||||
|
|
||||||
if notification:
|
if notification:
|
||||||
notifications.delay(
|
notifications.delay(
|
||||||
type=type,
|
type=type,
|
||||||
|
@ -0,0 +1,23 @@
|
|||||||
|
# Generated by Django 4.2.7 on 2024-01-24 18:55
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.db.models.deletion
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('db', '0057_auto_20240122_0901'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='moduleissue',
|
||||||
|
name='issue',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='issue_module', to='db.issue'),
|
||||||
|
),
|
||||||
|
migrations.AlterUniqueTogether(
|
||||||
|
name='moduleissue',
|
||||||
|
unique_together={('issue', 'module')},
|
||||||
|
),
|
||||||
|
]
|
@ -134,11 +134,12 @@ class ModuleIssue(ProjectBaseModel):
|
|||||||
module = models.ForeignKey(
|
module = models.ForeignKey(
|
||||||
"db.Module", on_delete=models.CASCADE, related_name="issue_module"
|
"db.Module", on_delete=models.CASCADE, related_name="issue_module"
|
||||||
)
|
)
|
||||||
issue = models.OneToOneField(
|
issue = models.ForeignKey(
|
||||||
"db.Issue", on_delete=models.CASCADE, related_name="issue_module"
|
"db.Issue", on_delete=models.CASCADE, related_name="issue_module"
|
||||||
)
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
|
unique_together = ["issue", "module"]
|
||||||
verbose_name = "Module Issue"
|
verbose_name = "Module Issue"
|
||||||
verbose_name_plural = "Module Issues"
|
verbose_name_plural = "Module Issues"
|
||||||
db_table = "module_issues"
|
db_table = "module_issues"
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
xmlns="http://www.w3.org/1999/xhtml"
|
xmlns="http://www.w3.org/1999/xhtml"
|
||||||
xmlns:v="urn:schemas-microsoft-com:vml"
|
xmlns:v="urn:schemas-microsoft-com:vml"
|
||||||
xmlns:o="urn:schemas-microsoft-com:office:office"
|
xmlns:o="urn:schemas-microsoft-com:office:office"
|
||||||
>
|
>
|
||||||
<head>
|
<head>
|
||||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
@ -28,6 +28,18 @@
|
|||||||
color: #3358d4 !important;
|
color: #3358d4 !important;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
<style>
|
||||||
|
*[class="gmail-fix"] {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<style type="text/css" emogrify="no">
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
.gmx-killpill {
|
||||||
|
content: " \03D1";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body
|
<body
|
||||||
bgcolor="#ffffff"
|
bgcolor="#ffffff"
|
||||||
@ -82,7 +94,7 @@
|
|||||||
{{ issue.issue_identifier }} updates
|
{{ issue.issue_identifier }} updates
|
||||||
</p>
|
</p>
|
||||||
<p style="font-size: 1rem; font-weight: 500; color: #1f2d5c; line-height: 28px">
|
<p style="font-size: 1rem; font-weight: 500; color: #1f2d5c; line-height: 28px">
|
||||||
{{ issue.name }}: {{ issue.issue_identifier }}
|
{{workspace}}/<a target="_blank" style="color: #1f2d5c; text-decoration: none;" href="{{project_url}}">{{project}}</a>/<a style="color: #1f2d5c; text-decoration: none;" target="_blank" href="{{issue_url}}">{{issue.issue_identifier}}</a>: {{ issue.name }}
|
||||||
</p>
|
</p>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
@ -96,42 +108,41 @@
|
|||||||
margin-bottom: 15px;
|
margin-bottom: 15px;
|
||||||
"
|
"
|
||||||
/>
|
/>
|
||||||
|
{% if actors_involved > 0 %}
|
||||||
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28pxl">
|
{% if data|length > 0 and comments|length == 0 %}
|
||||||
{% if data.1 %}{{ data|length }}{% endif %} {{ summary }}
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
||||||
{{ data.0.actor_detail.first_name}}
|
{{ data.0.actor_detail.first_name}}
|
||||||
{{data.0.actor_detail.last_name }}
|
{{data.0.actor_detail.last_name }}
|
||||||
</span>
|
</span>
|
||||||
|
made {{data|length}} {% if data|length > 1 %}updates{% else %}update{% endif %} to the issue.
|
||||||
</p>
|
</p>
|
||||||
{% if comments.0 %}
|
{% elif data|length == 0 and comments|length > 0 %}
|
||||||
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
{{ comments|length }} {% if comments|length == 1 %}comment was{% else %}comments were{% endif %} left by
|
|
||||||
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
||||||
{% if comments|length == 1 %}
|
{{ comments.0.actor_detail.first_name}}
|
||||||
{{ data.0.actor_detail.first_name }}
|
{{comments.0.actor_detail.last_name }}
|
||||||
{{ data.0.actor_detail.last_name }}
|
|
||||||
{% else %}
|
|
||||||
{{ data.0.actor_detail.first_name }}
|
|
||||||
{{ data.0.actor_detail.last_name }} and others
|
|
||||||
{% endif %}
|
|
||||||
</span>
|
</span>
|
||||||
|
added {{comments|length}} new {% if comments|length > 1 %}comments{% else %}comment{% endif %}.
|
||||||
</p>
|
</p>
|
||||||
{% endif %}
|
{% elif data|length > 0 and comments|length > 0 %}
|
||||||
{% if mentions and comments.0 and data.0 %}
|
|
||||||
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
There are 3 new updates, added 1 new comment and, you were
|
|
||||||
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
<span style="font-size: 1rem; font-weight: 700; line-height: 28px">
|
||||||
@{{ data.0.actor_detail.first_name}}
|
{{ data.0.actor_detail.first_name}}
|
||||||
{{data.0.actor_detail.last_name }}
|
{{data.0.actor_detail.last_name }}
|
||||||
</span>
|
</span>
|
||||||
mentioned a comment of this issue.
|
made {{data|length}} {% if data|length > 1 %}updates{% else %}update{% endif %} and added {{comments|length}} new {% if comments|length > 1 %}comments{% else %}comment{% endif %} on the issue.
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
<p style="font-size: 1rem;color: #1f2d5c; line-height: 28px">
|
||||||
|
There are {{ data|length }} new updates and {{comments|length}} new comments on the issue
|
||||||
</p>
|
</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% for update in data %} {% if update.changes.name %}
|
{% for update in data %} {% if update.changes.name %}
|
||||||
<!-- Issue title updated -->
|
<!-- Issue title updated -->
|
||||||
<p style="font-size: 1rem; line-height: 28px; color: #1f2d5c">
|
<p style="font-size: 1rem; line-height: 28px; color: #1f2d5c">
|
||||||
The issue title has been updated from “{{update.changes.user.old_value.0}}“ to "{{update.changes.user.new_value|last}}"
|
The issue title has been updated to {{ issue.name}}
|
||||||
</p>
|
</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<!-- Outer update Box start -->
|
<!-- Outer update Box start -->
|
||||||
@ -164,6 +175,7 @@
|
|||||||
word-wrap: break-word;
|
word-wrap: break-word;
|
||||||
padding-left: 15px;
|
padding-left: 15px;
|
||||||
padding-bottom: 15px;
|
padding-bottom: 15px;
|
||||||
|
border-radius: 8px;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
<!-- action performer -->
|
<!-- action performer -->
|
||||||
@ -264,6 +276,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
|
{% if update.changes.target_date.new_value.0 %}
|
||||||
<p
|
<p
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
@ -275,6 +288,20 @@
|
|||||||
>
|
>
|
||||||
{{ update.changes.target_date.new_value.0 }}
|
{{ update.changes.target_date.new_value.0 }}
|
||||||
</p>
|
</p>
|
||||||
|
{% else %}
|
||||||
|
<p
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #171717;
|
||||||
|
margin-left: 5px;
|
||||||
|
padding: 0px;
|
||||||
|
text-decoration: line-through;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
{{ update.changes.target_date.old_value.0 }}
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
@ -285,7 +312,7 @@
|
|||||||
style="max-width: 100%; padding-bottom: 15px"
|
style="max-width: 100%; padding-bottom: 15px"
|
||||||
>
|
>
|
||||||
<tr>
|
<tr>
|
||||||
<td style="overflow-wrap: break-word;">
|
<td>
|
||||||
<img
|
<img
|
||||||
src="https://plane-marketing.s3.ap-south-1.amazonaws.com/plane-assets/emails/duplicate.webp"
|
src="https://plane-marketing.s3.ap-south-1.amazonaws.com/plane-assets/emails/duplicate.webp"
|
||||||
width="12"
|
width="12"
|
||||||
@ -303,9 +330,9 @@
|
|||||||
Duplicate:
|
Duplicate:
|
||||||
</span>
|
</span>
|
||||||
</td>
|
</td>
|
||||||
|
{% if update.changes.duplicate.new_value.0 %}
|
||||||
<td style="padding-left: 5px;">
|
<td style="padding-left: 5px;overflow-wrap: break-word;">
|
||||||
{% for duplicate in update.changes.duplicate.new_value %}
|
{% for duplicate in update.changes.duplicate.new_value|slice:":2" %}
|
||||||
<span
|
<span
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
@ -319,7 +346,57 @@
|
|||||||
</span>
|
</span>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</td>
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if update.changes.duplicate.new_value.2 %}
|
||||||
|
<td>
|
||||||
|
<span
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #3a5bc7;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
+{{ update.changes.duplicate.new_value|length|add:"-2" }}
|
||||||
|
more
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if update.changes.duplicate.old_value.0 %}
|
||||||
|
<td style="padding-left: 8px;">
|
||||||
|
{% for duplicate in update.changes.duplicate.old_value|slice:":2" %}
|
||||||
|
<span
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #641723;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
text-decoration: line-through;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
{{ duplicate }}
|
||||||
|
</span>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if update.changes.duplicate.old_value.2 %}
|
||||||
|
<td>
|
||||||
|
<span
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #641723;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
+{{ update.changes.duplicate.old_value|length|add:"-2" }}
|
||||||
|
more
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -327,195 +404,192 @@
|
|||||||
{% if update.changes.assignees %}
|
{% if update.changes.assignees %}
|
||||||
<table
|
<table
|
||||||
role="presentation"
|
role="presentation"
|
||||||
style="max-width: 100%; padding-bottom: 15px"
|
style="padding-bottom: 15px; max-width: 100%; padding-right: 10px;"
|
||||||
>
|
>
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td valign="top" style="white-space: nowrap; padding: 0px;">
|
||||||
<img
|
<img
|
||||||
src="https://plane-marketing.s3.ap-south-1.amazonaws.com/plane-assets/emails/assignee.webp"
|
src="https://plane-marketing.s3.ap-south-1.amazonaws.com/plane-assets/emails/assignee.webp"
|
||||||
width="12"
|
width="12"
|
||||||
height="12"
|
height="12"
|
||||||
border="0"
|
border="0"
|
||||||
style="display: block"
|
style="display: inline-block"
|
||||||
/>
|
/>
|
||||||
</td>
|
<span
|
||||||
<td>
|
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #525252;
|
color: #525252;
|
||||||
|
padding-right: 5px;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
Assignees:
|
Assignee:
|
||||||
</p>
|
</span>
|
||||||
</td>
|
</td>
|
||||||
|
<td style="overflow-wrap: anywhere;word-break: break-all; padding: 0px;">
|
||||||
{% if update.changes.assignees.new_value.0 %}
|
{% if update.changes.assignees.new_value.0 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
color: #0d74ce;
|
color: #0d74ce;
|
||||||
background-color: #e6f4fe;
|
background-color: #e6f4fe;
|
||||||
margin-left: 5px;
|
margin-right: 5px;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
padding-top: 2px;
|
|
||||||
padding-bottom: 2px;
|
padding-bottom: 2px;
|
||||||
|
padding-top: 2px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
border-radius: 4px;
|
border-radius: 2px;
|
||||||
|
max-lines: 1;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
{{ update.changes.assignees.new_value.0 }}
|
{{update.changes.assignees.new_value.0}}
|
||||||
</p>
|
</span>
|
||||||
</td>
|
{% endif %}
|
||||||
{% endif %} {% if update.changes.assignees.new_value.1 %}
|
{% if update.changes.assignees.new_value.1 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #0d74ce;
|
color: #0d74ce;
|
||||||
margin-left: 2px;
|
margin-left: 2px;
|
||||||
|
margin-right: 5px;
|
||||||
|
white-space: nowrap;
|
||||||
|
padding: 0px;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
+{{ update.changes.assignees.new_value|length|add:"-1"}}
|
+{{ update.changes.assignees.new_value|length|add:"-1"}} more
|
||||||
more
|
</span>
|
||||||
</p>
|
{% endif %}
|
||||||
</td>
|
{% if update.changes.assignees.old_value.0 %}
|
||||||
{% endif %} {% if update.changes.assignees.old_value.0 %}
|
<span
|
||||||
<td>
|
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
padding-top: 2px;
|
|
||||||
padding-bottom: 2px;
|
padding-bottom: 2px;
|
||||||
|
padding-top: 2px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
text-decoration: line-through;
|
text-decoration: line-through;
|
||||||
color: #641723;
|
color: #641723;
|
||||||
background-color: #feebec;
|
background-color: #feebec;
|
||||||
margin-left: 5px;
|
margin-right: 5px;
|
||||||
border-radius: 4px;
|
border-radius: 2px;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
{{update.changes.assignees.old_value.0}}
|
{{update.changes.assignees.old_value.0}}
|
||||||
</p>
|
</span>
|
||||||
</td>
|
{% endif %}
|
||||||
{% endif %} {% if update.changes.assignees.old_value.1 %}
|
{% if update.changes.assignees.old_value.1 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #641723;
|
color: #641723;
|
||||||
margin-left: 2px;
|
margin-left: 2px;
|
||||||
|
white-space: nowrap;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
+{{ update.changes.assignees.old_value|length|add:"-1"}}
|
+{{ update.changes.assignees.old_value|length|add:"-1"}} more
|
||||||
more
|
</span>
|
||||||
</p>
|
|
||||||
</td>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
{% endif %} {% if update.changes.labels %}
|
{% endif %} {% if update.changes.labels %}
|
||||||
<!-- Labels -->
|
<!-- Labels -->
|
||||||
<table
|
<table
|
||||||
role="presentation"
|
role="presentation"
|
||||||
style="padding-bottom: 15px; max-width: 100%"
|
style="padding-bottom: 15px; max-width: 100%; padding-right: 10px;"
|
||||||
>
|
>
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td valign="top" style="white-space: nowrap; padding: 0px;">
|
||||||
<img
|
<img
|
||||||
src="https://plane-marketing.s3.ap-south-1.amazonaws.com/plane-assets/emails/labels.webp"
|
src="https://plane-marketing.s3.ap-south-1.amazonaws.com/plane-assets/emails/labels.webp"
|
||||||
width="12"
|
width="12"
|
||||||
height="12"
|
height="12"
|
||||||
border="0"
|
border="0"
|
||||||
style="display: block"
|
style="display: inline-block;"
|
||||||
/>
|
/>
|
||||||
</td>
|
<span
|
||||||
<td>
|
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #525252;
|
color: #525252;
|
||||||
|
padding-right: 5px;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
Labels:
|
Labels:
|
||||||
</p>
|
</span>
|
||||||
</td>
|
</td>
|
||||||
|
<td style="overflow-wrap: anywhere;word-break: break-all; padding: 0px;">
|
||||||
{% if update.changes.labels.new_value.0 %}
|
{% if update.changes.labels.new_value.0 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
|
line-height: 1px;
|
||||||
color: #0d74ce;
|
color: #0d74ce;
|
||||||
background-color: #e6f4fe;
|
background-color: #e6f4fe;
|
||||||
margin-left: 5px;
|
margin-right: 5px;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
padding-top: 2px;
|
|
||||||
padding-bottom: 2px;
|
padding-bottom: 2px;
|
||||||
|
padding-top: 2px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
border-radius: 4px;
|
border-radius: 2px;
|
||||||
|
max-lines: 1;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
{{update.changes.labels.new_value.0}}
|
{{update.changes.labels.new_value.0}}
|
||||||
</p>
|
</span>
|
||||||
</td>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if update.changes.labels.new_value.1 %}
|
{% if update.changes.labels.new_value.1 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #0d74ce;
|
color: #0d74ce;
|
||||||
margin-left: 2px;
|
margin-left: 2px;
|
||||||
|
margin-right: 5px;
|
||||||
|
white-space: nowrap;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
+{{ update.changes.labels.new_value|length|add:"-1"}} more
|
+{{ update.changes.labels.new_value|length|add:"-1"}} more
|
||||||
</p>
|
</span>
|
||||||
</td>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if update.changes.labels.old_value.0 %}
|
{% if update.changes.labels.old_value.0 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
padding-top: 2px;
|
|
||||||
padding-bottom: 2px;
|
padding-bottom: 2px;
|
||||||
|
padding-top: 2px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
text-decoration: line-through;
|
text-decoration: line-through;
|
||||||
color: #641723;
|
color: #641723;
|
||||||
background-color: #feebec;
|
background-color: #feebec;
|
||||||
margin-left: 5px;
|
margin-right: 5px;
|
||||||
border-radius: 4px;
|
border-radius: 2px;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
{{update.changes.labels.old_value.0}}
|
{{update.changes.labels.old_value.0}}
|
||||||
</p>
|
</span>
|
||||||
</td>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if update.changes.labels.old_value.1 %}
|
{% if update.changes.labels.old_value.1 %}
|
||||||
<td>
|
<span
|
||||||
<p
|
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #641723;
|
color: #641723;
|
||||||
margin-left: 2px;
|
margin-left: 2px;
|
||||||
|
white-space: nowrap;
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
+{{ update.changes.labels.old_value|length|add:"-1"}} more
|
+{{ update.changes.labels.old_value|length|add:"-1"}} more
|
||||||
</p>
|
</span>
|
||||||
</td>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -620,7 +694,6 @@
|
|||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #525252;
|
color: #525252;
|
||||||
margin-right: 5px;
|
margin-right: 5px;
|
||||||
|
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
Links:
|
Links:
|
||||||
@ -687,16 +760,15 @@
|
|||||||
Priority:
|
Priority:
|
||||||
</p>
|
</p>
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td>
|
<td>
|
||||||
<p
|
<p
|
||||||
style="
|
style="
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
padding-bottom: 2px;
|
padding-bottom: 0px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
border-radius: 4px;
|
border-radius: 2px;
|
||||||
text-transform: capitalize;
|
text-transform: capitalize;
|
||||||
{% if update.changes.priority.old_value.0 == 'urgent' %}background-color: #FFDBDC; color: #CE2C31;{% endif %}
|
{% if update.changes.priority.old_value.0 == 'urgent' %}background-color: #FFDBDC; color: #CE2C31;{% endif %}
|
||||||
{% if update.changes.priority.old_value.0 == 'high' %}background-color: #FFE2C7; color: #F04610;{% endif %}
|
{% if update.changes.priority.old_value.0 == 'high' %}background-color: #FFE2C7; color: #F04610;{% endif %}
|
||||||
@ -722,9 +794,9 @@
|
|||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
padding-bottom: 2px;
|
padding-bottom: 0px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
border-radius: 4px;
|
border-radius: 2px;
|
||||||
text-transform: capitalize;
|
text-transform: capitalize;
|
||||||
{% if update.changes.priority.new_value|last == 'urgent' %}background-color: #FFDBDC; color: #CE2C31;{% endif %}
|
{% if update.changes.priority.new_value|last == 'urgent' %}background-color: #FFDBDC; color: #CE2C31;{% endif %}
|
||||||
{% if update.changes.priority.new_value|last == 'high' %}background-color: #FEEBEC; color: #CE2C31;{% endif %}
|
{% if update.changes.priority.new_value|last == 'high' %}background-color: #FEEBEC; color: #CE2C31;{% endif %}
|
||||||
@ -764,15 +836,73 @@
|
|||||||
Blocking:
|
Blocking:
|
||||||
</span>
|
</span>
|
||||||
</td>
|
</td>
|
||||||
|
{% if update.changes.blocking.new_value.0 %}
|
||||||
<td>
|
<td style="padding-left: 5px;overflow-wrap: break-word;">
|
||||||
{% for blocking in update.changes.blocking.new_value %}
|
{% for blocking in update.changes.blocking.new_value|slice:":2" %}
|
||||||
<span href="" style="font-size: 0.8rem; font-weight: 500; margin-right: 5px; color: #3358d4;">
|
<span
|
||||||
{{blocking}}
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #3a5bc7;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
{{ blocking }}
|
||||||
</span>
|
</span>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</td>
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if update.changes.blocking.new_value.2 %}
|
||||||
|
<td>
|
||||||
|
<span
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #3a5bc7;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
+{{ update.changes.blocking.new_value|length|add:"-2" }}
|
||||||
|
more
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if update.changes.blocking.old_value.0 %}
|
||||||
|
<td style="padding-left: 8px;">
|
||||||
|
{% for blocking in update.changes.blocking.old_value|slice:":2" %}
|
||||||
|
<span
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #641723;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
text-decoration: line-through;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
{{ blocking }}
|
||||||
|
</span>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if update.changes.blocking.old_value.2 %}
|
||||||
|
<td>
|
||||||
|
<span
|
||||||
|
style="
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #641723;
|
||||||
|
margin-right: 3px;
|
||||||
|
padding-top: 0px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
+{{ update.changes.blocking.old_value|length|add:"-2" }}
|
||||||
|
more
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
2
packages/types/src/issues/issue.d.ts
vendored
2
packages/types/src/issues/issue.d.ts
vendored
@ -21,7 +21,7 @@ export type TIssue = {
|
|||||||
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_ids: string[] | null;
|
||||||
|
|
||||||
created_at: string;
|
created_at: string;
|
||||||
updated_at: string;
|
updated_at: string;
|
||||||
|
2
packages/types/src/projects.d.ts
vendored
2
packages/types/src/projects.d.ts
vendored
@ -117,7 +117,7 @@ export type TProjectIssuesSearchParams = {
|
|||||||
parent?: boolean;
|
parent?: boolean;
|
||||||
issue_relation?: boolean;
|
issue_relation?: boolean;
|
||||||
cycle?: boolean;
|
cycle?: boolean;
|
||||||
module?: boolean;
|
module?: string[];
|
||||||
sub_issue?: boolean;
|
sub_issue?: boolean;
|
||||||
issue_id?: string;
|
issue_id?: string;
|
||||||
workspace_search: boolean;
|
workspace_search: boolean;
|
||||||
|
@ -8,6 +8,8 @@ import useToast from "hooks/use-toast";
|
|||||||
import { Button, Input } from "@plane/ui";
|
import { Button, Input } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { checkEmailValidity } from "helpers/string.helper";
|
import { checkEmailValidity } from "helpers/string.helper";
|
||||||
|
// icons
|
||||||
|
import { Eye, EyeOff } from "lucide-react";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
email: string;
|
email: string;
|
||||||
@ -31,6 +33,7 @@ export const SignInOptionalSetPasswordForm: React.FC<Props> = (props) => {
|
|||||||
const { email, handleSignInRedirection } = props;
|
const { email, handleSignInRedirection } = props;
|
||||||
// states
|
// states
|
||||||
const [isGoingToWorkspace, setIsGoingToWorkspace] = useState(false);
|
const [isGoingToWorkspace, setIsGoingToWorkspace] = useState(false);
|
||||||
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
// toast alert
|
// toast alert
|
||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
// form info
|
// form info
|
||||||
@ -114,8 +117,9 @@ export const SignInOptionalSetPasswordForm: React.FC<Props> = (props) => {
|
|||||||
required: "Password is required",
|
required: "Password is required",
|
||||||
}}
|
}}
|
||||||
render={({ field: { value, onChange, ref } }) => (
|
render={({ field: { value, onChange, ref } }) => (
|
||||||
|
<div className="relative flex items-center rounded-md bg-onboarding-background-200">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type={showPassword ? "text" : "password"}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
ref={ref}
|
ref={ref}
|
||||||
@ -125,6 +129,18 @@ export const SignInOptionalSetPasswordForm: React.FC<Props> = (props) => {
|
|||||||
minLength={8}
|
minLength={8}
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
|
{showPassword ? (
|
||||||
|
<EyeOff
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(false)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Eye
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<p className="text-onboarding-text-200 text-xs mt-2 pb-3">
|
<p className="text-onboarding-text-200 text-xs mt-2 pb-3">
|
||||||
|
@ -2,7 +2,7 @@ import React, { useState } from "react";
|
|||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import { XCircle } from "lucide-react";
|
import { Eye, EyeOff, XCircle } from "lucide-react";
|
||||||
// services
|
// services
|
||||||
import { AuthService } from "services/auth.service";
|
import { AuthService } from "services/auth.service";
|
||||||
// hooks
|
// hooks
|
||||||
@ -40,6 +40,7 @@ export const SignInPasswordForm: React.FC<Props> = observer((props) => {
|
|||||||
const { email, handleStepChange, handleEmailClear, onSubmit } = props;
|
const { email, handleStepChange, handleEmailClear, onSubmit } = props;
|
||||||
// states
|
// states
|
||||||
const [isSendingUniqueCode, setIsSendingUniqueCode] = useState(false);
|
const [isSendingUniqueCode, setIsSendingUniqueCode] = useState(false);
|
||||||
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
// toast alert
|
// toast alert
|
||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
const {
|
const {
|
||||||
@ -157,8 +158,9 @@ export const SignInPasswordForm: React.FC<Props> = observer((props) => {
|
|||||||
required: "Password is required",
|
required: "Password is required",
|
||||||
}}
|
}}
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
|
<div className="relative flex items-center rounded-md bg-onboarding-background-200">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type={showPassword ? "text" : "password"}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
hasError={Boolean(errors.password)}
|
hasError={Boolean(errors.password)}
|
||||||
@ -166,6 +168,18 @@ export const SignInPasswordForm: React.FC<Props> = observer((props) => {
|
|||||||
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
|
{showPassword ? (
|
||||||
|
<EyeOff
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(false)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Eye
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<div className="w-full text-right mt-2 pb-3">
|
<div className="w-full text-right mt-2 pb-3">
|
||||||
|
@ -10,6 +10,8 @@ import { Button, Input } from "@plane/ui";
|
|||||||
import { checkEmailValidity } from "helpers/string.helper";
|
import { checkEmailValidity } from "helpers/string.helper";
|
||||||
// constants
|
// constants
|
||||||
import { ESignUpSteps } from "components/account";
|
import { ESignUpSteps } from "components/account";
|
||||||
|
// icons
|
||||||
|
import { Eye, EyeOff } from "lucide-react";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
email: string;
|
email: string;
|
||||||
@ -34,6 +36,7 @@ export const SignUpOptionalSetPasswordForm: React.FC<Props> = (props) => {
|
|||||||
const { email, handleSignInRedirection } = props;
|
const { email, handleSignInRedirection } = props;
|
||||||
// states
|
// states
|
||||||
const [isGoingToWorkspace, setIsGoingToWorkspace] = useState(false);
|
const [isGoingToWorkspace, setIsGoingToWorkspace] = useState(false);
|
||||||
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
// toast alert
|
// toast alert
|
||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
// form info
|
// form info
|
||||||
@ -119,8 +122,9 @@ export const SignUpOptionalSetPasswordForm: React.FC<Props> = (props) => {
|
|||||||
required: "Password is required",
|
required: "Password is required",
|
||||||
}}
|
}}
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
|
<div className="relative flex items-center rounded-md bg-onboarding-background-200">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type={showPassword ? "text" : "password"}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
hasError={Boolean(errors.password)}
|
hasError={Boolean(errors.password)}
|
||||||
@ -129,6 +133,18 @@ export const SignUpOptionalSetPasswordForm: React.FC<Props> = (props) => {
|
|||||||
minLength={8}
|
minLength={8}
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
|
{showPassword ? (
|
||||||
|
<EyeOff
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(false)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Eye
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<p className="text-onboarding-text-200 text-xs mt-2 pb-3">
|
<p className="text-onboarding-text-200 text-xs mt-2 pb-3">
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import React from "react";
|
import React, { useState } from "react";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import { XCircle } from "lucide-react";
|
import { Eye, EyeOff, XCircle } from "lucide-react";
|
||||||
// services
|
// services
|
||||||
import { AuthService } from "services/auth.service";
|
import { AuthService } from "services/auth.service";
|
||||||
// hooks
|
// hooks
|
||||||
@ -32,6 +32,8 @@ const authService = new AuthService();
|
|||||||
|
|
||||||
export const SignUpPasswordForm: React.FC<Props> = observer((props) => {
|
export const SignUpPasswordForm: React.FC<Props> = observer((props) => {
|
||||||
const { onSubmit } = props;
|
const { onSubmit } = props;
|
||||||
|
// states
|
||||||
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
// toast alert
|
// toast alert
|
||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
// form info
|
// form info
|
||||||
@ -112,8 +114,9 @@ export const SignUpPasswordForm: React.FC<Props> = observer((props) => {
|
|||||||
required: "Password is required",
|
required: "Password is required",
|
||||||
}}
|
}}
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
|
<div className="relative flex items-center rounded-md bg-onboarding-background-200">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type={showPassword ? "text" : "password"}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
hasError={Boolean(errors.password)}
|
hasError={Boolean(errors.password)}
|
||||||
@ -121,6 +124,18 @@ export const SignUpPasswordForm: React.FC<Props> = observer((props) => {
|
|||||||
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
|
{showPassword ? (
|
||||||
|
<EyeOff
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(false)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Eye
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<p className="text-onboarding-text-200 text-xs mt-2 pb-3">
|
<p className="text-onboarding-text-200 text-xs mt-2 pb-3">
|
||||||
|
@ -216,7 +216,7 @@ export const CommandPalette: FC = observer(() => {
|
|||||||
<CreateUpdateIssueModal
|
<CreateUpdateIssueModal
|
||||||
isOpen={isCreateIssueModalOpen}
|
isOpen={isCreateIssueModalOpen}
|
||||||
onClose={() => toggleCreateIssueModal(false)}
|
onClose={() => toggleCreateIssueModal(false)}
|
||||||
data={cycleId ? { cycle_id: cycleId.toString() } : moduleId ? { module_id: moduleId.toString() } : undefined}
|
data={cycleId ? { cycle_id: cycleId.toString() } : moduleId ? { module_ids: [moduleId.toString()] } : undefined}
|
||||||
storeType={createIssueStoreType}
|
storeType={createIssueStoreType}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
@ -0,0 +1,16 @@
|
|||||||
|
import { FC } from "react";
|
||||||
|
import { Menu } from "lucide-react";
|
||||||
|
import { useApplication } from "hooks/store";
|
||||||
|
import { observer } from "mobx-react";
|
||||||
|
|
||||||
|
export const SidebarHamburgerToggle: FC = observer (() => {
|
||||||
|
const { theme: themStore } = useApplication();
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className="w-7 h-7 rounded flex justify-center items-center bg-custom-background-80 transition-all hover:bg-custom-background-90 cursor-pointer group md:hidden"
|
||||||
|
onClick={() => themStore.toggleSidebar()}
|
||||||
|
>
|
||||||
|
<Menu size={14} className="text-custom-text-200 group-hover:text-custom-text-100 transition-all" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
});
|
@ -63,34 +63,27 @@ export const OverviewStatsWidget: React.FC<WidgetProps> = observer((props) => {
|
|||||||
if (!widgetStats) return <WidgetLoader widgetKey={WIDGET_KEY} />;
|
if (!widgetStats) return <WidgetLoader widgetKey={WIDGET_KEY} />;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="bg-custom-background-100 rounded-xl border-[0.5px] border-custom-border-200 w-full grid grid-cols-4 p-0.5 hover:shadow-custom-shadow-4xl duration-300">
|
<div
|
||||||
{STATS_LIST.map((stat, index) => {
|
className="bg-custom-background-100 rounded-xl border-[0.5px] border-custom-border-200 w-full grid lg:grid-cols-4 md:grid-cols-2 sm:grid-cols-2 grid-cols-2 p-0.5 hover:shadow-custom-shadow-4xl duration-300
|
||||||
const isFirst = index === 0;
|
[&>div>a>div]:border-r
|
||||||
const isLast = index === STATS_LIST.length - 1;
|
[&>div:last-child>a>div]:border-0
|
||||||
const isMiddle = !isFirst && !isLast;
|
[&>div>a>div]:border-custom-border-200
|
||||||
|
[&>div:nth-child(2)>a>div]:border-0
|
||||||
return (
|
[&>div:nth-child(2)>a>div]:lg:border-r
|
||||||
<div key={stat.key} className="flex relative">
|
"
|
||||||
{!isLast && (
|
|
||||||
<div className="absolute right-0 top-1/2 -translate-y-1/2 h-3/5 w-[0.5px] bg-custom-border-200" />
|
|
||||||
)}
|
|
||||||
<Link
|
|
||||||
href={stat.link}
|
|
||||||
className={cn(
|
|
||||||
`py-4 hover:bg-custom-background-80 duration-300 rounded-[10px] w-full break-words flex flex-col justify-center`,
|
|
||||||
{
|
|
||||||
"pl-11 pr-[4.725rem] mr-0.5": isFirst,
|
|
||||||
"px-[4.725rem] mx-0.5": isMiddle,
|
|
||||||
"px-[4.725rem] ml-0.5": isLast,
|
|
||||||
}
|
|
||||||
)}
|
|
||||||
>
|
>
|
||||||
|
{STATS_LIST.map((stat) => (
|
||||||
|
<div className="w-full flex flex-col gap-2 hover:bg-custom-background-80 rounded-[10px]">
|
||||||
|
<Link href={stat.link} className="py-4 duration-300 rounded-[10px] w-full ">
|
||||||
|
<div className={`relative flex justify-center items-center`}>
|
||||||
|
<div>
|
||||||
<h5 className="font-semibold text-xl">{stat.count}</h5>
|
<h5 className="font-semibold text-xl">{stat.count}</h5>
|
||||||
<p className="text-custom-text-300 text-sm xl:text-base">{stat.title}</p>
|
<p className="text-custom-text-300 text-sm xl:text-base">{stat.title}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</Link>
|
</Link>
|
||||||
</div>
|
</div>
|
||||||
);
|
))}
|
||||||
})}
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@ -3,6 +3,7 @@ export * from "./cycle";
|
|||||||
export * from "./date";
|
export * from "./date";
|
||||||
export * from "./estimate";
|
export * from "./estimate";
|
||||||
export * from "./module";
|
export * from "./module";
|
||||||
|
export * from "./module-select";
|
||||||
export * from "./priority";
|
export * from "./priority";
|
||||||
export * from "./project";
|
export * from "./project";
|
||||||
export * from "./state";
|
export * from "./state";
|
||||||
|
114
web/components/dropdowns/module-select/button.tsx
Normal file
114
web/components/dropdowns/module-select/button.tsx
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
import { FC } from "react";
|
||||||
|
import { twMerge } from "tailwind-merge";
|
||||||
|
import { observer } from "mobx-react-lite";
|
||||||
|
import { ChevronDown, X } from "lucide-react";
|
||||||
|
// hooks
|
||||||
|
import { useModule } from "hooks/store";
|
||||||
|
// ui and components
|
||||||
|
import { DiceIcon, Tooltip } from "@plane/ui";
|
||||||
|
// types
|
||||||
|
import { TModuleSelectButton } from "./types";
|
||||||
|
|
||||||
|
export const ModuleSelectButton: FC<TModuleSelectButton> = observer((props) => {
|
||||||
|
const {
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
placeholder,
|
||||||
|
buttonClassName,
|
||||||
|
buttonVariant,
|
||||||
|
hideIcon,
|
||||||
|
hideText,
|
||||||
|
dropdownArrow,
|
||||||
|
dropdownArrowClassName,
|
||||||
|
showTooltip,
|
||||||
|
showCount,
|
||||||
|
} = props;
|
||||||
|
// hooks
|
||||||
|
const { getModuleById } = useModule();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={twMerge(
|
||||||
|
`w-full h-full relative overflow-hidden flex justify-between items-center gap-1 rounded text-sm px-2`,
|
||||||
|
buttonVariant === "border-with-text"
|
||||||
|
? `border-[0.5px] border-custom-border-300 hover:bg-custom-background-80`
|
||||||
|
: ``,
|
||||||
|
buttonVariant === "border-without-text"
|
||||||
|
? `border-[0.5px] border-custom-border-300 hover:bg-custom-background-80`
|
||||||
|
: ``,
|
||||||
|
buttonVariant === "background-with-text" ? `bg-custom-background-80` : ``,
|
||||||
|
buttonVariant === "background-without-text" ? `bg-custom-background-80` : ``,
|
||||||
|
buttonVariant === "transparent-with-text" ? `hover:bg-custom-background-80` : ``,
|
||||||
|
buttonVariant === "transparent-without-text" ? `hover:bg-custom-background-80` : ``,
|
||||||
|
buttonClassName
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="relative overflow-hidden h-full flex flex-wrap items-center gap-1">
|
||||||
|
{value && typeof value === "string" ? (
|
||||||
|
<div className="relative overflow-hidden flex items-center gap-1.5">
|
||||||
|
{!hideIcon && <DiceIcon className="h-3 w-3 flex-shrink-0" />}
|
||||||
|
{!hideText && (
|
||||||
|
<span className="w-full overflow-hidden truncate inline-block line-clamp-1 capitalize">
|
||||||
|
{getModuleById(value)?.name || placeholder}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
) : value && Array.isArray(value) && value.length > 0 ? (
|
||||||
|
showCount ? (
|
||||||
|
<div className="relative overflow-hidden flex items-center gap-1.5">
|
||||||
|
{!hideIcon && <DiceIcon className="h-3 w-3 flex-shrink-0" />}
|
||||||
|
{!hideText && (
|
||||||
|
<span className="w-full overflow-hidden truncate inline-block line-clamp-1 capitalize">
|
||||||
|
{value.length} Modules
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
value.map((moduleId) => {
|
||||||
|
const _module = getModuleById(moduleId);
|
||||||
|
if (!_module) return <></>;
|
||||||
|
return (
|
||||||
|
<div className="relative flex justify-between items-center gap-1 min-w-[60px] max-w-[84px] overflow-hidden bg-custom-background-80 px-1.5 py-1 rounded">
|
||||||
|
<Tooltip tooltipContent={_module?.name} disabled={!showTooltip}>
|
||||||
|
<div className="relative overflow-hidden flex items-center gap-1.5">
|
||||||
|
{!hideIcon && <DiceIcon className="h-3 w-3 flex-shrink-0" />}
|
||||||
|
{!hideText && (
|
||||||
|
<span className="w-full truncate inline-block line-clamp-1 capitalize">{_module?.name}</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</Tooltip>
|
||||||
|
<Tooltip tooltipContent="Remove" disabled={!showTooltip}>
|
||||||
|
<span
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
onChange(_module.id);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<X className="h-2.5 w-2.5 text-custom-text-300 hover:text-red-500" />
|
||||||
|
</span>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
!hideText && (
|
||||||
|
<div className="relative overflow-hidden flex items-center gap-1.5">
|
||||||
|
{!hideIcon && <DiceIcon className="h-3 w-3 flex-shrink-0" />}
|
||||||
|
{!hideText && (
|
||||||
|
<span className="w-full overflow-hidden truncate inline-block line-clamp-1 capitalize">
|
||||||
|
{placeholder}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{dropdownArrow && (
|
||||||
|
<ChevronDown className={twMerge("h-2.5 w-2.5 flex-shrink-0", dropdownArrowClassName)} aria-hidden="true" />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
});
|
2
web/components/dropdowns/module-select/index.ts
Normal file
2
web/components/dropdowns/module-select/index.ts
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
export * from "./button";
|
||||||
|
export * from "./select";
|
227
web/components/dropdowns/module-select/select.tsx
Normal file
227
web/components/dropdowns/module-select/select.tsx
Normal file
@ -0,0 +1,227 @@
|
|||||||
|
import { FC, useEffect, useRef, useState, Fragment } from "react";
|
||||||
|
import { observer } from "mobx-react-lite";
|
||||||
|
import { Combobox } from "@headlessui/react";
|
||||||
|
import { usePopper } from "react-popper";
|
||||||
|
import { Check, Search } from "lucide-react";
|
||||||
|
import { twMerge } from "tailwind-merge";
|
||||||
|
// hooks
|
||||||
|
import { useModule } from "hooks/store";
|
||||||
|
import { useDropdownKeyDown } from "hooks/use-dropdown-key-down";
|
||||||
|
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
||||||
|
// components
|
||||||
|
import { ModuleSelectButton } from "./";
|
||||||
|
// types
|
||||||
|
import { TModuleSelectDropdown, TModuleSelectDropdownOption } from "./types";
|
||||||
|
import { DiceIcon } from "@plane/ui";
|
||||||
|
|
||||||
|
export const ModuleSelectDropdown: FC<TModuleSelectDropdown> = observer((props) => {
|
||||||
|
// props
|
||||||
|
const {
|
||||||
|
workspaceSlug,
|
||||||
|
projectId,
|
||||||
|
value = undefined,
|
||||||
|
onChange,
|
||||||
|
placeholder = "Module",
|
||||||
|
multiple = false,
|
||||||
|
disabled = false,
|
||||||
|
className = "",
|
||||||
|
buttonContainerClassName = "",
|
||||||
|
buttonClassName = "",
|
||||||
|
buttonVariant = "transparent-with-text",
|
||||||
|
hideIcon = false,
|
||||||
|
dropdownArrow = false,
|
||||||
|
dropdownArrowClassName = "",
|
||||||
|
showTooltip = false,
|
||||||
|
showCount = false,
|
||||||
|
placement,
|
||||||
|
tabIndex,
|
||||||
|
button,
|
||||||
|
} = props;
|
||||||
|
// states
|
||||||
|
const [query, setQuery] = useState("");
|
||||||
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
// refs
|
||||||
|
const dropdownRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
// popper-js refs
|
||||||
|
const [referenceElement, setReferenceElement] = useState<HTMLButtonElement | null>(null);
|
||||||
|
const [popperElement, setPopperElement] = useState<HTMLDivElement | null>(null);
|
||||||
|
// popper-js init
|
||||||
|
const { styles, attributes } = usePopper(referenceElement, popperElement, {
|
||||||
|
placement: placement ?? "bottom-start",
|
||||||
|
modifiers: [
|
||||||
|
{
|
||||||
|
name: "preventOverflow",
|
||||||
|
options: {
|
||||||
|
padding: 12,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
// store hooks
|
||||||
|
const { getProjectModuleIds, fetchModules, getModuleById } = useModule();
|
||||||
|
|
||||||
|
const moduleIds = getProjectModuleIds(projectId);
|
||||||
|
|
||||||
|
const options: TModuleSelectDropdownOption[] | undefined = moduleIds?.map((moduleId) => {
|
||||||
|
const moduleDetails = getModuleById(moduleId);
|
||||||
|
return {
|
||||||
|
value: moduleId,
|
||||||
|
query: `${moduleDetails?.name}`,
|
||||||
|
content: (
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<DiceIcon className="h-3 w-3 flex-shrink-0" />
|
||||||
|
<span className="flex-grow truncate">{moduleDetails?.name}</span>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
!multiple &&
|
||||||
|
options?.unshift({
|
||||||
|
value: undefined,
|
||||||
|
query: "No module",
|
||||||
|
content: (
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<DiceIcon className="h-3 w-3 flex-shrink-0" />
|
||||||
|
<span className="flex-grow truncate">No module</span>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
});
|
||||||
|
|
||||||
|
const filteredOptions =
|
||||||
|
query === "" ? options : options?.filter((o) => o.query.toLowerCase().includes(query.toLowerCase()));
|
||||||
|
|
||||||
|
// fetch modules of the project if not already present in the store
|
||||||
|
useEffect(() => {
|
||||||
|
if (!workspaceSlug) return;
|
||||||
|
|
||||||
|
if (!moduleIds) fetchModules(workspaceSlug, projectId);
|
||||||
|
}, [moduleIds, fetchModules, projectId, workspaceSlug]);
|
||||||
|
|
||||||
|
const openDropdown = () => {
|
||||||
|
if (isOpen) closeDropdown();
|
||||||
|
else {
|
||||||
|
setIsOpen(true);
|
||||||
|
if (referenceElement) referenceElement.focus();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const closeDropdown = () => setIsOpen(false);
|
||||||
|
const handleKeyDown = useDropdownKeyDown(openDropdown, closeDropdown, isOpen);
|
||||||
|
useOutsideClickDetector(dropdownRef, closeDropdown);
|
||||||
|
|
||||||
|
const comboboxProps: any = {};
|
||||||
|
if (multiple) comboboxProps.multiple = true;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Combobox
|
||||||
|
as="div"
|
||||||
|
ref={dropdownRef}
|
||||||
|
tabIndex={tabIndex}
|
||||||
|
className={twMerge("h-full", className)}
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
disabled={disabled}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
{...comboboxProps}
|
||||||
|
>
|
||||||
|
<Combobox.Button as={Fragment}>
|
||||||
|
{button ? (
|
||||||
|
<button
|
||||||
|
ref={setReferenceElement}
|
||||||
|
type="button"
|
||||||
|
className={twMerge(
|
||||||
|
"block h-full max-w-full outline-none",
|
||||||
|
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer",
|
||||||
|
buttonContainerClassName
|
||||||
|
)}
|
||||||
|
onClick={openDropdown}
|
||||||
|
>
|
||||||
|
{button}
|
||||||
|
</button>
|
||||||
|
) : (
|
||||||
|
<button
|
||||||
|
ref={setReferenceElement}
|
||||||
|
type="button"
|
||||||
|
className={twMerge(
|
||||||
|
"block h-full max-w-full outline-none ",
|
||||||
|
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer",
|
||||||
|
buttonContainerClassName
|
||||||
|
)}
|
||||||
|
onClick={openDropdown}
|
||||||
|
>
|
||||||
|
<ModuleSelectButton
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
placeholder={placeholder}
|
||||||
|
disabled={disabled}
|
||||||
|
buttonClassName={buttonClassName}
|
||||||
|
buttonVariant={buttonVariant}
|
||||||
|
hideIcon={hideIcon}
|
||||||
|
hideText={["border-without-text", "background-without-text", "transparent-without-text"].includes(
|
||||||
|
buttonVariant
|
||||||
|
)}
|
||||||
|
dropdownArrow={dropdownArrow}
|
||||||
|
dropdownArrowClassName={dropdownArrowClassName}
|
||||||
|
showTooltip={showTooltip}
|
||||||
|
showCount={showCount}
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</Combobox.Button>
|
||||||
|
{isOpen && (
|
||||||
|
<Combobox.Options className="fixed z-10" static>
|
||||||
|
<div
|
||||||
|
className="my-1 w-48 rounded border-[0.5px] border-custom-border-300 bg-custom-background-100 px-2 py-2.5 text-xs shadow-custom-shadow-rg focus:outline-none"
|
||||||
|
ref={setPopperElement}
|
||||||
|
style={styles.popper}
|
||||||
|
{...attributes.popper}
|
||||||
|
>
|
||||||
|
<div className="flex items-center gap-1.5 rounded border border-custom-border-100 bg-custom-background-90 px-2">
|
||||||
|
<Search className="h-3.5 w-3.5 text-custom-text-400" strokeWidth={1.5} />
|
||||||
|
<Combobox.Input
|
||||||
|
className="w-full bg-transparent py-1 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
|
||||||
|
value={query}
|
||||||
|
onChange={(e) => setQuery(e.target.value)}
|
||||||
|
placeholder="Search"
|
||||||
|
displayValue={(moduleIds: any) => {
|
||||||
|
const displayValueOptions: TModuleSelectDropdownOption[] | undefined = options?.filter((_module) =>
|
||||||
|
moduleIds.includes(_module.value)
|
||||||
|
);
|
||||||
|
return displayValueOptions?.map((_option) => _option.query).join(", ") || "Select Module";
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="mt-2 max-h-48 space-y-1 overflow-y-scroll">
|
||||||
|
{filteredOptions ? (
|
||||||
|
filteredOptions.length > 0 ? (
|
||||||
|
filteredOptions.map((option) => (
|
||||||
|
<Combobox.Option
|
||||||
|
key={option.value}
|
||||||
|
value={option.value}
|
||||||
|
className={({ active, selected }) =>
|
||||||
|
`w-full truncate flex items-center justify-between gap-2 rounded px-1 py-1.5 cursor-pointer select-none ${
|
||||||
|
active ? "bg-custom-background-80" : ""
|
||||||
|
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
|
||||||
|
}
|
||||||
|
onClick={() => !multiple && closeDropdown()}
|
||||||
|
>
|
||||||
|
{({ selected }) => (
|
||||||
|
<>
|
||||||
|
<span className="flex-grow truncate">{option.content}</span>
|
||||||
|
{selected && <Check className="h-3.5 w-3.5 flex-shrink-0" />}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Combobox.Option>
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
<p className="text-custom-text-400 italic py-1 px-1.5">No matching results</p>
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<p className="text-custom-text-400 italic py-1 px-1.5">Loading...</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Combobox.Options>
|
||||||
|
)}
|
||||||
|
</Combobox>
|
||||||
|
);
|
||||||
|
});
|
50
web/components/dropdowns/module-select/types.d.ts
vendored
Normal file
50
web/components/dropdowns/module-select/types.d.ts
vendored
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
import { ReactNode } from "react";
|
||||||
|
import { Placement } from "@popperjs/core";
|
||||||
|
import { TDropdownProps, TButtonVariants } from "../types";
|
||||||
|
|
||||||
|
type TModuleSelectDropdownRoot = Omit<
|
||||||
|
TDropdownProps,
|
||||||
|
"buttonClassName",
|
||||||
|
"buttonContainerClassName",
|
||||||
|
"buttonContainerClassName",
|
||||||
|
"className",
|
||||||
|
"disabled",
|
||||||
|
"hideIcon",
|
||||||
|
"placeholder",
|
||||||
|
"placement",
|
||||||
|
"tabIndex",
|
||||||
|
"tooltip"
|
||||||
|
>;
|
||||||
|
|
||||||
|
export type TModuleSelectDropdownBase = {
|
||||||
|
value: string | string[] | undefined;
|
||||||
|
onChange: (moduleIds: undefined | string | (string | undefined)[]) => void;
|
||||||
|
placeholder?: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
buttonClassName?: string;
|
||||||
|
buttonVariant?: TButtonVariants;
|
||||||
|
hideIcon?: boolean;
|
||||||
|
dropdownArrow?: boolean;
|
||||||
|
dropdownArrowClassName?: string;
|
||||||
|
showTooltip?: boolean;
|
||||||
|
showCount?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type TModuleSelectButton = TModuleSelectDropdownBase & { hideText?: boolean };
|
||||||
|
|
||||||
|
export type TModuleSelectDropdown = TModuleSelectDropdownBase & {
|
||||||
|
workspaceSlug: string;
|
||||||
|
projectId: string;
|
||||||
|
multiple?: boolean;
|
||||||
|
className?: string;
|
||||||
|
buttonContainerClassName?: string;
|
||||||
|
placement?: Placement;
|
||||||
|
tabIndex?: number;
|
||||||
|
button?: ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type TModuleSelectDropdownOption = {
|
||||||
|
value: string | undefined;
|
||||||
|
query: string;
|
||||||
|
content: JSX.Element;
|
||||||
|
};
|
@ -1,9 +1,11 @@
|
|||||||
import { FC } from "react";
|
import { FC } from "react";
|
||||||
// hooks
|
// hooks
|
||||||
|
import { useIssueDetail } from "hooks/store";
|
||||||
import { useChart } from "../hooks";
|
import { useChart } from "../hooks";
|
||||||
// helpers
|
// helpers
|
||||||
import { ChartAddBlock, ChartDraggable } from "components/gantt-chart";
|
import { ChartAddBlock, ChartDraggable } from "components/gantt-chart";
|
||||||
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
import { renderFormattedPayloadDate } from "helpers/date-time.helper";
|
||||||
|
import { cn } from "helpers/common.helper";
|
||||||
// types
|
// types
|
||||||
import { IBlockUpdateData, IGanttBlock } from "../types";
|
import { IBlockUpdateData, IGanttBlock } from "../types";
|
||||||
|
|
||||||
@ -31,6 +33,7 @@ export const GanttChartBlocks: FC<GanttChartBlocksProps> = (props) => {
|
|||||||
} = props;
|
} = props;
|
||||||
|
|
||||||
const { activeBlock, dispatch } = useChart();
|
const { activeBlock, dispatch } = useChart();
|
||||||
|
const { peekIssue } = useIssueDetail();
|
||||||
|
|
||||||
// update the active block on hover
|
// update the active block on hover
|
||||||
const updateActiveBlock = (block: IGanttBlock | null) => {
|
const updateActiveBlock = (block: IGanttBlock | null) => {
|
||||||
@ -88,7 +91,14 @@ export const GanttChartBlocks: FC<GanttChartBlocksProps> = (props) => {
|
|||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
key={`block-${block.id}`}
|
key={`block-${block.id}`}
|
||||||
className={`h-11 ${activeBlock?.id === block.id ? "bg-custom-background-80" : ""}`}
|
className={cn(
|
||||||
|
"h-11",
|
||||||
|
{ "rounded bg-custom-background-80": activeBlock?.id === block.id },
|
||||||
|
{
|
||||||
|
"rounded-l border border-r-0 border-custom-primary-70 hover:border-custom-primary-70":
|
||||||
|
peekIssue?.issueId === block.data.id,
|
||||||
|
}
|
||||||
|
)}
|
||||||
onMouseEnter={() => updateActiveBlock(block)}
|
onMouseEnter={() => updateActiveBlock(block)}
|
||||||
onMouseLeave={() => updateActiveBlock(null)}
|
onMouseLeave={() => updateActiveBlock(null)}
|
||||||
>
|
>
|
||||||
|
@ -3,12 +3,14 @@ import { DragDropContext, Draggable, Droppable, DropResult } from "@hello-pangea
|
|||||||
import { MoreVertical } from "lucide-react";
|
import { MoreVertical } from "lucide-react";
|
||||||
// hooks
|
// hooks
|
||||||
import { useChart } from "components/gantt-chart/hooks";
|
import { useChart } from "components/gantt-chart/hooks";
|
||||||
|
import { useIssueDetail } from "hooks/store";
|
||||||
// ui
|
// ui
|
||||||
import { Loader } from "@plane/ui";
|
import { Loader } from "@plane/ui";
|
||||||
// components
|
// components
|
||||||
import { GanttQuickAddIssueForm, IssueGanttSidebarBlock } from "components/issues";
|
import { GanttQuickAddIssueForm, IssueGanttSidebarBlock } from "components/issues";
|
||||||
// helpers
|
// helpers
|
||||||
import { findTotalDaysInRange } from "helpers/date-time.helper";
|
import { findTotalDaysInRange } from "helpers/date-time.helper";
|
||||||
|
import { cn } from "helpers/common.helper";
|
||||||
// types
|
// types
|
||||||
import { IGanttBlock, IBlockUpdateData } from "components/gantt-chart/types";
|
import { IGanttBlock, IBlockUpdateData } from "components/gantt-chart/types";
|
||||||
import { TIssue } from "@plane/types";
|
import { TIssue } from "@plane/types";
|
||||||
@ -45,6 +47,7 @@ export const IssueGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
const { cycleId } = router.query;
|
const { cycleId } = router.query;
|
||||||
|
|
||||||
const { activeBlock, dispatch } = useChart();
|
const { activeBlock, dispatch } = useChart();
|
||||||
|
const { peekIssue } = useIssueDetail();
|
||||||
|
|
||||||
// update the active block on hover
|
// update the active block on hover
|
||||||
const updateActiveBlock = (block: IGanttBlock | null) => {
|
const updateActiveBlock = (block: IGanttBlock | null) => {
|
||||||
@ -104,7 +107,7 @@ export const IssueGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
{(droppableProvided) => (
|
{(droppableProvided) => (
|
||||||
<div
|
<div
|
||||||
id={`gantt-sidebar-${cycleId}`}
|
id={`gantt-sidebar-${cycleId}`}
|
||||||
className="mt-3 max-h-full overflow-y-auto pl-2.5"
|
className="mt-[12px] max-h-full overflow-y-auto pl-2.5"
|
||||||
ref={droppableProvided.innerRef}
|
ref={droppableProvided.innerRef}
|
||||||
{...droppableProvided.droppableProps}
|
{...droppableProvided.droppableProps}
|
||||||
>
|
>
|
||||||
@ -130,7 +133,14 @@ export const IssueGanttSidebar: React.FC<Props> = (props) => {
|
|||||||
>
|
>
|
||||||
{(provided, snapshot) => (
|
{(provided, snapshot) => (
|
||||||
<div
|
<div
|
||||||
className={`h-11 ${snapshot.isDragging ? "rounded bg-custom-background-80" : ""}`}
|
className={cn(
|
||||||
|
"h-11",
|
||||||
|
{ "rounded bg-custom-background-80": snapshot.isDragging },
|
||||||
|
{
|
||||||
|
"rounded-l border border-r-0 border-custom-primary-70 hover:border-custom-primary-70":
|
||||||
|
peekIssue?.issueId === block.data.id,
|
||||||
|
}
|
||||||
|
)}
|
||||||
onMouseEnter={() => updateActiveBlock(block)}
|
onMouseEnter={() => updateActiveBlock(block)}
|
||||||
onMouseLeave={() => updateActiveBlock(null)}
|
onMouseLeave={() => updateActiveBlock(null)}
|
||||||
ref={provided.innerRef}
|
ref={provided.innerRef}
|
||||||
|
@ -17,6 +17,7 @@ import useLocalStorage from "hooks/use-local-storage";
|
|||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
||||||
import { ProjectAnalyticsModal } from "components/analytics";
|
import { ProjectAnalyticsModal } from "components/analytics";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs, Button, ContrastIcon, CustomMenu } from "@plane/ui";
|
import { Breadcrumbs, Button, ContrastIcon, CustomMenu } from "@plane/ui";
|
||||||
// icons
|
// icons
|
||||||
@ -146,6 +147,7 @@ export const CycleIssuesHeader: React.FC = observer(() => {
|
|||||||
/>
|
/>
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
|
<SidebarHamburgerToggle />
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
type="text"
|
type="text"
|
||||||
|
@ -9,6 +9,8 @@ import { Breadcrumbs, Button, ContrastIcon } from "@plane/ui";
|
|||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
import { EUserProjectRoles } from "constants/project";
|
import { EUserProjectRoles } from "constants/project";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const CyclesHeader: FC = observer(() => {
|
export const CyclesHeader: FC = observer(() => {
|
||||||
// router
|
// router
|
||||||
@ -30,6 +32,7 @@ export const CyclesHeader: FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -7,6 +7,7 @@ import { useLabel, useMember, useUser, useIssues } from "hooks/store";
|
|||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection } from "components/issues";
|
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection } from "components/issues";
|
||||||
import { CreateUpdateWorkspaceViewModal } from "components/workspace";
|
import { CreateUpdateWorkspaceViewModal } from "components/workspace";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs, Button, LayersIcon, PhotoFilterIcon, Tooltip } from "@plane/ui";
|
import { Breadcrumbs, Button, LayersIcon, PhotoFilterIcon, Tooltip } from "@plane/ui";
|
||||||
// icons
|
// icons
|
||||||
@ -106,7 +107,8 @@ export const GlobalIssuesHeader: React.FC<Props> = observer((props) => {
|
|||||||
<>
|
<>
|
||||||
<CreateUpdateWorkspaceViewModal isOpen={createViewModal} onClose={() => setCreateViewModal(false)} />
|
<CreateUpdateWorkspaceViewModal isOpen={createViewModal} onClose={() => setCreateViewModal(false)} />
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div>
|
<div className="relative flex gap-2">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
type="text"
|
type="text"
|
||||||
|
@ -17,6 +17,7 @@ import useLocalStorage from "hooks/use-local-storage";
|
|||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
||||||
import { ProjectAnalyticsModal } from "components/analytics";
|
import { ProjectAnalyticsModal } from "components/analytics";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs, Button, CustomMenu, DiceIcon } from "@plane/ui";
|
import { Breadcrumbs, Button, CustomMenu, DiceIcon } from "@plane/ui";
|
||||||
// icons
|
// icons
|
||||||
@ -149,6 +150,7 @@ export const ModuleIssuesHeader: React.FC = observer(() => {
|
|||||||
/>
|
/>
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
|
<SidebarHamburgerToggle />
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
type="text"
|
type="text"
|
||||||
|
@ -11,6 +11,8 @@ import { renderEmoji } from "helpers/emoji.helper";
|
|||||||
// constants
|
// constants
|
||||||
import { MODULE_VIEW_LAYOUTS } from "constants/module";
|
import { MODULE_VIEW_LAYOUTS } from "constants/module";
|
||||||
import { EUserProjectRoles } from "constants/project";
|
import { EUserProjectRoles } from "constants/project";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const ModulesListHeader: React.FC = observer(() => {
|
export const ModulesListHeader: React.FC = observer(() => {
|
||||||
// router
|
// router
|
||||||
@ -31,6 +33,7 @@ export const ModulesListHeader: React.FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -8,6 +8,8 @@ import { useApplication, usePage, useProject } from "hooks/store";
|
|||||||
import { Breadcrumbs, Button } from "@plane/ui";
|
import { Breadcrumbs, Button } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export interface IPagesHeaderProps {
|
export interface IPagesHeaderProps {
|
||||||
showButton?: boolean;
|
showButton?: boolean;
|
||||||
@ -27,6 +29,7 @@ export const PageDetailsHeader: FC<IPagesHeaderProps> = observer((props) => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -9,6 +9,8 @@ import { Breadcrumbs, Button } from "@plane/ui";
|
|||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
// constants
|
// constants
|
||||||
import { EUserProjectRoles } from "constants/project";
|
import { EUserProjectRoles } from "constants/project";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const PagesHeader = observer(() => {
|
export const PagesHeader = observer(() => {
|
||||||
// router
|
// router
|
||||||
@ -29,6 +31,7 @@ export const PagesHeader = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -14,6 +14,8 @@ import { ISSUE_DETAILS } from "constants/fetch-keys";
|
|||||||
import { IssueArchiveService } from "services/issue";
|
import { IssueArchiveService } from "services/issue";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
const issueArchiveService = new IssueArchiveService();
|
const issueArchiveService = new IssueArchiveService();
|
||||||
|
|
||||||
@ -39,6 +41,7 @@ export const ProjectArchivedIssueDetailsHeader: FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -10,6 +10,7 @@ import { EIssueFilterType, EIssuesStoreType, ISSUE_DISPLAY_FILTERS_BY_LAYOUT } f
|
|||||||
import { Breadcrumbs, LayersIcon } from "@plane/ui";
|
import { Breadcrumbs, LayersIcon } from "@plane/ui";
|
||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FilterSelection, FiltersDropdown } from "components/issues";
|
import { DisplayFiltersSelection, FilterSelection, FiltersDropdown } from "components/issues";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
// types
|
// types
|
||||||
@ -70,6 +71,7 @@ export const ProjectArchivedIssuesHeader: FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-14 w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-14 w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div className="block md:hidden">
|
<div className="block md:hidden">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
@ -5,6 +5,7 @@ import { observer } from "mobx-react-lite";
|
|||||||
import { useIssues, useLabel, useMember, useProject, useProjectState } from "hooks/store";
|
import { useIssues, useLabel, useMember, useProject, useProjectState } from "hooks/store";
|
||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs, LayersIcon } from "@plane/ui";
|
import { Breadcrumbs, LayersIcon } from "@plane/ui";
|
||||||
// helper
|
// helper
|
||||||
@ -74,6 +75,7 @@ export const ProjectDraftIssueHeader: FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -8,6 +8,7 @@ import { useProject } from "hooks/store";
|
|||||||
import { Breadcrumbs, Button, LayersIcon } from "@plane/ui";
|
import { Breadcrumbs, Button, LayersIcon } from "@plane/ui";
|
||||||
// components
|
// components
|
||||||
import { CreateInboxIssueModal } from "components/inbox";
|
import { CreateInboxIssueModal } from "components/inbox";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// helper
|
// helper
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
|
|
||||||
@ -23,6 +24,7 @@ export const ProjectInboxHeader: FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -12,6 +12,8 @@ import { renderEmoji } from "helpers/emoji.helper";
|
|||||||
import { IssueService } from "services/issue";
|
import { IssueService } from "services/issue";
|
||||||
// constants
|
// constants
|
||||||
import { ISSUE_DETAILS } from "constants/fetch-keys";
|
import { ISSUE_DETAILS } from "constants/fetch-keys";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
// services
|
// services
|
||||||
const issueService = new IssueService();
|
const issueService = new IssueService();
|
||||||
@ -33,6 +35,7 @@ export const ProjectIssueDetailsHeader: FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -8,6 +8,7 @@ import { useApplication, useLabel, useProject, useProjectState, useUser, useInbo
|
|||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
||||||
import { ProjectAnalyticsModal } from "components/analytics";
|
import { ProjectAnalyticsModal } from "components/analytics";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs, Button, LayersIcon } from "@plane/ui";
|
import { Breadcrumbs, Button, LayersIcon } from "@plane/ui";
|
||||||
// types
|
// types
|
||||||
@ -105,6 +106,7 @@ export const ProjectIssuesHeader: React.FC = observer(() => {
|
|||||||
/>
|
/>
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div className="block md:hidden">
|
<div className="block md:hidden">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
@ -9,6 +9,8 @@ import { renderEmoji } from "helpers/emoji.helper";
|
|||||||
import { useProject, useUser } from "hooks/store";
|
import { useProject, useUser } from "hooks/store";
|
||||||
// constants
|
// constants
|
||||||
import { EUserProjectRoles } from "constants/project";
|
import { EUserProjectRoles } from "constants/project";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export interface IProjectSettingHeader {
|
export interface IProjectSettingHeader {
|
||||||
title: string;
|
title: string;
|
||||||
@ -30,6 +32,7 @@ export const ProjectSettingHeader: FC<IProjectSettingHeader> = observer((props)
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -16,6 +16,7 @@ import {
|
|||||||
} from "hooks/store";
|
} from "hooks/store";
|
||||||
// components
|
// components
|
||||||
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs, Button, CustomMenu, PhotoFilterIcon } from "@plane/ui";
|
import { Breadcrumbs, Button, CustomMenu, PhotoFilterIcon } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
@ -107,6 +108,7 @@ export const ProjectViewIssuesHeader: React.FC = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
|
<SidebarHamburgerToggle />
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
type="text"
|
type="text"
|
||||||
|
@ -5,6 +5,7 @@ import { Plus } from "lucide-react";
|
|||||||
import { useApplication, useProject, useUser } from "hooks/store";
|
import { useApplication, useProject, useUser } from "hooks/store";
|
||||||
// components
|
// components
|
||||||
import { Breadcrumbs, PhotoFilterIcon, Button } from "@plane/ui";
|
import { Breadcrumbs, PhotoFilterIcon, Button } from "@plane/ui";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
// helpers
|
// helpers
|
||||||
import { renderEmoji } from "helpers/emoji.helper";
|
import { renderEmoji } from "helpers/emoji.helper";
|
||||||
// constants
|
// constants
|
||||||
@ -30,6 +31,7 @@ export const ProjectViewsHeader: React.FC = observer(() => {
|
|||||||
<>
|
<>
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -6,6 +6,8 @@ import { useApplication, useProject, useUser } from "hooks/store";
|
|||||||
import { Breadcrumbs, Button } from "@plane/ui";
|
import { Breadcrumbs, Button } from "@plane/ui";
|
||||||
// constants
|
// constants
|
||||||
import { EUserWorkspaceRoles } from "constants/workspace";
|
import { EUserWorkspaceRoles } from "constants/workspace";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const ProjectsHeader = observer(() => {
|
export const ProjectsHeader = observer(() => {
|
||||||
// store hooks
|
// store hooks
|
||||||
@ -23,6 +25,7 @@ export const ProjectsHeader = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -1,9 +1,12 @@
|
|||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs } from "@plane/ui";
|
import { Breadcrumbs } from "@plane/ui";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const UserProfileHeader = () => (
|
export const UserProfileHeader = () => (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem type="text" label="Activity Overview" link="/profile" />
|
<Breadcrumbs.BreadcrumbItem type="text" label="Activity Overview" link="/profile" />
|
||||||
|
@ -3,10 +3,12 @@ import { observer } from "mobx-react-lite";
|
|||||||
import { Breadcrumbs, ContrastIcon } from "@plane/ui";
|
import { Breadcrumbs, ContrastIcon } from "@plane/ui";
|
||||||
// icons
|
// icons
|
||||||
import { Crown } from "lucide-react";
|
import { Crown } from "lucide-react";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const WorkspaceActiveCycleHeader = observer(() => (
|
export const WorkspaceActiveCycleHeader = observer(() => (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle />
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -2,6 +2,8 @@ import { useRouter } from "next/router";
|
|||||||
import { ArrowLeft, BarChart2 } from "lucide-react";
|
import { ArrowLeft, BarChart2 } from "lucide-react";
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs } from "@plane/ui";
|
import { Breadcrumbs } from "@plane/ui";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const WorkspaceAnalyticsHeader = () => {
|
export const WorkspaceAnalyticsHeader = () => {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
@ -12,6 +14,7 @@ export const WorkspaceAnalyticsHeader = () => {
|
|||||||
className={`relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4`}
|
className={`relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4`}
|
||||||
>
|
>
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle />
|
||||||
<div className="block md:hidden">
|
<div className="block md:hidden">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
@ -8,10 +8,11 @@ import githubWhiteImage from "/public/logos/github-white.png";
|
|||||||
// components
|
// components
|
||||||
import { ProductUpdatesModal } from "components/common";
|
import { ProductUpdatesModal } from "components/common";
|
||||||
import { Breadcrumbs } from "@plane/ui";
|
import { Breadcrumbs } from "@plane/ui";
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export const WorkspaceDashboardHeader = () => {
|
export const WorkspaceDashboardHeader = () => {
|
||||||
const [isProductUpdatesModalOpen, setIsProductUpdatesModalOpen] = useState(false);
|
const [isProductUpdatesModalOpen, setIsProductUpdatesModalOpen] = useState(false);
|
||||||
// theme
|
// hooks
|
||||||
const { resolvedTheme } = useTheme();
|
const { resolvedTheme } = useTheme();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -19,6 +20,7 @@ export const WorkspaceDashboardHeader = () => {
|
|||||||
<ProductUpdatesModal isOpen={isProductUpdatesModalOpen} setIsOpen={setIsProductUpdatesModalOpen} />
|
<ProductUpdatesModal isOpen={isProductUpdatesModalOpen} setIsOpen={setIsProductUpdatesModalOpen} />
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle />
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -1,13 +1,12 @@
|
|||||||
import { FC } from "react";
|
import { FC } from "react";
|
||||||
|
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
// ui
|
// ui
|
||||||
import { Breadcrumbs } from "@plane/ui";
|
import { Breadcrumbs } from "@plane/ui";
|
||||||
import { Settings } from "lucide-react";
|
import { Settings } from "lucide-react";
|
||||||
|
|
||||||
// hooks
|
// hooks
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
// components
|
||||||
|
import { SidebarHamburgerToggle } from "components/core/sidebar/sidebar-menu-hamburger-toggle";
|
||||||
|
|
||||||
export interface IWorkspaceSettingHeader {
|
export interface IWorkspaceSettingHeader {
|
||||||
title: string;
|
title: string;
|
||||||
@ -22,6 +21,7 @@ export const WorkspaceSettingHeader: FC<IWorkspaceSettingHeader> = observer((pro
|
|||||||
return (
|
return (
|
||||||
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
<div className="relative z-10 flex h-[3.75rem] w-full flex-shrink-0 flex-row items-center justify-between gap-x-2 gap-y-4 border-b border-custom-border-200 bg-custom-sidebar-background-100 p-4">
|
||||||
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
<div className="flex w-full flex-grow items-center gap-2 overflow-ellipsis whitespace-nowrap">
|
||||||
|
<SidebarHamburgerToggle/>
|
||||||
<div>
|
<div>
|
||||||
<Breadcrumbs>
|
<Breadcrumbs>
|
||||||
<Breadcrumbs.BreadcrumbItem
|
<Breadcrumbs.BreadcrumbItem
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { FC } from "react";
|
import { FC, useState } from "react";
|
||||||
import { useForm, Controller } from "react-hook-form";
|
import { useForm, Controller } from "react-hook-form";
|
||||||
import { XCircle } from "lucide-react";
|
import { Eye, EyeOff, XCircle } from "lucide-react";
|
||||||
// hooks
|
// hooks
|
||||||
import { useUser } from "hooks/store";
|
import { useUser } from "hooks/store";
|
||||||
// ui
|
// ui
|
||||||
@ -24,6 +24,8 @@ export interface IInstanceSetupEmailForm {
|
|||||||
|
|
||||||
export const InstanceSetupSignInForm: FC<IInstanceSetupEmailForm> = (props) => {
|
export const InstanceSetupSignInForm: FC<IInstanceSetupEmailForm> = (props) => {
|
||||||
const { handleNextStep } = props;
|
const { handleNextStep } = props;
|
||||||
|
// states
|
||||||
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
// store hooks
|
// store hooks
|
||||||
const { fetchCurrentUser } = useUser();
|
const { fetchCurrentUser } = useUser();
|
||||||
// form info
|
// form info
|
||||||
@ -107,14 +109,27 @@ export const InstanceSetupSignInForm: FC<IInstanceSetupEmailForm> = (props) => {
|
|||||||
required: "Password is required",
|
required: "Password is required",
|
||||||
}}
|
}}
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
|
<div className="relative flex items-center rounded-md bg-onboarding-background-200">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type={showPassword ? "text" : "password"}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
hasError={Boolean(errors.password)}
|
hasError={Boolean(errors.password)}
|
||||||
placeholder="Enter password"
|
placeholder="Enter password"
|
||||||
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
||||||
/>
|
/>
|
||||||
|
{showPassword ? (
|
||||||
|
<EyeOff
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(false)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Eye
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<p className="pb-2 text-xs text-custom-text-200">
|
<p className="pb-2 text-xs text-custom-text-200">
|
||||||
|
@ -21,7 +21,7 @@ import {
|
|||||||
CycleDropdown,
|
CycleDropdown,
|
||||||
DateDropdown,
|
DateDropdown,
|
||||||
EstimateDropdown,
|
EstimateDropdown,
|
||||||
ModuleDropdown,
|
ModuleSelectDropdown,
|
||||||
PriorityDropdown,
|
PriorityDropdown,
|
||||||
ProjectDropdown,
|
ProjectDropdown,
|
||||||
ProjectMemberDropdown,
|
ProjectMemberDropdown,
|
||||||
@ -152,7 +152,7 @@ export const DraftIssueForm: FC<IssueFormProps> = observer((props) => {
|
|||||||
project_id: watch("project_id"),
|
project_id: watch("project_id"),
|
||||||
parent_id: watch("parent_id"),
|
parent_id: watch("parent_id"),
|
||||||
cycle_id: watch("cycle_id"),
|
cycle_id: watch("cycle_id"),
|
||||||
module_id: watch("module_id"),
|
module_ids: watch("module_ids"),
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@ -570,15 +570,17 @@ export const DraftIssueForm: FC<IssueFormProps> = observer((props) => {
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{projectDetails?.module_view && (
|
|
||||||
|
{projectDetails?.module_view && workspaceSlug && (
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="module_id"
|
name="module_ids"
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
<div className="h-7">
|
<div className="h-7">
|
||||||
<ModuleDropdown
|
<ModuleSelectDropdown
|
||||||
|
workspaceSlug={workspaceSlug?.toString()}
|
||||||
projectId={projectId}
|
projectId={projectId}
|
||||||
value={value}
|
value={value || undefined}
|
||||||
onChange={(moduleId) => onChange(moduleId)}
|
onChange={(moduleId) => onChange(moduleId)}
|
||||||
buttonVariant="border-with-text"
|
buttonVariant="border-with-text"
|
||||||
/>
|
/>
|
||||||
@ -586,6 +588,7 @@ export const DraftIssueForm: FC<IssueFormProps> = observer((props) => {
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{(fieldsToShow.includes("all") || fieldsToShow.includes("estimate")) &&
|
{(fieldsToShow.includes("all") || fieldsToShow.includes("estimate")) &&
|
||||||
areEstimatesEnabledForProject(projectId) && (
|
areEstimatesEnabledForProject(projectId) && (
|
||||||
<Controller
|
<Controller
|
||||||
|
@ -94,7 +94,7 @@ export const CreateUpdateDraftIssueModal: React.FC<IssuesModalProps> = observer(
|
|||||||
cycle: cycleId.toString(),
|
cycle: cycleId.toString(),
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
if (moduleId && !prePopulateDataProps?.module_id) {
|
if (moduleId && !prePopulateDataProps?.module_ids) {
|
||||||
setPreloadedData((prevData) => ({
|
setPreloadedData((prevData) => ({
|
||||||
...(prevData ?? {}),
|
...(prevData ?? {}),
|
||||||
...prePopulateDataProps,
|
...prePopulateDataProps,
|
||||||
@ -123,7 +123,7 @@ export const CreateUpdateDraftIssueModal: React.FC<IssuesModalProps> = observer(
|
|||||||
cycle: cycleId.toString(),
|
cycle: cycleId.toString(),
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
if (moduleId && !prePopulateDataProps?.module_id) {
|
if (moduleId && !prePopulateDataProps?.module_ids) {
|
||||||
setPreloadedData((prevData) => ({
|
setPreloadedData((prevData) => ({
|
||||||
...(prevData ?? {}),
|
...(prevData ?? {}),
|
||||||
...prePopulateDataProps,
|
...prePopulateDataProps,
|
||||||
@ -233,11 +233,11 @@ export const CreateUpdateDraftIssueModal: React.FC<IssuesModalProps> = observer(
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const addIssueToModule = async (issueId: string, moduleId: string) => {
|
const addIssueToModule = async (issueId: string, moduleIds: string[]) => {
|
||||||
if (!workspaceSlug || !activeProject) return;
|
if (!workspaceSlug || !activeProject) return;
|
||||||
|
|
||||||
await moduleService.addIssuesToModule(workspaceSlug as string, activeProject ?? "", moduleId as string, {
|
await moduleService.addModulesToIssue(workspaceSlug as string, activeProject ?? "", issueId as string, {
|
||||||
issues: [issueId],
|
modules: moduleIds,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -248,7 +248,7 @@ export const CreateUpdateDraftIssueModal: React.FC<IssuesModalProps> = observer(
|
|||||||
.createIssue(workspaceSlug.toString(), activeProject, payload)
|
.createIssue(workspaceSlug.toString(), activeProject, payload)
|
||||||
.then(async (res) => {
|
.then(async (res) => {
|
||||||
if (payload.cycle_id && payload.cycle_id !== "") await addIssueToCycle(res.id, payload.cycle_id);
|
if (payload.cycle_id && payload.cycle_id !== "") await addIssueToCycle(res.id, payload.cycle_id);
|
||||||
if (payload.module_id && payload.module_id !== "") await addIssueToModule(res.id, payload.module_id);
|
if (payload.module_ids && payload.module_ids.length > 0) await addIssueToModule(res.id, payload.module_ids);
|
||||||
|
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
|
@ -79,7 +79,7 @@ export const InboxIssueMainContent: React.FC<Props> = observer((props) => {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="pb-12">
|
<div className="pb-12">
|
||||||
<IssueActivity workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} disabled={!is_editable} />
|
<IssueActivity workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} />
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
@ -59,7 +59,7 @@ export const IssueModuleActivity: FC<TIssueModuleActivity> = observer((props) =>
|
|||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
className="inline-flex items-center gap-1 truncate font-medium text-custom-text-100 hover:underline"
|
className="inline-flex items-center gap-1 truncate font-medium text-custom-text-100 hover:underline"
|
||||||
>
|
>
|
||||||
<span className="truncate"> {activity.new_value}</span>
|
<span className="truncate"> {activity.old_value}</span>
|
||||||
</a>
|
</a>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
@ -17,7 +17,6 @@ const fileService = new FileService();
|
|||||||
type TIssueCommentCreate = {
|
type TIssueCommentCreate = {
|
||||||
workspaceSlug: string;
|
workspaceSlug: string;
|
||||||
activityOperations: TActivityOperations;
|
activityOperations: TActivityOperations;
|
||||||
disabled: boolean;
|
|
||||||
showAccessSpecifier?: boolean;
|
showAccessSpecifier?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -40,7 +39,7 @@ const commentAccess: commentAccessType[] = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
export const IssueCommentCreate: FC<TIssueCommentCreate> = (props) => {
|
export const IssueCommentCreate: FC<TIssueCommentCreate> = (props) => {
|
||||||
const { workspaceSlug, activityOperations, disabled, showAccessSpecifier = false } = props;
|
const { workspaceSlug, activityOperations, showAccessSpecifier = false } = props;
|
||||||
const workspaceStore = useWorkspace();
|
const workspaceStore = useWorkspace();
|
||||||
const workspaceId = workspaceStore.getWorkspaceBySlug(workspaceSlug as string)?.id as string;
|
const workspaceId = workspaceStore.getWorkspaceBySlug(workspaceSlug as string)?.id as string;
|
||||||
|
|
||||||
@ -94,7 +93,7 @@ export const IssueCommentCreate: FC<TIssueCommentCreate> = (props) => {
|
|||||||
}
|
}
|
||||||
submitButton={
|
submitButton={
|
||||||
<Button
|
<Button
|
||||||
disabled={isSubmitting || disabled}
|
disabled={isSubmitting}
|
||||||
variant="primary"
|
variant="primary"
|
||||||
type="submit"
|
type="submit"
|
||||||
className="!px-2.5 !py-1.5 !text-xs"
|
className="!px-2.5 !py-1.5 !text-xs"
|
||||||
|
@ -13,7 +13,6 @@ type TIssueActivity = {
|
|||||||
workspaceSlug: string;
|
workspaceSlug: string;
|
||||||
projectId: string;
|
projectId: string;
|
||||||
issueId: string;
|
issueId: string;
|
||||||
disabled: boolean;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
type TActivityTabs = "all" | "activity" | "comments";
|
type TActivityTabs = "all" | "activity" | "comments";
|
||||||
@ -43,7 +42,7 @@ export type TActivityOperations = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const IssueActivity: FC<TIssueActivity> = observer((props) => {
|
export const IssueActivity: FC<TIssueActivity> = observer((props) => {
|
||||||
const { workspaceSlug, projectId, issueId, disabled } = props;
|
const { workspaceSlug, projectId, issueId } = props;
|
||||||
// hooks
|
// hooks
|
||||||
const { createComment, updateComment, removeComment } = useIssueDetail();
|
const { createComment, updateComment, removeComment } = useIssueDetail();
|
||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
@ -147,14 +146,11 @@ export const IssueActivity: FC<TIssueActivity> = observer((props) => {
|
|||||||
activityOperations={activityOperations}
|
activityOperations={activityOperations}
|
||||||
showAccessSpecifier={project.is_deployed}
|
showAccessSpecifier={project.is_deployed}
|
||||||
/>
|
/>
|
||||||
{!disabled && (
|
|
||||||
<IssueCommentCreate
|
<IssueCommentCreate
|
||||||
workspaceSlug={workspaceSlug}
|
workspaceSlug={workspaceSlug}
|
||||||
activityOperations={activityOperations}
|
activityOperations={activityOperations}
|
||||||
disabled={disabled}
|
|
||||||
showAccessSpecifier={project.is_deployed}
|
showAccessSpecifier={project.is_deployed}
|
||||||
/>
|
/>
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
) : activityTab === "activity" ? (
|
) : activityTab === "activity" ? (
|
||||||
<IssueActivityRoot issueId={issueId} />
|
<IssueActivityRoot issueId={issueId} />
|
||||||
@ -166,14 +162,11 @@ export const IssueActivity: FC<TIssueActivity> = observer((props) => {
|
|||||||
activityOperations={activityOperations}
|
activityOperations={activityOperations}
|
||||||
showAccessSpecifier={project.is_deployed}
|
showAccessSpecifier={project.is_deployed}
|
||||||
/>
|
/>
|
||||||
{!disabled && (
|
|
||||||
<IssueCommentCreate
|
<IssueCommentCreate
|
||||||
workspaceSlug={workspaceSlug}
|
workspaceSlug={workspaceSlug}
|
||||||
activityOperations={activityOperations}
|
activityOperations={activityOperations}
|
||||||
disabled={disabled}
|
|
||||||
showAccessSpecifier={project.is_deployed}
|
showAccessSpecifier={project.is_deployed}
|
||||||
/>
|
/>
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
@ -99,7 +99,7 @@ export const IssueMainContent: React.FC<Props> = observer((props) => {
|
|||||||
disabled={!is_editable}
|
disabled={!is_editable}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<IssueActivity workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} disabled={!is_editable} />
|
<IssueActivity workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
import xor from "lodash/xor";
|
||||||
// hooks
|
// hooks
|
||||||
import { useIssueDetail } from "hooks/store";
|
import { useIssueDetail } from "hooks/store";
|
||||||
// components
|
// components
|
||||||
import { ModuleDropdown } from "components/dropdowns";
|
import { ModuleSelectDropdown } from "components/dropdowns";
|
||||||
// ui
|
// ui
|
||||||
import { Spinner } from "@plane/ui";
|
import { Spinner } from "@plane/ui";
|
||||||
// helpers
|
// helpers
|
||||||
@ -32,58 +33,56 @@ export const IssueModuleSelect: React.FC<TIssueModuleSelect> = observer((props)
|
|||||||
const issue = getIssueById(issueId);
|
const issue = getIssueById(issueId);
|
||||||
const disableSelect = disabled || isUpdating;
|
const disableSelect = disabled || isUpdating;
|
||||||
|
|
||||||
const handleIssueModuleChange = async (moduleId: string | null) => {
|
const handleIssueModuleChange = async (moduleIds: undefined | string | (string | undefined)[]) => {
|
||||||
if (!issue || issue.module_id === moduleId) return;
|
if (!issue) return;
|
||||||
|
|
||||||
setIsUpdating(true);
|
setIsUpdating(true);
|
||||||
if (moduleId) await issueOperations.addIssueToModule?.(workspaceSlug, projectId, moduleId, [issueId]);
|
if (moduleIds === undefined && issue?.module_ids && issue?.module_ids.length > 0)
|
||||||
else await issueOperations.removeIssueFromModule?.(workspaceSlug, projectId, issue.module_id ?? "", issueId);
|
await issueOperations.removeModulesFromIssue?.(workspaceSlug, projectId, issueId, issue?.module_ids);
|
||||||
|
|
||||||
|
if (typeof moduleIds === "string" && moduleIds)
|
||||||
|
await issueOperations.removeModulesFromIssue?.(workspaceSlug, projectId, issueId, [moduleIds]);
|
||||||
|
|
||||||
|
if (Array.isArray(moduleIds)) {
|
||||||
|
if (moduleIds.includes(undefined)) {
|
||||||
|
await issueOperations.removeModulesFromIssue?.(
|
||||||
|
workspaceSlug,
|
||||||
|
projectId,
|
||||||
|
issueId,
|
||||||
|
moduleIds.filter((x) => x != undefined) as string[]
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
const _moduleIds = xor(issue?.module_ids, moduleIds)[0];
|
||||||
|
if (_moduleIds) {
|
||||||
|
if (issue?.module_ids?.includes(_moduleIds))
|
||||||
|
await issueOperations.removeModulesFromIssue?.(workspaceSlug, projectId, issueId, [_moduleIds]);
|
||||||
|
else await issueOperations.addModulesToIssue?.(workspaceSlug, projectId, issueId, [_moduleIds]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
setIsUpdating(false);
|
setIsUpdating(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cn("flex items-center gap-1 h-full", className)}>
|
<div className={cn(`flex items-center gap-1 h-full`, className)}>
|
||||||
<ModuleDropdown
|
<ModuleSelectDropdown
|
||||||
value={issue?.module_id ?? null}
|
workspaceSlug={workspaceSlug}
|
||||||
onChange={handleIssueModuleChange}
|
|
||||||
buttonVariant="transparent-with-text"
|
|
||||||
projectId={projectId}
|
projectId={projectId}
|
||||||
disabled={disableSelect}
|
value={issue?.module_ids?.length ? issue?.module_ids : undefined}
|
||||||
className="w-full group"
|
onChange={handleIssueModuleChange}
|
||||||
buttonContainerClassName="w-full text-left"
|
multiple={true}
|
||||||
buttonClassName={`text-sm ${issue?.module_id ? "" : "text-custom-text-400"}`}
|
|
||||||
placeholder="No module"
|
placeholder="No module"
|
||||||
hideIcon
|
|
||||||
dropdownArrow
|
|
||||||
dropdownArrowClassName="h-3.5 w-3.5 hidden group-hover:inline"
|
|
||||||
/>
|
|
||||||
{/* <CustomSearchSelect
|
|
||||||
value={issue?.module_id}
|
|
||||||
onChange={(value: any) => handleIssueModuleChange(value)}
|
|
||||||
options={options}
|
|
||||||
customButton={
|
|
||||||
<div>
|
|
||||||
<Tooltip position="left" tooltipContent={`${issueModule?.name ?? "No module"}`}>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
className={`flex w-full items-center rounded bg-custom-background-80 px-2.5 py-0.5 text-xs ${
|
|
||||||
disableSelect ? "cursor-not-allowed" : ""
|
|
||||||
} max-w-[10rem]`}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className={`flex items-center gap-1.5 truncate ${
|
|
||||||
issueModule ? "text-custom-text-100" : "text-custom-text-200"
|
|
||||||
}`}
|
|
||||||
>
|
|
||||||
<span className="flex-shrink-0">{issueModule && <DiceIcon className="h-3.5 w-3.5" />}</span>
|
|
||||||
<span className="truncate">{issueModule?.name ?? "No module"}</span>
|
|
||||||
</span>
|
|
||||||
</button>
|
|
||||||
</Tooltip>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
noChevron
|
|
||||||
disabled={disableSelect}
|
disabled={disableSelect}
|
||||||
/> */}
|
className={`w-full h-full group`}
|
||||||
|
buttonContainerClassName="w-full"
|
||||||
|
buttonClassName={`min-h-8 ${issue?.module_ids?.length ? `` : `text-custom-text-400`}`}
|
||||||
|
buttonVariant="transparent-with-text"
|
||||||
|
hideIcon={false}
|
||||||
|
dropdownArrow={true}
|
||||||
|
dropdownArrowClassName="h-3.5 w-3.5 hidden group-hover:inline"
|
||||||
|
showTooltip={true}
|
||||||
|
/>
|
||||||
|
|
||||||
{isUpdating && <Spinner className="h-4 w-4" />}
|
{isUpdating && <Spinner className="h-4 w-4" />}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -29,13 +29,19 @@ export type TIssueOperations = {
|
|||||||
remove: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
|
remove: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
|
||||||
addIssueToCycle?: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise<void>;
|
addIssueToCycle?: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise<void>;
|
||||||
removeIssueFromCycle?: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise<void>;
|
removeIssueFromCycle?: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise<void>;
|
||||||
addIssueToModule?: (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => Promise<void>;
|
addModulesToIssue?: (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => Promise<void>;
|
||||||
removeIssueFromModule?: (
|
removeIssueFromModule?: (
|
||||||
workspaceSlug: string,
|
workspaceSlug: string,
|
||||||
projectId: string,
|
projectId: string,
|
||||||
moduleId: string,
|
moduleId: string,
|
||||||
issueId: string
|
issueId: string
|
||||||
) => Promise<void>;
|
) => Promise<void>;
|
||||||
|
removeModulesFromIssue?: (
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
moduleIds: string[]
|
||||||
|
) => Promise<void>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type TIssueDetailRoot = {
|
export type TIssueDetailRoot = {
|
||||||
@ -57,8 +63,9 @@ export const IssueDetailRoot: FC<TIssueDetailRoot> = (props) => {
|
|||||||
removeIssue,
|
removeIssue,
|
||||||
addIssueToCycle,
|
addIssueToCycle,
|
||||||
removeIssueFromCycle,
|
removeIssueFromCycle,
|
||||||
addIssueToModule,
|
addModulesToIssue,
|
||||||
removeIssueFromModule,
|
removeIssueFromModule,
|
||||||
|
removeModulesFromIssue,
|
||||||
} = useIssueDetail();
|
} = useIssueDetail();
|
||||||
const {
|
const {
|
||||||
issues: { removeIssue: removeArchivedIssue },
|
issues: { removeIssue: removeArchivedIssue },
|
||||||
@ -150,9 +157,9 @@ export const IssueDetailRoot: FC<TIssueDetailRoot> = (props) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
addIssueToModule: async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => {
|
addModulesToIssue: async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => {
|
||||||
try {
|
try {
|
||||||
await addIssueToModule(workspaceSlug, projectId, moduleId, issueIds);
|
await addModulesToIssue(workspaceSlug, projectId, issueId, moduleIds);
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
title: "Module added to issue successfully",
|
title: "Module added to issue successfully",
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -182,6 +189,27 @@ export const IssueDetailRoot: FC<TIssueDetailRoot> = (props) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
removeModulesFromIssue: async (
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
moduleIds: string[]
|
||||||
|
) => {
|
||||||
|
try {
|
||||||
|
await removeModulesFromIssue(workspaceSlug, projectId, issueId, moduleIds);
|
||||||
|
setToastAlert({
|
||||||
|
title: "Module removed from issue successfully",
|
||||||
|
type: "success",
|
||||||
|
message: "Module removed from issue successfully",
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
setToastAlert({
|
||||||
|
title: "Module remove from issue failed",
|
||||||
|
type: "error",
|
||||||
|
message: "Module remove from issue failed",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
}),
|
}),
|
||||||
[
|
[
|
||||||
is_archived,
|
is_archived,
|
||||||
@ -191,8 +219,9 @@ export const IssueDetailRoot: FC<TIssueDetailRoot> = (props) => {
|
|||||||
removeArchivedIssue,
|
removeArchivedIssue,
|
||||||
addIssueToCycle,
|
addIssueToCycle,
|
||||||
removeIssueFromCycle,
|
removeIssueFromCycle,
|
||||||
addIssueToModule,
|
addModulesToIssue,
|
||||||
removeIssueFromModule,
|
removeIssueFromModule,
|
||||||
|
removeModulesFromIssue,
|
||||||
setToastAlert,
|
setToastAlert,
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
|
@ -286,7 +286,7 @@ export const IssueDetailsSidebar: React.FC<Props> = observer((props) => {
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{projectDetails?.module_view && (
|
{projectDetails?.module_view && (
|
||||||
<div className="flex items-center gap-2 h-8">
|
<div className="flex items-center gap-2 min-h-8 h-full">
|
||||||
<div className="flex items-center gap-1 w-2/5 flex-shrink-0 text-sm text-custom-text-300">
|
<div className="flex items-center gap-1 w-2/5 flex-shrink-0 text-sm text-custom-text-300">
|
||||||
<DiceIcon className="h-4 w-4 flex-shrink-0" />
|
<DiceIcon className="h-4 w-4 flex-shrink-0" />
|
||||||
<span>Module</span>
|
<span>Module</span>
|
||||||
|
@ -6,7 +6,8 @@ import { MoreHorizontal } from "lucide-react";
|
|||||||
import { Tooltip, ControlLink } from "@plane/ui";
|
import { Tooltip, ControlLink } from "@plane/ui";
|
||||||
// hooks
|
// hooks
|
||||||
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
||||||
// ui
|
// helpers
|
||||||
|
import { cn } from "helpers/common.helper";
|
||||||
// types
|
// types
|
||||||
import { TIssue, TIssueMap } from "@plane/types";
|
import { TIssue, TIssueMap } from "@plane/types";
|
||||||
import { useApplication, useIssueDetail, useProject, useProjectState } from "hooks/store";
|
import { useApplication, useIssueDetail, useProject, useProjectState } from "hooks/store";
|
||||||
@ -26,7 +27,7 @@ export const CalendarIssueBlocks: React.FC<Props> = observer((props) => {
|
|||||||
} = useApplication();
|
} = useApplication();
|
||||||
const { getProjectById } = useProject();
|
const { getProjectById } = useProject();
|
||||||
const { getProjectStates } = useProjectState();
|
const { getProjectStates } = useProjectState();
|
||||||
const { setPeekIssue } = useIssueDetail();
|
const { peekIssue, setPeekIssue } = useIssueDetail();
|
||||||
// states
|
// states
|
||||||
const [isMenuActive, setIsMenuActive] = useState(false);
|
const [isMenuActive, setIsMenuActive] = useState(false);
|
||||||
|
|
||||||
@ -84,11 +85,18 @@ export const CalendarIssueBlocks: React.FC<Props> = observer((props) => {
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
<div
|
<div
|
||||||
className={`group/calendar-block flex h-8 w-full items-center justify-between gap-1.5 rounded border-[0.5px] border-custom-border-100 px-1 py-1.5 shadow-custom-shadow-2xs ${
|
className={cn(
|
||||||
snapshot.isDragging
|
"group/calendar-block flex h-8 w-full items-center justify-between gap-1.5 rounded border-[0.5px] border-custom-border-200 hover:border-custom-border-400 px-1 py-1.5 ",
|
||||||
? "bg-custom-background-90 shadow-custom-shadow-rg"
|
{
|
||||||
: "bg-custom-background-100 hover:bg-custom-background-90"
|
"bg-custom-background-90 shadow-custom-shadow-rg border-custom-primary-100":
|
||||||
}`}
|
snapshot.isDragging,
|
||||||
|
},
|
||||||
|
{ "bg-custom-background-100 hover:bg-custom-background-90": !snapshot.isDragging },
|
||||||
|
{
|
||||||
|
"border border-custom-primary-70 hover:border-custom-primary-70":
|
||||||
|
peekIssue?.issueId === issue.id,
|
||||||
|
}
|
||||||
|
)}
|
||||||
>
|
>
|
||||||
<div className="flex h-full items-center gap-1.5">
|
<div className="flex h-full items-center gap-1.5">
|
||||||
<span
|
<span
|
||||||
|
@ -46,11 +46,7 @@ export const ModuleEmptyState: React.FC<Props> = observer((props) => {
|
|||||||
|
|
||||||
const issueIds = data.map((i) => i.id);
|
const issueIds = data.map((i) => i.id);
|
||||||
await issues
|
await issues
|
||||||
.addIssueToModule(workspaceSlug.toString(), projectId?.toString(), moduleId.toString(), issueIds)
|
.addIssuesToModule(workspaceSlug.toString(), projectId?.toString(), moduleId.toString(), issueIds)
|
||||||
.then((res) => {
|
|
||||||
updateIssue(workspaceSlug, projectId, res.id, res);
|
|
||||||
fetchIssue(workspaceSlug, projectId, res.id);
|
|
||||||
})
|
|
||||||
.catch(() =>
|
.catch(() =>
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "error",
|
type: "error",
|
||||||
@ -69,7 +65,7 @@ export const ModuleEmptyState: React.FC<Props> = observer((props) => {
|
|||||||
projectId={projectId}
|
projectId={projectId}
|
||||||
isOpen={moduleIssuesListModal}
|
isOpen={moduleIssuesListModal}
|
||||||
handleClose={() => setModuleIssuesListModal(false)}
|
handleClose={() => setModuleIssuesListModal(false)}
|
||||||
searchParams={{ module: true }}
|
searchParams={{ module: moduleId != undefined ? [moduleId.toString()] : [] }}
|
||||||
handleOnSubmit={handleAddIssuesToModule}
|
handleOnSubmit={handleAddIssuesToModule}
|
||||||
/>
|
/>
|
||||||
<div className="grid h-full w-full place-items-center">
|
<div className="grid h-full w-full place-items-center">
|
||||||
|
@ -44,7 +44,7 @@ export interface IBaseKanBanLayout {
|
|||||||
showLoader?: boolean;
|
showLoader?: boolean;
|
||||||
viewId?: string;
|
viewId?: string;
|
||||||
storeType?: TCreateModalStoreTypes;
|
storeType?: TCreateModalStoreTypes;
|
||||||
addIssuesToView?: (issueIds: string[]) => Promise<TIssue>;
|
addIssuesToView?: (issueIds: string[]) => Promise<any>;
|
||||||
canEditPropertiesBasedOnProject?: (projectId: string) => boolean;
|
canEditPropertiesBasedOnProject?: (projectId: string) => boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import { memo } from "react";
|
import { memo } from "react";
|
||||||
import { Draggable, DraggableProvided, DraggableStateSnapshot } from "@hello-pangea/dnd";
|
import { Draggable, DraggableProvided, DraggableStateSnapshot } from "@hello-pangea/dnd";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
// hooks
|
||||||
|
import { useApplication, useIssueDetail, useProject } from "hooks/store";
|
||||||
// components
|
// components
|
||||||
import { WithDisplayPropertiesHOC } from "../properties/with-display-properties-HOC";
|
import { WithDisplayPropertiesHOC } from "../properties/with-display-properties-HOC";
|
||||||
import { IssueProperties } from "../properties/all-properties";
|
import { IssueProperties } from "../properties/all-properties";
|
||||||
@ -9,9 +11,11 @@ import { Tooltip, ControlLink } from "@plane/ui";
|
|||||||
// types
|
// types
|
||||||
import { TIssue, IIssueDisplayProperties, IIssueMap } from "@plane/types";
|
import { TIssue, IIssueDisplayProperties, IIssueMap } from "@plane/types";
|
||||||
import { EIssueActions } from "../types";
|
import { EIssueActions } from "../types";
|
||||||
import { useApplication, useIssueDetail, useProject } from "hooks/store";
|
// helper
|
||||||
|
import { cn } from "helpers/common.helper";
|
||||||
|
|
||||||
interface IssueBlockProps {
|
interface IssueBlockProps {
|
||||||
|
peekIssueId?: string;
|
||||||
issueId: string;
|
issueId: string;
|
||||||
issuesMap: IIssueMap;
|
issuesMap: IIssueMap;
|
||||||
displayProperties: IIssueDisplayProperties | undefined;
|
displayProperties: IIssueDisplayProperties | undefined;
|
||||||
@ -86,6 +90,7 @@ const KanbanIssueDetailsBlock: React.FC<IssueDetailsBlockProps> = observer((prop
|
|||||||
|
|
||||||
export const KanbanIssueBlock: React.FC<IssueBlockProps> = memo((props) => {
|
export const KanbanIssueBlock: React.FC<IssueBlockProps> = memo((props) => {
|
||||||
const {
|
const {
|
||||||
|
peekIssueId,
|
||||||
issueId,
|
issueId,
|
||||||
issuesMap,
|
issuesMap,
|
||||||
displayProperties,
|
displayProperties,
|
||||||
@ -121,9 +126,12 @@ export const KanbanIssueBlock: React.FC<IssueBlockProps> = memo((props) => {
|
|||||||
<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" />
|
||||||
)}
|
)}
|
||||||
<div
|
<div
|
||||||
className={`space-y-2 rounded border-[0.5px] border-custom-border-200 bg-custom-background-100 px-3 py-2 text-sm shadow-custom-shadow-2xs transition-all ${
|
className={cn(
|
||||||
isDragDisabled ? "" : "hover:cursor-grab"
|
"space-y-2 rounded border-[0.5px] border-custom-border-200 bg-custom-background-100 px-3 py-2 text-sm transition-all hover:border-custom-border-400",
|
||||||
} ${snapshot.isDragging ? `border-custom-primary-100` : `border-transparent`}`}
|
{ "hover:cursor-grab": !isDragDisabled },
|
||||||
|
{ "border-custom-primary-100": snapshot.isDragging },
|
||||||
|
{ "border border-custom-primary-70 hover:border-custom-primary-70": peekIssueId === issue.id }
|
||||||
|
)}
|
||||||
>
|
>
|
||||||
<KanbanIssueDetailsBlock
|
<KanbanIssueDetailsBlock
|
||||||
issue={issue}
|
issue={issue}
|
||||||
|
@ -9,6 +9,7 @@ interface IssueBlocksListProps {
|
|||||||
sub_group_id: string;
|
sub_group_id: string;
|
||||||
columnId: string;
|
columnId: string;
|
||||||
issuesMap: IIssueMap;
|
issuesMap: IIssueMap;
|
||||||
|
peekIssueId?: string;
|
||||||
issueIds: string[];
|
issueIds: string[];
|
||||||
displayProperties: IIssueDisplayProperties | undefined;
|
displayProperties: IIssueDisplayProperties | undefined;
|
||||||
isDragDisabled: boolean;
|
isDragDisabled: boolean;
|
||||||
@ -22,6 +23,7 @@ const KanbanIssueBlocksListMemo: React.FC<IssueBlocksListProps> = (props) => {
|
|||||||
sub_group_id,
|
sub_group_id,
|
||||||
columnId,
|
columnId,
|
||||||
issuesMap,
|
issuesMap,
|
||||||
|
peekIssueId,
|
||||||
issueIds,
|
issueIds,
|
||||||
displayProperties,
|
displayProperties,
|
||||||
isDragDisabled,
|
isDragDisabled,
|
||||||
@ -44,6 +46,7 @@ const KanbanIssueBlocksListMemo: React.FC<IssueBlocksListProps> = (props) => {
|
|||||||
return (
|
return (
|
||||||
<KanbanIssueBlock
|
<KanbanIssueBlock
|
||||||
key={draggableId}
|
key={draggableId}
|
||||||
|
peekIssueId={peekIssueId}
|
||||||
issueId={issueId}
|
issueId={issueId}
|
||||||
issuesMap={issuesMap}
|
issuesMap={issuesMap}
|
||||||
displayProperties={displayProperties}
|
displayProperties={displayProperties}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
// hooks
|
// hooks
|
||||||
import { useKanbanView, useLabel, useMember, useProject, useProjectState } from "hooks/store";
|
import { useIssueDetail, useKanbanView, useLabel, useMember, useProject, useProjectState } from "hooks/store";
|
||||||
// components
|
// components
|
||||||
import { HeaderGroupByCard } from "./headers/group-by-card";
|
import { HeaderGroupByCard } from "./headers/group-by-card";
|
||||||
import { KanbanGroup } from "./kanban-group";
|
import { KanbanGroup } from "./kanban-group";
|
||||||
@ -73,6 +73,7 @@ const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
|
|||||||
const project = useProject();
|
const project = useProject();
|
||||||
const label = useLabel();
|
const label = useLabel();
|
||||||
const projectState = useProjectState();
|
const projectState = useProjectState();
|
||||||
|
const { peekIssue } = useIssueDetail();
|
||||||
|
|
||||||
const list = getGroupByColumns(group_by as GroupByColumnTypes, project, label, projectState, member);
|
const list = getGroupByColumns(group_by as GroupByColumnTypes, project, label, projectState, member);
|
||||||
|
|
||||||
@ -120,6 +121,7 @@ const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
|
|||||||
groupId={_list.id}
|
groupId={_list.id}
|
||||||
issuesMap={issuesMap}
|
issuesMap={issuesMap}
|
||||||
issueIds={issueIds}
|
issueIds={issueIds}
|
||||||
|
peekIssueId={peekIssue?.issueId ?? ""}
|
||||||
displayProperties={displayProperties}
|
displayProperties={displayProperties}
|
||||||
sub_group_by={sub_group_by}
|
sub_group_by={sub_group_by}
|
||||||
group_by={group_by}
|
group_by={group_by}
|
||||||
|
@ -56,7 +56,7 @@ export const HeaderGroupByCard: FC<IHeaderGroupByCard> = observer((props) => {
|
|||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
|
|
||||||
const renderExistingIssueModal = moduleId || cycleId;
|
const renderExistingIssueModal = moduleId || cycleId;
|
||||||
const ExistingIssuesListModalPayload = moduleId ? { module: true } : { cycle: true };
|
const ExistingIssuesListModalPayload = moduleId ? { module: [moduleId.toString()] } : { cycle: true };
|
||||||
|
|
||||||
const handleAddIssuesToView = async (data: ISearchIssueResponse[]) => {
|
const handleAddIssuesToView = async (data: ISearchIssueResponse[]) => {
|
||||||
if (!workspaceSlug || !projectId) return;
|
if (!workspaceSlug || !projectId) return;
|
||||||
|
@ -17,6 +17,7 @@ import { EIssueActions } from "../types";
|
|||||||
interface IKanbanGroup {
|
interface IKanbanGroup {
|
||||||
groupId: string;
|
groupId: string;
|
||||||
issuesMap: IIssueMap;
|
issuesMap: IIssueMap;
|
||||||
|
peekIssueId?: string;
|
||||||
issueIds: TGroupedIssues | TSubGroupedIssues | TUnGroupedIssues;
|
issueIds: TGroupedIssues | TSubGroupedIssues | TUnGroupedIssues;
|
||||||
displayProperties: IIssueDisplayProperties | undefined;
|
displayProperties: IIssueDisplayProperties | undefined;
|
||||||
sub_group_by: string | null;
|
sub_group_by: string | null;
|
||||||
@ -47,6 +48,7 @@ export const KanbanGroup = (props: IKanbanGroup) => {
|
|||||||
issuesMap,
|
issuesMap,
|
||||||
displayProperties,
|
displayProperties,
|
||||||
issueIds,
|
issueIds,
|
||||||
|
peekIssueId,
|
||||||
isDragDisabled,
|
isDragDisabled,
|
||||||
handleIssues,
|
handleIssues,
|
||||||
quickActions,
|
quickActions,
|
||||||
@ -118,6 +120,7 @@ export const KanbanGroup = (props: IKanbanGroup) => {
|
|||||||
sub_group_id={sub_group_id}
|
sub_group_id={sub_group_id}
|
||||||
columnId={groupId}
|
columnId={groupId}
|
||||||
issuesMap={issuesMap}
|
issuesMap={issuesMap}
|
||||||
|
peekIssueId={peekIssueId}
|
||||||
issueIds={(issueIds as TGroupedIssues)?.[groupId] || []}
|
issueIds={(issueIds as TGroupedIssues)?.[groupId] || []}
|
||||||
displayProperties={displayProperties}
|
displayProperties={displayProperties}
|
||||||
isDragDisabled={isDragDisabled}
|
isDragDisabled={isDragDisabled}
|
||||||
|
@ -53,7 +53,7 @@ export const ModuleKanBanLayout: React.FC = observer(() => {
|
|||||||
storeType={EIssuesStoreType.MODULE}
|
storeType={EIssuesStoreType.MODULE}
|
||||||
addIssuesToView={(issueIds: string[]) => {
|
addIssuesToView={(issueIds: string[]) => {
|
||||||
if (!workspaceSlug || !projectId || !moduleId) throw new Error();
|
if (!workspaceSlug || !projectId || !moduleId) throw new Error();
|
||||||
return issues.addIssueToModule(workspaceSlug.toString(), projectId.toString(), moduleId.toString(), issueIds);
|
return issues.addIssuesToModule(workspaceSlug.toString(), projectId.toString(), moduleId.toString(), issueIds);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -49,7 +49,7 @@ interface IBaseListRoot {
|
|||||||
};
|
};
|
||||||
viewId?: string;
|
viewId?: string;
|
||||||
storeType: TCreateModalStoreTypes;
|
storeType: TCreateModalStoreTypes;
|
||||||
addIssuesToView?: (issueIds: string[]) => Promise<TIssue>;
|
addIssuesToView?: (issueIds: string[]) => Promise<any>;
|
||||||
canEditPropertiesBasedOnProject?: (projectId: string) => boolean;
|
canEditPropertiesBasedOnProject?: (projectId: string) => boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,6 +5,8 @@ import { IssueProperties } from "../properties/all-properties";
|
|||||||
import { useApplication, useIssueDetail, useProject } from "hooks/store";
|
import { useApplication, useIssueDetail, useProject } from "hooks/store";
|
||||||
// ui
|
// ui
|
||||||
import { Spinner, Tooltip, ControlLink } from "@plane/ui";
|
import { Spinner, Tooltip, ControlLink } from "@plane/ui";
|
||||||
|
// helper
|
||||||
|
import { cn } from "helpers/common.helper";
|
||||||
// types
|
// types
|
||||||
import { TIssue, IIssueDisplayProperties, TIssueMap } from "@plane/types";
|
import { TIssue, IIssueDisplayProperties, TIssueMap } from "@plane/types";
|
||||||
import { EIssueActions } from "../types";
|
import { EIssueActions } from "../types";
|
||||||
@ -25,7 +27,7 @@ export const IssueBlock: React.FC<IssueBlockProps> = observer((props: IssueBlock
|
|||||||
router: { workspaceSlug, projectId },
|
router: { workspaceSlug, projectId },
|
||||||
} = useApplication();
|
} = useApplication();
|
||||||
const { getProjectById } = useProject();
|
const { getProjectById } = useProject();
|
||||||
const { setPeekIssue } = useIssueDetail();
|
const { peekIssue, setPeekIssue } = useIssueDetail();
|
||||||
|
|
||||||
const updateIssue = (issueToUpdate: TIssue) => {
|
const updateIssue = (issueToUpdate: TIssue) => {
|
||||||
handleIssues(issueToUpdate, EIssueActions.UPDATE);
|
handleIssues(issueToUpdate, EIssueActions.UPDATE);
|
||||||
@ -47,7 +49,15 @@ export const IssueBlock: React.FC<IssueBlockProps> = observer((props: IssueBlock
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="relative flex items-center gap-3 bg-custom-background-100 p-3 text-sm">
|
<div
|
||||||
|
className={cn(
|
||||||
|
"relative flex items-center gap-3 bg-custom-background-100 p-3 text-sm border border-transparent border-b-custom-border-200 last:border-b-transparent",
|
||||||
|
{
|
||||||
|
"border border-custom-primary-70 hover:border-custom-primary-70":
|
||||||
|
peekIssue && peekIssue.issueId === issue.id,
|
||||||
|
}
|
||||||
|
)}
|
||||||
|
>
|
||||||
{displayProperties && displayProperties?.key && (
|
{displayProperties && displayProperties?.key && (
|
||||||
<div className="flex-shrink-0 text-xs font-medium text-custom-text-300">
|
<div className="flex-shrink-0 text-xs font-medium text-custom-text-300">
|
||||||
{projectDetails?.identifier}-{issue.sequence_id}
|
{projectDetails?.identifier}-{issue.sequence_id}
|
||||||
|
@ -18,7 +18,7 @@ export const IssueBlocksList: FC<Props> = (props) => {
|
|||||||
const { issueIds, issuesMap, handleIssues, quickActions, displayProperties, canEditProperties } = props;
|
const { issueIds, issuesMap, handleIssues, quickActions, displayProperties, canEditProperties } = props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="relative h-full w-full divide-y-[0.5px] divide-custom-border-200">
|
<div className="relative h-full w-full">
|
||||||
{issueIds && issueIds.length > 0 ? (
|
{issueIds && issueIds.length > 0 ? (
|
||||||
issueIds.map((issueId: string) => {
|
issueIds.map((issueId: string) => {
|
||||||
if (!issueId) return null;
|
if (!issueId) return null;
|
||||||
|
@ -37,7 +37,7 @@ export const HeaderGroupByCard = observer(
|
|||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
|
|
||||||
const renderExistingIssueModal = moduleId || cycleId;
|
const renderExistingIssueModal = moduleId || cycleId;
|
||||||
const ExistingIssuesListModalPayload = moduleId ? { module: true } : { cycle: true };
|
const ExistingIssuesListModalPayload = moduleId ? { module: [moduleId.toString()] } : { cycle: true };
|
||||||
|
|
||||||
const handleAddIssuesToView = async (data: ISearchIssueResponse[]) => {
|
const handleAddIssuesToView = async (data: ISearchIssueResponse[]) => {
|
||||||
if (!workspaceSlug || !projectId) return;
|
if (!workspaceSlug || !projectId) return;
|
||||||
|
@ -51,7 +51,7 @@ export const ModuleListLayout: React.FC = observer(() => {
|
|||||||
storeType={EIssuesStoreType.MODULE}
|
storeType={EIssuesStoreType.MODULE}
|
||||||
addIssuesToView={(issueIds: string[]) => {
|
addIssuesToView={(issueIds: string[]) => {
|
||||||
if (!workspaceSlug || !projectId || !moduleId) throw new Error();
|
if (!workspaceSlug || !projectId || !moduleId) throw new Error();
|
||||||
return issues.addIssueToModule(workspaceSlug.toString(), projectId.toString(), moduleId.toString(), issueIds);
|
return issues.addIssuesToModule(workspaceSlug.toString(), projectId.toString(), moduleId.toString(), issueIds);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -155,7 +155,6 @@ export const IssueProperties: React.FC<IIssueProperties> = observer((props) => {
|
|||||||
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" : ""}
|
||||||
tooltip
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</WithDisplayPropertiesHOC>
|
</WithDisplayPropertiesHOC>
|
||||||
|
@ -12,6 +12,8 @@ import { ControlLink, Tooltip } from "@plane/ui";
|
|||||||
// hooks
|
// hooks
|
||||||
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
||||||
import { useIssueDetail, useProject } from "hooks/store";
|
import { useIssueDetail, useProject } from "hooks/store";
|
||||||
|
// helper
|
||||||
|
import { cn } from "helpers/common.helper";
|
||||||
// types
|
// types
|
||||||
import { IIssueDisplayProperties, TIssue } from "@plane/types";
|
import { IIssueDisplayProperties, TIssue } from "@plane/types";
|
||||||
import { EIssueActions } from "../types";
|
import { EIssueActions } from "../types";
|
||||||
@ -48,7 +50,7 @@ export const SpreadsheetIssueRow = observer((props: Props) => {
|
|||||||
const { workspaceSlug } = router.query;
|
const { workspaceSlug } = router.query;
|
||||||
//hooks
|
//hooks
|
||||||
const { getProjectById } = useProject();
|
const { getProjectById } = useProject();
|
||||||
const { setPeekIssue } = useIssueDetail();
|
const { peekIssue, setPeekIssue } = useIssueDetail();
|
||||||
// states
|
// states
|
||||||
const [isMenuActive, setIsMenuActive] = useState(false);
|
const [isMenuActive, setIsMenuActive] = useState(false);
|
||||||
const [isExpanded, setExpanded] = useState<boolean>(false);
|
const [isExpanded, setExpanded] = useState<boolean>(false);
|
||||||
@ -95,9 +97,20 @@ export const SpreadsheetIssueRow = observer((props: Props) => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<tr>
|
<tr
|
||||||
|
className={cn({
|
||||||
|
"border border-custom-primary-70 hover:border-custom-primary-70": peekIssue?.issueId === issueDetail.id,
|
||||||
|
})}
|
||||||
|
>
|
||||||
{/* first column/ issue name and key column */}
|
{/* first column/ issue name and key column */}
|
||||||
<td className="sticky group left-0 h-11 w-[28rem] flex items-center bg-custom-background-100 text-sm after:absolute after:w-full after:bottom-[-1px] after:border after:border-l-0 after:border-custom-border-100 before:absolute before:h-full before:right-0 before:border before:border-l-0 before:border-custom-border-100">
|
<td
|
||||||
|
className={cn(
|
||||||
|
"sticky group left-0 h-11 w-[28rem] flex items-center bg-custom-background-100 text-sm after:absolute border-r-[0.5px] border-custom-border-200",
|
||||||
|
{
|
||||||
|
"border-b-[0.5px]": peekIssue?.issueId !== issueDetail.id,
|
||||||
|
}
|
||||||
|
)}
|
||||||
|
>
|
||||||
<WithDisplayPropertiesHOC displayProperties={displayProperties} displayPropertyKey="key">
|
<WithDisplayPropertiesHOC displayProperties={displayProperties} displayPropertyKey="key">
|
||||||
<div
|
<div
|
||||||
className="flex min-w-min items-center gap-1.5 px-4 py-2.5 pr-0"
|
className="flex min-w-min items-center gap-1.5 px-4 py-2.5 pr-0"
|
||||||
|
@ -36,7 +36,7 @@ export const SpreadsheetTable = observer((props: Props) => {
|
|||||||
} = props;
|
} = props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<table className="divide-x-[0.5px] divide-custom-border-200 overflow-y-auto">
|
<table className="overflow-y-auto">
|
||||||
<SpreadsheetHeader
|
<SpreadsheetHeader
|
||||||
displayProperties={displayProperties}
|
displayProperties={displayProperties}
|
||||||
displayFilters={displayFilters}
|
displayFilters={displayFilters}
|
||||||
|
@ -20,7 +20,7 @@ import {
|
|||||||
CycleDropdown,
|
CycleDropdown,
|
||||||
DateDropdown,
|
DateDropdown,
|
||||||
EstimateDropdown,
|
EstimateDropdown,
|
||||||
ModuleDropdown,
|
ModuleSelectDropdown,
|
||||||
PriorityDropdown,
|
PriorityDropdown,
|
||||||
ProjectDropdown,
|
ProjectDropdown,
|
||||||
ProjectMemberDropdown,
|
ProjectMemberDropdown,
|
||||||
@ -44,7 +44,7 @@ const defaultValues: Partial<TIssue> = {
|
|||||||
assignee_ids: [],
|
assignee_ids: [],
|
||||||
label_ids: [],
|
label_ids: [],
|
||||||
cycle_id: null,
|
cycle_id: null,
|
||||||
module_id: null,
|
module_ids: null,
|
||||||
start_date: null,
|
start_date: null,
|
||||||
target_date: null,
|
target_date: null,
|
||||||
};
|
};
|
||||||
@ -541,21 +541,24 @@ export const IssueFormRoot: FC<IssueFormProps> = observer((props) => {
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{projectDetails?.module_view && (
|
{projectDetails?.module_view && workspaceSlug && (
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="module_id"
|
name="module_ids"
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
<div className="h-7">
|
<div className="h-7">
|
||||||
<ModuleDropdown
|
<ModuleSelectDropdown
|
||||||
|
workspaceSlug={workspaceSlug.toString()}
|
||||||
projectId={projectId}
|
projectId={projectId}
|
||||||
value={value}
|
value={value || undefined}
|
||||||
onChange={(moduleId) => {
|
onChange={(moduleId) => {
|
||||||
onChange(moduleId);
|
onChange(moduleId);
|
||||||
handleFormChange();
|
handleFormChange();
|
||||||
}}
|
}}
|
||||||
buttonVariant="border-with-text"
|
buttonVariant="border-with-text"
|
||||||
tabIndex={13}
|
tabIndex={13}
|
||||||
|
multiple={true}
|
||||||
|
showCount={true}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
@ -108,11 +108,11 @@ export const CreateUpdateIssueModal: React.FC<IssuesModalProps> = observer((prop
|
|||||||
fetchCycleDetails(workspaceSlug, activeProjectId, cycleId);
|
fetchCycleDetails(workspaceSlug, activeProjectId, cycleId);
|
||||||
};
|
};
|
||||||
|
|
||||||
const addIssueToModule = async (issue: TIssue, moduleId: string) => {
|
const addIssueToModule = async (issue: TIssue, moduleIds: string[]) => {
|
||||||
if (!workspaceSlug || !activeProjectId) return;
|
if (!workspaceSlug || !activeProjectId) return;
|
||||||
|
|
||||||
await moduleIssues.addIssueToModule(workspaceSlug, activeProjectId, moduleId, [issue.id]);
|
await moduleIssues.addModulesToIssue(workspaceSlug, activeProjectId, issue.id, moduleIds);
|
||||||
fetchModuleDetails(workspaceSlug, activeProjectId, moduleId);
|
moduleIds.forEach((moduleId) => fetchModuleDetails(workspaceSlug, activeProjectId, moduleId));
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleCreateMoreToggleChange = (value: boolean) => {
|
const handleCreateMoreToggleChange = (value: boolean) => {
|
||||||
@ -139,8 +139,8 @@ export const CreateUpdateIssueModal: React.FC<IssuesModalProps> = observer((prop
|
|||||||
|
|
||||||
if (payload.cycle_id && payload.cycle_id !== "" && storeType !== EIssuesStoreType.CYCLE)
|
if (payload.cycle_id && payload.cycle_id !== "" && storeType !== EIssuesStoreType.CYCLE)
|
||||||
await addIssueToCycle(response, payload.cycle_id);
|
await addIssueToCycle(response, payload.cycle_id);
|
||||||
if (payload.module_id && payload.module_id !== "" && storeType !== EIssuesStoreType.MODULE)
|
if (payload.module_ids && payload.module_ids.length > 0 && storeType !== EIssuesStoreType.MODULE)
|
||||||
await addIssueToModule(response, payload.module_id);
|
await addIssueToModule(response, payload.module_ids);
|
||||||
|
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -278,7 +278,7 @@ export const CreateUpdateIssueModal: React.FC<IssuesModalProps> = observer((prop
|
|||||||
data={{
|
data={{
|
||||||
...data,
|
...data,
|
||||||
cycle_id: data?.cycle_id ? data?.cycle_id : cycleId ? cycleId : null,
|
cycle_id: data?.cycle_id ? data?.cycle_id : cycleId ? cycleId : null,
|
||||||
module_id: data?.module_id ? data?.module_id : moduleId ? moduleId : null,
|
module_ids: data?.module_ids ? data?.module_ids : moduleId ? [moduleId] : null,
|
||||||
}}
|
}}
|
||||||
onChange={handleFormChange}
|
onChange={handleFormChange}
|
||||||
onClose={handleClose}
|
onClose={handleClose}
|
||||||
@ -292,7 +292,7 @@ export const CreateUpdateIssueModal: React.FC<IssuesModalProps> = observer((prop
|
|||||||
data={{
|
data={{
|
||||||
...data,
|
...data,
|
||||||
cycle_id: data?.cycle_id ? data?.cycle_id : cycleId ? cycleId : null,
|
cycle_id: data?.cycle_id ? data?.cycle_id : cycleId ? cycleId : null,
|
||||||
module_id: data?.module_id ? data?.module_id : moduleId ? moduleId : null,
|
module_ids: data?.module_ids ? data?.module_ids : moduleId ? [moduleId] : null,
|
||||||
}}
|
}}
|
||||||
onClose={() => handleClose(false)}
|
onClose={() => handleClose(false)}
|
||||||
isCreateMoreToggleEnabled={createMore}
|
isCreateMoreToggleEnabled={createMore}
|
||||||
|
@ -203,7 +203,7 @@ export const PeekOverviewProperties: FC<IPeekOverviewProperties> = observer((pro
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{projectDetails?.module_view && (
|
{projectDetails?.module_view && (
|
||||||
<div className="flex w-full items-center gap-3 h-8">
|
<div className="flex w-full items-center gap-3 min-h-8 h-full">
|
||||||
<div className="flex items-center gap-1 w-1/4 flex-shrink-0 text-sm text-custom-text-300">
|
<div className="flex items-center gap-1 w-1/4 flex-shrink-0 text-sm text-custom-text-300">
|
||||||
<DiceIcon className="h-4 w-4 flex-shrink-0" />
|
<DiceIcon className="h-4 w-4 flex-shrink-0" />
|
||||||
<span>Module</span>
|
<span>Module</span>
|
||||||
|
@ -28,8 +28,19 @@ export type TIssuePeekOperations = {
|
|||||||
remove: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
|
remove: (workspaceSlug: string, projectId: string, issueId: string) => Promise<void>;
|
||||||
addIssueToCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise<void>;
|
addIssueToCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise<void>;
|
||||||
removeIssueFromCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise<void>;
|
removeIssueFromCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise<void>;
|
||||||
addIssueToModule: (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => Promise<void>;
|
addModulesToIssue?: (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => Promise<void>;
|
||||||
removeIssueFromModule: (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => Promise<void>;
|
removeIssueFromModule?: (
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
moduleId: string,
|
||||||
|
issueId: string
|
||||||
|
) => Promise<void>;
|
||||||
|
removeModulesFromIssue?: (
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
moduleIds: string[]
|
||||||
|
) => Promise<void>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
|
export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
|
||||||
@ -48,7 +59,8 @@ export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
|
|||||||
removeIssue,
|
removeIssue,
|
||||||
issue: { getIssueById, fetchIssue },
|
issue: { getIssueById, fetchIssue },
|
||||||
} = useIssueDetail();
|
} = useIssueDetail();
|
||||||
const { addIssueToCycle, removeIssueFromCycle, addIssueToModule, removeIssueFromModule } = useIssueDetail();
|
const { addIssueToCycle, removeIssueFromCycle, addModulesToIssue, removeIssueFromModule, removeModulesFromIssue } =
|
||||||
|
useIssueDetail();
|
||||||
// state
|
// state
|
||||||
const [loader, setLoader] = useState(false);
|
const [loader, setLoader] = useState(false);
|
||||||
|
|
||||||
@ -143,9 +155,9 @@ export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
addIssueToModule: async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => {
|
addModulesToIssue: async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => {
|
||||||
try {
|
try {
|
||||||
await addIssueToModule(workspaceSlug, projectId, moduleId, issueIds);
|
await addModulesToIssue(workspaceSlug, projectId, issueId, moduleIds);
|
||||||
setToastAlert({
|
setToastAlert({
|
||||||
title: "Module added to issue successfully",
|
title: "Module added to issue successfully",
|
||||||
type: "success",
|
type: "success",
|
||||||
@ -175,6 +187,27 @@ export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
removeModulesFromIssue: async (
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
moduleIds: string[]
|
||||||
|
) => {
|
||||||
|
try {
|
||||||
|
await removeModulesFromIssue(workspaceSlug, projectId, issueId, moduleIds);
|
||||||
|
setToastAlert({
|
||||||
|
title: "Module removed from issue successfully",
|
||||||
|
type: "success",
|
||||||
|
message: "Module removed from issue successfully",
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
setToastAlert({
|
||||||
|
title: "Module remove from issue failed",
|
||||||
|
type: "error",
|
||||||
|
message: "Module remove from issue failed",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
}),
|
}),
|
||||||
[
|
[
|
||||||
is_archived,
|
is_archived,
|
||||||
@ -184,8 +217,9 @@ export const IssuePeekOverview: FC<IIssuePeekOverview> = observer((props) => {
|
|||||||
removeArchivedIssue,
|
removeArchivedIssue,
|
||||||
addIssueToCycle,
|
addIssueToCycle,
|
||||||
removeIssueFromCycle,
|
removeIssueFromCycle,
|
||||||
addIssueToModule,
|
addModulesToIssue,
|
||||||
removeIssueFromModule,
|
removeIssueFromModule,
|
||||||
|
removeModulesFromIssue,
|
||||||
setToastAlert,
|
setToastAlert,
|
||||||
onIssueUpdate,
|
onIssueUpdate,
|
||||||
]
|
]
|
||||||
|
@ -234,7 +234,6 @@ export const IssueView: FC<IIssueView> = observer((props) => {
|
|||||||
workspaceSlug={workspaceSlug}
|
workspaceSlug={workspaceSlug}
|
||||||
projectId={projectId}
|
projectId={projectId}
|
||||||
issueId={issueId}
|
issueId={issueId}
|
||||||
disabled={disabled}
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
@ -255,7 +254,6 @@ export const IssueView: FC<IIssueView> = observer((props) => {
|
|||||||
workspaceSlug={workspaceSlug}
|
workspaceSlug={workspaceSlug}
|
||||||
projectId={projectId}
|
projectId={projectId}
|
||||||
issueId={issueId}
|
issueId={issueId}
|
||||||
disabled={disabled}
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -131,6 +131,12 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
setLeaveProjectModal(false);
|
setLeaveProjectModal(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleProjectClick = () => {
|
||||||
|
if (window.innerWidth < 768) {
|
||||||
|
themeStore.toggleSidebar();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
useOutsideClickDetector(actionSectionRef, () => setIsMenuActive(false));
|
useOutsideClickDetector(actionSectionRef, () => setIsMenuActive(false));
|
||||||
|
|
||||||
if (!project) return null;
|
if (!project) return null;
|
||||||
@ -143,8 +149,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
{({ open }) => (
|
{({ open }) => (
|
||||||
<>
|
<>
|
||||||
<div
|
<div
|
||||||
className={`group relative flex w-full items-center rounded-md px-2 py-1 text-custom-sidebar-text-10 hover:bg-custom-sidebar-background-80 ${
|
className={`group relative flex w-full items-center rounded-md px-2 py-1 text-custom-sidebar-text-10 hover:bg-custom-sidebar-background-80 ${snapshot?.isDragging ? "opacity-60" : ""
|
||||||
snapshot?.isDragging ? "opacity-60" : ""
|
|
||||||
} ${isMenuActive ? "!bg-custom-sidebar-background-80" : ""}`}
|
} ${isMenuActive ? "!bg-custom-sidebar-background-80" : ""}`}
|
||||||
>
|
>
|
||||||
{provided && (
|
{provided && (
|
||||||
@ -154,10 +159,8 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className={`absolute -left-2.5 top-1/2 hidden -translate-y-1/2 rounded p-0.5 text-custom-sidebar-text-400 ${
|
className={`absolute -left-2.5 top-1/2 hidden -translate-y-1/2 rounded p-0.5 text-custom-sidebar-text-400 ${isCollapsed ? "" : "group-hover:!flex"
|
||||||
isCollapsed ? "" : "group-hover:!flex"
|
} ${project.sort_order === null ? "cursor-not-allowed opacity-60" : ""} ${isMenuActive ? "!flex" : ""
|
||||||
} ${project.sort_order === null ? "cursor-not-allowed opacity-60" : ""} ${
|
|
||||||
isMenuActive ? "!flex" : ""
|
|
||||||
}`}
|
}`}
|
||||||
{...provided?.dragHandleProps}
|
{...provided?.dragHandleProps}
|
||||||
>
|
>
|
||||||
@ -169,13 +172,11 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
<Tooltip tooltipContent={`${project.name}`} position="right" className="ml-2" disabled={!isCollapsed}>
|
<Tooltip tooltipContent={`${project.name}`} position="right" className="ml-2" disabled={!isCollapsed}>
|
||||||
<Disclosure.Button
|
<Disclosure.Button
|
||||||
as="div"
|
as="div"
|
||||||
className={`flex flex-grow cursor-pointer select-none items-center truncate text-left text-sm font-medium ${
|
className={`flex flex-grow cursor-pointer select-none items-center truncate text-left text-sm font-medium ${isCollapsed ? "justify-center" : `justify-between`
|
||||||
isCollapsed ? "justify-center" : `justify-between`
|
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className={`flex w-full flex-grow items-center gap-x-2 truncate ${
|
className={`flex w-full flex-grow items-center gap-x-2 truncate ${isCollapsed ? "justify-center" : ""
|
||||||
isCollapsed ? "justify-center" : ""
|
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{project.emoji ? (
|
{project.emoji ? (
|
||||||
@ -196,8 +197,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
</div>
|
</div>
|
||||||
{!isCollapsed && (
|
{!isCollapsed && (
|
||||||
<ChevronDown
|
<ChevronDown
|
||||||
className={`hidden h-4 w-4 flex-shrink-0 ${open ? "rotate-180" : ""} ${
|
className={`hidden h-4 w-4 flex-shrink-0 ${open ? "rotate-180" : ""} ${isMenuActive ? "!block" : ""
|
||||||
isMenuActive ? "!block" : ""
|
|
||||||
} mb-0.5 text-custom-sidebar-text-400 duration-300 group-hover:!block`}
|
} mb-0.5 text-custom-sidebar-text-400 duration-300 group-hover:!block`}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@ -313,7 +313,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Link key={item.name} href={item.href}>
|
<Link key={item.name} href={item.href} onClick={handleProjectClick}>
|
||||||
<span className="block w-full">
|
<span className="block w-full">
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={`${project?.name}: ${item.name}`}
|
tooltipContent={`${project?.name}: ${item.name}`}
|
||||||
@ -322,8 +322,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
|
|||||||
disabled={!isCollapsed}
|
disabled={!isCollapsed}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className={`group flex items-center gap-2.5 rounded-md px-2 py-1.5 text-xs font-medium outline-none ${
|
className={`group flex items-center gap-2.5 rounded-md px-2 py-1.5 text-xs font-medium outline-none ${router.asPath.includes(item.href)
|
||||||
router.asPath.includes(item.href)
|
|
||||||
? "bg-custom-primary-100/10 text-custom-primary-100"
|
? "bg-custom-primary-100/10 text-custom-primary-100"
|
||||||
: "text-custom-sidebar-text-300 hover:bg-custom-sidebar-background-80 focus:bg-custom-sidebar-background-80"
|
: "text-custom-sidebar-text-300 hover:bg-custom-sidebar-background-80 focus:bg-custom-sidebar-background-80"
|
||||||
} ${isCollapsed ? "justify-center" : ""}`}
|
} ${isCollapsed ? "justify-center" : ""}`}
|
||||||
|
@ -56,7 +56,7 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
const { workspaceSlug } = router.query;
|
const { workspaceSlug } = router.query;
|
||||||
// store hooks
|
// store hooks
|
||||||
const {
|
const {
|
||||||
theme: { sidebarCollapsed },
|
theme: { sidebarCollapsed, toggleSidebar },
|
||||||
eventTracker: { setTrackElement },
|
eventTracker: { setTrackElement },
|
||||||
} = useApplication();
|
} = useApplication();
|
||||||
const { currentUser, updateCurrentUser, isUserInstanceAdmin, signOut } = useUser();
|
const { currentUser, updateCurrentUser, isUserInstanceAdmin, signOut } = useUser();
|
||||||
@ -86,6 +86,13 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleItemClick = () => {
|
||||||
|
console.log('CLICKED')
|
||||||
|
if (window.innerWidth < 768) {
|
||||||
|
toggleSidebar();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const workspacesList = Object.values(workspaces ?? {});
|
const workspacesList = Object.values(workspaces ?? {});
|
||||||
|
|
||||||
// TODO: fix workspaces list scroll
|
// TODO: fix workspaces list scroll
|
||||||
@ -96,14 +103,12 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
<>
|
<>
|
||||||
<Menu.Button className="group/menu-button h-full w-full truncate rounded-md text-sm font-medium text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 focus:outline-none">
|
<Menu.Button className="group/menu-button h-full w-full truncate rounded-md text-sm font-medium text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 focus:outline-none">
|
||||||
<div
|
<div
|
||||||
className={`flex items-center gap-x-2 truncate rounded p-1 ${
|
className={`flex items-center gap-x-2 truncate rounded p-1 ${sidebarCollapsed ? "justify-center" : "justify-between"
|
||||||
sidebarCollapsed ? "justify-center" : "justify-between"
|
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
<div className="flex items-center gap-2 truncate">
|
<div className="flex items-center gap-2 truncate">
|
||||||
<div
|
<div
|
||||||
className={`relative grid h-6 w-6 flex-shrink-0 place-items-center uppercase ${
|
className={`relative grid h-6 w-6 flex-shrink-0 place-items-center uppercase ${!activeWorkspace?.logo && "rounded bg-custom-primary-500 text-white"
|
||||||
!activeWorkspace?.logo && "rounded bg-custom-primary-500 text-white"
|
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{activeWorkspace?.logo && activeWorkspace.logo !== "" ? (
|
{activeWorkspace?.logo && activeWorkspace.logo !== "" ? (
|
||||||
@ -126,8 +131,7 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
|
|
||||||
{!sidebarCollapsed && (
|
{!sidebarCollapsed && (
|
||||||
<ChevronDown
|
<ChevronDown
|
||||||
className={`mx-1 hidden h-4 w-4 flex-shrink-0 group-hover/menu-button:block ${
|
className={`mx-1 hidden h-4 w-4 flex-shrink-0 group-hover/menu-button:block ${open ? "rotate-180" : ""
|
||||||
open ? "rotate-180" : ""
|
|
||||||
} text-custom-sidebar-text-400 duration-300`}
|
} text-custom-sidebar-text-400 duration-300`}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@ -156,7 +160,10 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
<Link
|
<Link
|
||||||
key={workspace.id}
|
key={workspace.id}
|
||||||
href={`/${workspace.slug}`}
|
href={`/${workspace.slug}`}
|
||||||
onClick={() => handleWorkspaceNavigation(workspace)}
|
onClick={() => {
|
||||||
|
handleWorkspaceNavigation(workspace);
|
||||||
|
handleItemClick();
|
||||||
|
}}
|
||||||
className="w-full"
|
className="w-full"
|
||||||
>
|
>
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
@ -165,8 +172,7 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
>
|
>
|
||||||
<div className="flex items-center justify-start gap-2.5 truncate">
|
<div className="flex items-center justify-start gap-2.5 truncate">
|
||||||
<span
|
<span
|
||||||
className={`relative flex h-6 w-6 flex-shrink-0 items-center justify-center p-2 text-xs uppercase ${
|
className={`relative flex h-6 w-6 flex-shrink-0 items-center justify-center p-2 text-xs uppercase ${!workspace?.logo && "rounded bg-custom-primary-500 text-white"
|
||||||
!workspace?.logo && "rounded bg-custom-primary-500 text-white"
|
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{workspace?.logo && workspace.logo !== "" ? (
|
{workspace?.logo && workspace.logo !== "" ? (
|
||||||
@ -181,8 +187,7 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
</span>
|
</span>
|
||||||
|
|
||||||
<h5
|
<h5
|
||||||
className={`truncate text-sm font-medium ${
|
className={`truncate text-sm font-medium ${workspaceSlug === workspace.slug ? "" : "text-custom-text-200"
|
||||||
workspaceSlug === workspace.slug ? "" : "text-custom-text-200"
|
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{workspace.name}
|
{workspace.name}
|
||||||
@ -220,8 +225,10 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
Create workspace
|
Create workspace
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
</Link>
|
</Link>
|
||||||
{userLinks(workspaceSlug?.toString() ?? "", currentUser?.id ?? "").map((link) => (
|
{userLinks(workspaceSlug?.toString() ?? "", currentUser?.id ?? "").map((link, index) => (
|
||||||
<Link key={link.key} href={link.href} className="w-full">
|
<Link key={link.key} href={link.href} className="w-full" onClick={() => {
|
||||||
|
if (index > 0) handleItemClick();
|
||||||
|
}}>
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
as="div"
|
as="div"
|
||||||
className="flex items-center gap-2 rounded px-2 py-1 text-sm text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 font-medium"
|
className="flex items-center gap-2 rounded px-2 py-1 text-sm text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 font-medium"
|
||||||
@ -278,7 +285,7 @@ export const WorkspaceSidebarDropdown = observer(() => {
|
|||||||
<div className="flex flex-col gap-2.5 pb-2">
|
<div className="flex flex-col gap-2.5 pb-2">
|
||||||
<span className="px-2 text-custom-sidebar-text-200">{currentUser?.email}</span>
|
<span className="px-2 text-custom-sidebar-text-200">{currentUser?.email}</span>
|
||||||
{profileLinks(workspaceSlug?.toString() ?? "", currentUser?.id ?? "").map((link, index) => (
|
{profileLinks(workspaceSlug?.toString() ?? "", currentUser?.id ?? "").map((link, index) => (
|
||||||
<Link key={index} href={link.link}>
|
<Link key={index} href={link.link} onClick={() => { if (index == 0) handleItemClick(); }}>
|
||||||
<Menu.Item key={index} as="div">
|
<Menu.Item key={index} as="div">
|
||||||
<span className="flex w-full items-center gap-2 rounded px-2 py-1 hover:bg-custom-sidebar-background-80">
|
<span className="flex w-full items-center gap-2 rounded px-2 py-1 hover:bg-custom-sidebar-background-80">
|
||||||
<link.icon className="h-4 w-4 stroke-[1.5]" />
|
<link.icon className="h-4 w-4 stroke-[1.5]" />
|
||||||
|
@ -27,12 +27,21 @@ export const WorkspaceSidebarMenu = observer(() => {
|
|||||||
// computed
|
// computed
|
||||||
const workspaceMemberInfo = currentWorkspaceRole || EUserWorkspaceRoles.GUEST;
|
const workspaceMemberInfo = currentWorkspaceRole || EUserWorkspaceRoles.GUEST;
|
||||||
|
|
||||||
|
const handleLinkClick = () => {
|
||||||
|
if (window.innerWidth < 768) {
|
||||||
|
themeStore.toggleSidebar();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-full cursor-pointer space-y-2 p-4">
|
<div className="w-full cursor-pointer space-y-2 p-4">
|
||||||
{SIDEBAR_MENU_ITEMS.map(
|
{SIDEBAR_MENU_ITEMS.map(
|
||||||
(link) =>
|
(link) =>
|
||||||
workspaceMemberInfo >= link.access && (
|
workspaceMemberInfo >= link.access && (
|
||||||
<Link key={link.key} href={`/${workspaceSlug}${link.href}`}>
|
<Link key={link.key}
|
||||||
|
href={`/${workspaceSlug}${link.href}`}
|
||||||
|
onClick={handleLinkClick}
|
||||||
|
>
|
||||||
<span className="block w-full my-1">
|
<span className="block w-full my-1">
|
||||||
<Tooltip
|
<Tooltip
|
||||||
tooltipContent={link.label}
|
tooltipContent={link.label}
|
||||||
@ -41,8 +50,7 @@ export const WorkspaceSidebarMenu = observer(() => {
|
|||||||
disabled={!themeStore?.sidebarCollapsed}
|
disabled={!themeStore?.sidebarCollapsed}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className={`group flex w-full items-center gap-2.5 rounded-md px-3 py-2 text-sm font-medium outline-none ${
|
className={`group flex w-full items-center gap-2.5 rounded-md px-3 py-2 text-sm font-medium outline-none ${link.highlight(router.asPath, `/${workspaceSlug}`)
|
||||||
link.highlight(router.asPath, `/${workspaceSlug}`)
|
|
||||||
? "bg-custom-primary-100/10 text-custom-primary-100"
|
? "bg-custom-primary-100/10 text-custom-primary-100"
|
||||||
: "text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 focus:bg-custom-sidebar-background-80"
|
: "text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 focus:bg-custom-sidebar-background-80"
|
||||||
} ${themeStore?.sidebarCollapsed ? "justify-center" : ""}`}
|
} ${themeStore?.sidebarCollapsed ? "justify-center" : ""}`}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { FC } from "react";
|
import { FC, useEffect, useRef } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
// components
|
// components
|
||||||
import {
|
import {
|
||||||
@ -10,20 +10,48 @@ import {
|
|||||||
import { ProjectSidebarList } from "components/project";
|
import { ProjectSidebarList } from "components/project";
|
||||||
// hooks
|
// hooks
|
||||||
import { useApplication } from "hooks/store";
|
import { useApplication } from "hooks/store";
|
||||||
|
import useOutsideClickDetector from "hooks/use-outside-click-detector";
|
||||||
|
|
||||||
export interface IAppSidebar {}
|
export interface IAppSidebar { }
|
||||||
|
|
||||||
export const AppSidebar: FC<IAppSidebar> = observer(() => {
|
export const AppSidebar: FC<IAppSidebar> = observer(() => {
|
||||||
// store hooks
|
// store hooks
|
||||||
const { theme: themStore } = useApplication();
|
const { theme: themStore } = useApplication();
|
||||||
|
const ref = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
useOutsideClickDetector(ref, () => {
|
||||||
|
if (themStore.sidebarCollapsed === false) {
|
||||||
|
if (window.innerWidth < 768) {
|
||||||
|
themStore.toggleSidebar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleResize = () => {
|
||||||
|
if (window.innerWidth <= 768) {
|
||||||
|
themStore.toggleSidebar(true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
handleResize();
|
||||||
|
window.addEventListener('resize', handleResize);
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('resize', handleResize);
|
||||||
|
};
|
||||||
|
}, [themStore]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={`fixed inset-y-0 z-20 flex h-full flex-shrink-0 flex-grow-0 flex-col border-r border-custom-sidebar-border-200 bg-custom-sidebar-background-100 duration-300 md:relative ${
|
className={`inset-y-0 z-20 flex h-full flex-shrink-0 flex-grow-0 flex-col border-r border-custom-sidebar-border-200 bg-custom-sidebar-background-100 duration-300
|
||||||
themStore?.sidebarCollapsed ? "" : "md:w-[280px]"
|
fixed md:relative
|
||||||
} ${themStore?.sidebarCollapsed ? "left-0" : "-left-full md:left-0"}`}
|
${themStore.sidebarCollapsed ? "-ml-[280px]" : ""}
|
||||||
>
|
sm:${themStore.sidebarCollapsed ? "-ml-[280px]" : ""}
|
||||||
<div className="flex h-full w-full flex-1 flex-col">
|
md:ml-0 ${themStore.sidebarCollapsed ? 'w-[80px]' : 'w-[280px]'}
|
||||||
|
lg:ml-0 ${themStore.sidebarCollapsed ? 'w-[80px]' : 'w-[280px]'}
|
||||||
|
`} >
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className="flex h-full w-full flex-1 flex-col">
|
||||||
<WorkspaceSidebarDropdown />
|
<WorkspaceSidebarDropdown />
|
||||||
<WorkspaceSidebarQuickAction />
|
<WorkspaceSidebarQuickAction />
|
||||||
<WorkspaceSidebarMenu />
|
<WorkspaceSidebarMenu />
|
||||||
@ -33,3 +61,6 @@ export const AppSidebar: FC<IAppSidebar> = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -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>
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { ReactElement } from "react";
|
import { ReactElement, useState } from "react";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
@ -19,6 +19,8 @@ import BluePlaneLogoWithoutText from "public/plane-logos/blue-without-text.png";
|
|||||||
import { checkEmailValidity } from "helpers/string.helper";
|
import { checkEmailValidity } from "helpers/string.helper";
|
||||||
// type
|
// type
|
||||||
import { NextPageWithLayout } from "lib/types";
|
import { NextPageWithLayout } from "lib/types";
|
||||||
|
// icons
|
||||||
|
import { Eye, EyeOff } from "lucide-react";
|
||||||
|
|
||||||
type TResetPasswordFormValues = {
|
type TResetPasswordFormValues = {
|
||||||
email: string;
|
email: string;
|
||||||
@ -37,6 +39,8 @@ const ResetPasswordPage: NextPageWithLayout = () => {
|
|||||||
// router
|
// router
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const { uidb64, token, email } = router.query;
|
const { uidb64, token, email } = router.query;
|
||||||
|
// states
|
||||||
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
// toast
|
// toast
|
||||||
const { setToastAlert } = useToast();
|
const { setToastAlert } = useToast();
|
||||||
// sign in redirection hook
|
// sign in redirection hook
|
||||||
@ -117,8 +121,9 @@ const ResetPasswordPage: NextPageWithLayout = () => {
|
|||||||
required: "Password is required",
|
required: "Password is required",
|
||||||
}}
|
}}
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
|
<div className="relative flex items-center rounded-md bg-onboarding-background-200">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type={showPassword ? "text" : "password"}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
hasError={Boolean(errors.password)}
|
hasError={Boolean(errors.password)}
|
||||||
@ -126,6 +131,18 @@ const ResetPasswordPage: NextPageWithLayout = () => {
|
|||||||
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400"
|
||||||
minLength={8}
|
minLength={8}
|
||||||
/>
|
/>
|
||||||
|
{showPassword ? (
|
||||||
|
<EyeOff
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(false)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Eye
|
||||||
|
className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer"
|
||||||
|
onClick={() => setShowPassword(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<Button
|
<Button
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
// services
|
// services
|
||||||
import { APIService } from "services/api.service";
|
import { APIService } from "services/api.service";
|
||||||
// types
|
// types
|
||||||
import type { IModule, TIssue, ILinkDetails, ModuleLink, TIssueMap } from "@plane/types";
|
import type { IModule, TIssue, ILinkDetails, ModuleLink } from "@plane/types";
|
||||||
import { API_BASE_URL } from "helpers/common.helper";
|
import { API_BASE_URL } from "helpers/common.helper";
|
||||||
|
|
||||||
export class ModuleService extends APIService {
|
export class ModuleService extends APIService {
|
||||||
@ -63,22 +63,7 @@ export class ModuleService extends APIService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async getModuleIssues(workspaceSlug: string, projectId: string, moduleId: string, queries?: any): Promise<TIssue[]> {
|
async getModuleIssues(workspaceSlug: string, projectId: string, moduleId: string, queries?: any): Promise<TIssue[]> {
|
||||||
return this.get(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/module-issues/`, {
|
return this.get(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/issues/`, {
|
||||||
params: queries,
|
|
||||||
})
|
|
||||||
.then((response) => response?.data)
|
|
||||||
.catch((error) => {
|
|
||||||
throw error?.response?.data;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async getModuleIssuesWithParams(
|
|
||||||
workspaceSlug: string,
|
|
||||||
projectId: string,
|
|
||||||
moduleId: string,
|
|
||||||
queries?: any
|
|
||||||
): Promise<TIssue[] | { [key: string]: TIssue[] }> {
|
|
||||||
return this.get(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/module-issues/`, {
|
|
||||||
params: queries,
|
params: queries,
|
||||||
})
|
})
|
||||||
.then((response) => response?.data)
|
.then((response) => response?.data)
|
||||||
@ -92,15 +77,21 @@ export class ModuleService extends APIService {
|
|||||||
projectId: string,
|
projectId: string,
|
||||||
moduleId: string,
|
moduleId: string,
|
||||||
data: { issues: string[] }
|
data: { issues: string[] }
|
||||||
): Promise<
|
): Promise<void> {
|
||||||
{
|
return this.post(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/issues/`, data)
|
||||||
issue: string;
|
.then((response) => response?.data)
|
||||||
issue_detail: TIssue;
|
.catch((error) => {
|
||||||
module: string;
|
throw error?.response?.data;
|
||||||
module_detail: IModule;
|
});
|
||||||
}[]
|
}
|
||||||
> {
|
|
||||||
return this.post(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/module-issues/`, data)
|
async addModulesToIssue(
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
data: { modules: string[] }
|
||||||
|
): Promise<void> {
|
||||||
|
return this.post(`/api/workspaces/${workspaceSlug}/projects/${projectId}/issues/${issueId}/modules/`, data)
|
||||||
.then((response) => response?.data)
|
.then((response) => response?.data)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
throw error?.response?.data;
|
throw error?.response?.data;
|
||||||
@ -111,17 +102,53 @@ export class ModuleService extends APIService {
|
|||||||
workspaceSlug: string,
|
workspaceSlug: string,
|
||||||
projectId: string,
|
projectId: string,
|
||||||
moduleId: string,
|
moduleId: string,
|
||||||
bridgeId: string
|
issueId: string
|
||||||
): Promise<any> {
|
): Promise<any> {
|
||||||
return this.delete(
|
return this.delete(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/issues/${issueId}/`)
|
||||||
`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/module-issues/${bridgeId}/`
|
|
||||||
)
|
|
||||||
.then((response) => response?.data)
|
.then((response) => response?.data)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
throw error?.response?.data;
|
throw error?.response?.data;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async removeIssuesFromModuleBulk(
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
moduleId: string,
|
||||||
|
issueIds: string[]
|
||||||
|
): Promise<any> {
|
||||||
|
const promiseDataUrls: any = [];
|
||||||
|
issueIds.forEach((issueId) => {
|
||||||
|
promiseDataUrls.push(
|
||||||
|
this.delete(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/issues/${issueId}/`)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
return await Promise.all(promiseDataUrls)
|
||||||
|
.then((response) => response)
|
||||||
|
.catch((error) => {
|
||||||
|
throw error?.response?.data;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async removeModulesFromIssueBulk(
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
moduleIds: string[]
|
||||||
|
): Promise<any> {
|
||||||
|
const promiseDataUrls: any = [];
|
||||||
|
moduleIds.forEach((moduleId) => {
|
||||||
|
promiseDataUrls.push(
|
||||||
|
this.delete(`/api/workspaces/${workspaceSlug}/projects/${projectId}/modules/${moduleId}/issues/${issueId}/`)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
return await Promise.all(promiseDataUrls)
|
||||||
|
.then((response) => response)
|
||||||
|
.catch((error) => {
|
||||||
|
throw error?.response?.data;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async createModuleLink(
|
async createModuleLink(
|
||||||
workspaceSlug: string,
|
workspaceSlug: string,
|
||||||
projectId: string,
|
projectId: string,
|
||||||
|
@ -268,7 +268,7 @@ export class CycleIssues extends IssueHelperStore implements ICycleIssues {
|
|||||||
|
|
||||||
runInAction(() => {
|
runInAction(() => {
|
||||||
update(this.issues, cycleId, (cycleIssueIds = []) => {
|
update(this.issues, cycleId, (cycleIssueIds = []) => {
|
||||||
uniq(concat(cycleIssueIds, issueIds));
|
return uniq(concat(cycleIssueIds, issueIds));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
issueIds.forEach((issueId) => {
|
issueIds.forEach((issueId) => {
|
||||||
|
@ -12,13 +12,14 @@ export interface IIssueStoreActions {
|
|||||||
removeIssue: (workspaceSlug: string, projectId: string, issueId: string) => Promise<TIssue>;
|
removeIssue: (workspaceSlug: string, projectId: string, issueId: string) => Promise<TIssue>;
|
||||||
addIssueToCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise<TIssue>;
|
addIssueToCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise<TIssue>;
|
||||||
removeIssueFromCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise<TIssue>;
|
removeIssueFromCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise<TIssue>;
|
||||||
addIssueToModule: (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => Promise<any>;
|
addModulesToIssue: (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => Promise<any>;
|
||||||
removeIssueFromModule: (
|
removeModulesFromIssue: (
|
||||||
workspaceSlug: string,
|
workspaceSlug: string,
|
||||||
projectId: string,
|
projectId: string,
|
||||||
moduleId: string,
|
issueId: string,
|
||||||
issueId: string
|
moduleIds: string[]
|
||||||
) => Promise<TIssue>;
|
) => Promise<void>;
|
||||||
|
removeIssueFromModule: (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IIssueStore extends IIssueStoreActions {
|
export interface IIssueStore extends IIssueStoreActions {
|
||||||
@ -143,15 +144,26 @@ export class IssueStore implements IIssueStore {
|
|||||||
return cycle;
|
return cycle;
|
||||||
};
|
};
|
||||||
|
|
||||||
addIssueToModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => {
|
addModulesToIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => {
|
||||||
const _module = await this.rootIssueDetailStore.rootIssueStore.moduleIssues.addIssueToModule(
|
const _module = await this.rootIssueDetailStore.rootIssueStore.moduleIssues.addModulesToIssue(
|
||||||
workspaceSlug,
|
workspaceSlug,
|
||||||
projectId,
|
projectId,
|
||||||
moduleId,
|
issueId,
|
||||||
issueIds
|
moduleIds
|
||||||
);
|
);
|
||||||
if (issueIds && issueIds.length > 0)
|
if (moduleIds && moduleIds.length > 0)
|
||||||
await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueIds[0]);
|
await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId);
|
||||||
|
return _module;
|
||||||
|
};
|
||||||
|
|
||||||
|
removeModulesFromIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => {
|
||||||
|
const _module = await this.rootIssueDetailStore.rootIssueStore.moduleIssues.removeModulesFromIssue(
|
||||||
|
workspaceSlug,
|
||||||
|
projectId,
|
||||||
|
issueId,
|
||||||
|
moduleIds
|
||||||
|
);
|
||||||
|
await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId);
|
||||||
return _module;
|
return _module;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -143,8 +143,10 @@ export class IssueDetail implements IIssueDetail {
|
|||||||
this.issue.addIssueToCycle(workspaceSlug, projectId, cycleId, issueIds);
|
this.issue.addIssueToCycle(workspaceSlug, projectId, cycleId, issueIds);
|
||||||
removeIssueFromCycle = async (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) =>
|
removeIssueFromCycle = async (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) =>
|
||||||
this.issue.removeIssueFromCycle(workspaceSlug, projectId, cycleId, issueId);
|
this.issue.removeIssueFromCycle(workspaceSlug, projectId, cycleId, issueId);
|
||||||
addIssueToModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) =>
|
addModulesToIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) =>
|
||||||
this.issue.addIssueToModule(workspaceSlug, projectId, moduleId, issueIds);
|
this.issue.addModulesToIssue(workspaceSlug, projectId, issueId, moduleIds);
|
||||||
|
removeModulesFromIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) =>
|
||||||
|
this.issue.removeModulesFromIssue(workspaceSlug, projectId, issueId, moduleIds);
|
||||||
removeIssueFromModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) =>
|
removeIssueFromModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) =>
|
||||||
this.issue.removeIssueFromModule(workspaceSlug, projectId, moduleId, issueId);
|
this.issue.removeIssueFromModule(workspaceSlug, projectId, moduleId, issueId);
|
||||||
|
|
||||||
|
@ -52,13 +52,21 @@ export interface IModuleIssues {
|
|||||||
data: TIssue,
|
data: TIssue,
|
||||||
moduleId?: string | undefined
|
moduleId?: string | undefined
|
||||||
) => Promise<TIssue | undefined>;
|
) => Promise<TIssue | undefined>;
|
||||||
addIssueToModule: (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => Promise<any>;
|
addIssuesToModule: (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => Promise<void>;
|
||||||
removeIssueFromModule: (
|
removeIssuesFromModule: (
|
||||||
workspaceSlug: string,
|
workspaceSlug: string,
|
||||||
projectId: string,
|
projectId: string,
|
||||||
moduleId: string,
|
moduleId: string,
|
||||||
issueId: string
|
issueIds: string[]
|
||||||
) => Promise<TIssue>;
|
) => Promise<void>;
|
||||||
|
addModulesToIssue: (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => Promise<void>;
|
||||||
|
removeModulesFromIssue: (
|
||||||
|
workspaceSlug: string,
|
||||||
|
projectId: string,
|
||||||
|
issueId: string,
|
||||||
|
moduleIds: string[]
|
||||||
|
) => Promise<void>;
|
||||||
|
removeIssueFromModule: (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
||||||
@ -90,7 +98,10 @@ export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
|||||||
updateIssue: action,
|
updateIssue: action,
|
||||||
removeIssue: action,
|
removeIssue: action,
|
||||||
quickAddIssue: action,
|
quickAddIssue: action,
|
||||||
addIssueToModule: action,
|
addIssuesToModule: action,
|
||||||
|
removeIssuesFromModule: action,
|
||||||
|
addModulesToIssue: action,
|
||||||
|
removeModulesFromIssue: action,
|
||||||
removeIssueFromModule: action,
|
removeIssueFromModule: action,
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -175,7 +186,7 @@ export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
|||||||
if (!moduleId) throw new Error("Module Id is required");
|
if (!moduleId) throw new Error("Module Id is required");
|
||||||
|
|
||||||
const response = await this.rootIssueStore.projectIssues.createIssue(workspaceSlug, projectId, data);
|
const response = await this.rootIssueStore.projectIssues.createIssue(workspaceSlug, projectId, data);
|
||||||
await this.addIssueToModule(workspaceSlug, projectId, moduleId, [response.id]);
|
await this.addIssuesToModule(workspaceSlug, projectId, moduleId, [response.id]);
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -253,7 +264,7 @@ export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
addIssueToModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => {
|
addIssuesToModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => {
|
||||||
try {
|
try {
|
||||||
const issueToModule = await this.moduleService.addIssuesToModule(workspaceSlug, projectId, moduleId, {
|
const issueToModule = await this.moduleService.addIssuesToModule(workspaceSlug, projectId, moduleId, {
|
||||||
issues: issueIds,
|
issues: issueIds,
|
||||||
@ -261,11 +272,16 @@ export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
|||||||
|
|
||||||
runInAction(() => {
|
runInAction(() => {
|
||||||
update(this.issues, moduleId, (moduleIssueIds = []) => {
|
update(this.issues, moduleId, (moduleIssueIds = []) => {
|
||||||
uniq(concat(moduleIssueIds, issueIds));
|
if (!moduleIssueIds) return [...issueIds];
|
||||||
|
else return uniq(concat(moduleIssueIds, issueIds));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
issueIds.forEach((issueId) => {
|
issueIds.forEach((issueId) => {
|
||||||
this.rootStore.issues.updateIssue(issueId, { module_id: moduleId });
|
update(this.rootStore.issues.issuesMap, [issueId, "module_ids"], (issueModuleIds = []) => {
|
||||||
|
if (issueModuleIds.includes(moduleId)) return issueModuleIds;
|
||||||
|
else return uniq(concat(issueModuleIds, [moduleId]));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
return issueToModule;
|
return issueToModule;
|
||||||
@ -274,14 +290,96 @@ export class ModuleIssues extends IssueHelperStore implements IModuleIssues {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
removeIssuesFromModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueIds: string[]) => {
|
||||||
|
try {
|
||||||
|
runInAction(() => {
|
||||||
|
issueIds.forEach((issueId) => {
|
||||||
|
pull(this.issues[moduleId], issueId);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
runInAction(() => {
|
||||||
|
issueIds.forEach((issueId) => {
|
||||||
|
update(this.rootStore.issues.issuesMap, [issueId, "module_ids"], (issueModuleIds = []) => {
|
||||||
|
if (issueModuleIds.includes(moduleId)) return pull(issueModuleIds, moduleId);
|
||||||
|
else return uniq(concat(issueModuleIds, [moduleId]));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const response = await this.moduleService.removeIssuesFromModuleBulk(
|
||||||
|
workspaceSlug,
|
||||||
|
projectId,
|
||||||
|
moduleId,
|
||||||
|
issueIds
|
||||||
|
);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
addModulesToIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => {
|
||||||
|
try {
|
||||||
|
const issueToModule = await this.moduleService.addModulesToIssue(workspaceSlug, projectId, issueId, {
|
||||||
|
modules: moduleIds,
|
||||||
|
});
|
||||||
|
|
||||||
|
runInAction(() => {
|
||||||
|
moduleIds.forEach((moduleId) => {
|
||||||
|
update(this.issues, moduleId, (moduleIssueIds = []) => {
|
||||||
|
if (moduleIssueIds.includes(issueId)) return moduleIssueIds;
|
||||||
|
else return uniq(concat(moduleIssueIds, [issueId]));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
update(this.rootStore.issues.issuesMap, [issueId, "module_ids"], (issueModuleIds = []) =>
|
||||||
|
uniq(concat(issueModuleIds, moduleIds))
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
return issueToModule;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
removeModulesFromIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => {
|
||||||
|
try {
|
||||||
|
runInAction(() => {
|
||||||
|
moduleIds.forEach((moduleId) => {
|
||||||
|
update(this.issues, moduleId, (moduleIssueIds = []) => {
|
||||||
|
if (moduleIssueIds.includes(issueId)) return moduleIssueIds;
|
||||||
|
else return uniq(concat(moduleIssueIds, [issueId]));
|
||||||
|
});
|
||||||
|
update(this.rootStore.issues.issuesMap, [issueId, "module_ids"], (issueModuleIds = []) =>
|
||||||
|
pull(issueModuleIds, moduleId)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const response = await this.moduleService.removeModulesFromIssueBulk(
|
||||||
|
workspaceSlug,
|
||||||
|
projectId,
|
||||||
|
issueId,
|
||||||
|
moduleIds
|
||||||
|
);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
removeIssueFromModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => {
|
removeIssueFromModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => {
|
||||||
try {
|
try {
|
||||||
runInAction(() => {
|
runInAction(() => {
|
||||||
pull(this.issues[moduleId], issueId);
|
pull(this.issues[moduleId], issueId);
|
||||||
|
update(this.rootStore.issues.issuesMap, [issueId, "module_ids"], (issueModuleIds = []) =>
|
||||||
|
pull(issueModuleIds, moduleId)
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.rootStore.issues.updateIssue(issueId, { module_id: null });
|
|
||||||
|
|
||||||
const response = await this.moduleService.removeIssueFromModule(workspaceSlug, projectId, moduleId, issueId);
|
const response = await this.moduleService.removeIssueFromModule(workspaceSlug, projectId, moduleId, issueId);
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
|
Loading…
Reference in New Issue
Block a user