mirror of
https://github.com/lingble/chatwoot.git
synced 2025-11-01 19:48:08 +00:00
# Pull Request Template ## Description Fixes https://linear.app/chatwoot/issue/CW-5573/feat-createedit-agent-capacity-policy-page ## Type of change - [x] New feature (non-breaking change which adds functionality) ## How Has This Been Tested? ### Loom video https://www.loom.com/share/8de9e3c5d8824cd998d242636540dd18?sid=1314536f-c8d6-41fd-8139-cae9bf94f942 ### Screenshots **Light mode** <img width="1666" height="1225" alt="image" src="https://github.com/user-attachments/assets/7e6d83a4-ce02-47a7-91f6-87745f8f5549" /> <img width="1666" height="1225" alt="image" src="https://github.com/user-attachments/assets/7dd1f840-2e25-4365-aa1d-ed9dac13385a" /> **Dark mode** <img width="1666" height="1225" alt="image" src="https://github.com/user-attachments/assets/0c787095-7146-4fb3-a61a-e2232973bcba" /> <img width="1666" height="1225" alt="image" src="https://github.com/user-attachments/assets/481c21fd-03b5-4c1f-b59e-7f8c8017f9ce" /> ## Checklist: - [x] My code follows the style guidelines of this project - [x] I have performed a self-review of my code - [x] I have commented on my code, particularly in hard-to-understand areas - [ ] I have made corresponding changes to the documentation - [x] My changes generate no new warnings - [x] I have added tests that prove my fix is effective or that my feature works - [x] New and existing unit tests pass locally with my changes - [ ] Any dependent changes have been merged and published in downstream modules --------- Co-authored-by: Muhsin Keloth <muhsinkeramam@gmail.com>
137 lines
3.6 KiB
Vue
137 lines
3.6 KiB
Vue
<script setup>
|
|
import { computed, watch, onMounted, ref } from 'vue';
|
|
import { useRoute } from 'vue-router';
|
|
import { useMapGetter, useStore } from 'dashboard/composables/store';
|
|
|
|
import LabelItem from 'dashboard/components-next/Label/LabelItem.vue';
|
|
import AddLabel from 'dashboard/components-next/Label/AddLabel.vue';
|
|
|
|
const props = defineProps({
|
|
contactId: {
|
|
type: [String, Number],
|
|
default: null,
|
|
},
|
|
});
|
|
|
|
const store = useStore();
|
|
const route = useRoute();
|
|
|
|
const showDropdown = ref(false);
|
|
|
|
// Store the currently hovered label's ID
|
|
// Using JS state management instead of CSS :hover / group hover
|
|
// This will solve the flickering issue when hovering over the last label item
|
|
const hoveredLabel = ref(null);
|
|
|
|
const allLabels = useMapGetter('labels/getLabels');
|
|
const contactLabels = useMapGetter('contactLabels/getContactLabels');
|
|
|
|
const savedLabels = computed(() => {
|
|
const availableContactLabels = contactLabels.value(props.contactId);
|
|
return allLabels.value.filter(({ title }) =>
|
|
availableContactLabels.includes(title)
|
|
);
|
|
});
|
|
|
|
const labelMenuItems = computed(() => {
|
|
return allLabels.value
|
|
?.map(label => ({
|
|
label: label.title,
|
|
value: label.id,
|
|
thumbnail: { name: label.title, color: label.color },
|
|
isSelected: savedLabels.value.some(
|
|
savedLabel => savedLabel.id === label.id
|
|
),
|
|
action: 'contactLabel',
|
|
}))
|
|
.toSorted((a, b) => Number(a.isSelected) - Number(b.isSelected));
|
|
});
|
|
|
|
const fetchLabels = async contactId => {
|
|
if (!contactId) {
|
|
return;
|
|
}
|
|
store.dispatch('contactLabels/get', contactId);
|
|
};
|
|
|
|
const handleLabelAction = async ({ value }) => {
|
|
try {
|
|
// Get current label titles
|
|
const currentLabels = savedLabels.value.map(label => label.title);
|
|
|
|
// Find the label title for the ID (value)
|
|
const selectedLabel = allLabels.value.find(label => label.id === value);
|
|
if (!selectedLabel) return;
|
|
|
|
let updatedLabels;
|
|
|
|
// If label is already selected, remove it (toggle behavior)
|
|
if (currentLabels.includes(selectedLabel.title)) {
|
|
updatedLabels = currentLabels.filter(
|
|
labelTitle => labelTitle !== selectedLabel.title
|
|
);
|
|
} else {
|
|
// Add the new label
|
|
updatedLabels = [...currentLabels, selectedLabel.title];
|
|
}
|
|
|
|
await store.dispatch('contactLabels/update', {
|
|
contactId: props.contactId,
|
|
labels: updatedLabels,
|
|
});
|
|
|
|
showDropdown.value = false;
|
|
} catch (error) {
|
|
// error
|
|
}
|
|
};
|
|
|
|
const handleRemoveLabel = label => {
|
|
return handleLabelAction({ value: label.id });
|
|
};
|
|
|
|
watch(
|
|
() => props.contactId,
|
|
(newVal, oldVal) => {
|
|
if (newVal !== oldVal) {
|
|
fetchLabels(newVal);
|
|
}
|
|
}
|
|
);
|
|
onMounted(() => {
|
|
if (route.params.contactId) {
|
|
fetchLabels(route.params.contactId);
|
|
}
|
|
});
|
|
|
|
const handleMouseLeave = () => {
|
|
// Reset hover state when mouse leaves the container
|
|
// This ensures all labels return to their default state
|
|
hoveredLabel.value = null;
|
|
};
|
|
|
|
const handleLabelHover = labelId => {
|
|
// Added this to prevent flickering on when showing remove button on hover
|
|
// If the label item is at end of the line, it will show the remove button
|
|
// when hovering over the last label item
|
|
hoveredLabel.value = labelId;
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div class="flex flex-wrap items-center gap-2" @mouseleave="handleMouseLeave">
|
|
<LabelItem
|
|
v-for="label in savedLabels"
|
|
:key="label.id"
|
|
:label="label"
|
|
:is-hovered="hoveredLabel === label.id"
|
|
@remove="handleRemoveLabel"
|
|
@hover="handleLabelHover(label.id)"
|
|
/>
|
|
<AddLabel
|
|
:label-menu-items="labelMenuItems"
|
|
@update-label="handleLabelAction"
|
|
/>
|
|
</div>
|
|
</template>
|