mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-11-03 20:48:07 +00:00 
			
		
		
		
	# Pull Request Template ## Description **This PR includes:** 1. Adds the ability to delete a contact from the contacts list accordion section. 2. Improves the expand/collapse transition for the accordion. Fixes [CW-4375](https://linear.app/chatwoot/issue/CW-4375/allow-users-to-delete-a-contact-from-the-contacts-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/8c897d24737f40f6b8b29fef76ba18e2?sid=70910b9d-f3db-4d54-8bfc-820db680e537 ## Checklist: - [x] My code follows the style guidelines of this project - [x] I have performed a self-review of my code - [ ] 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 - [ ] 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: Pranav <pranavrajs@gmail.com>
		
			
				
	
	
		
			248 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			248 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<script setup>
 | 
						|
import { computed, useSlots, useAttrs } from 'vue';
 | 
						|
 | 
						|
import Spinner from 'dashboard/components-next/spinner/Spinner.vue';
 | 
						|
import Icon from 'dashboard/components-next/icon/Icon.vue';
 | 
						|
import {
 | 
						|
  VARIANT_OPTIONS,
 | 
						|
  COLOR_OPTIONS,
 | 
						|
  JUSTIFY_OPTIONS,
 | 
						|
  SIZE_OPTIONS,
 | 
						|
  EXCLUDED_ATTRS,
 | 
						|
} from './constants.js';
 | 
						|
 | 
						|
const props = defineProps({
 | 
						|
  label: { type: [String, Number], default: '' },
 | 
						|
  variant: {
 | 
						|
    type: String,
 | 
						|
    default: null,
 | 
						|
    validator: value => VARIANT_OPTIONS.includes(value) || value === null,
 | 
						|
  },
 | 
						|
  color: {
 | 
						|
    type: String,
 | 
						|
    default: null,
 | 
						|
    validator: value => COLOR_OPTIONS.includes(value) || value === null,
 | 
						|
  },
 | 
						|
  size: {
 | 
						|
    type: String,
 | 
						|
    default: null,
 | 
						|
    validator: value => SIZE_OPTIONS.includes(value) || value === null,
 | 
						|
  },
 | 
						|
  justify: {
 | 
						|
    type: String,
 | 
						|
    default: null,
 | 
						|
    validator: value => JUSTIFY_OPTIONS.includes(value) || value === null,
 | 
						|
  },
 | 
						|
  icon: { type: [String, Object, Function], default: '' },
 | 
						|
  trailingIcon: { type: Boolean, default: false },
 | 
						|
  isLoading: { type: Boolean, default: false },
 | 
						|
});
 | 
						|
 | 
						|
const slots = useSlots();
 | 
						|
const attrs = useAttrs();
 | 
						|
 | 
						|
defineOptions({
 | 
						|
  inheritAttrs: false,
 | 
						|
});
 | 
						|
 | 
						|
const filteredAttrs = computed(() => {
 | 
						|
  const standardAttrs = {};
 | 
						|
 | 
						|
  Object.entries(attrs)
 | 
						|
    .filter(([key]) => !EXCLUDED_ATTRS.includes(key))
 | 
						|
    .forEach(([key, value]) => {
 | 
						|
      standardAttrs[key] = value;
 | 
						|
    });
 | 
						|
 | 
						|
  return standardAttrs;
 | 
						|
});
 | 
						|
 | 
						|
const computedVariant = computed(() => {
 | 
						|
  if (props.variant) return props.variant;
 | 
						|
  // The useAttrs method returns attributes values an empty string (not boolean value as in props).
 | 
						|
  if (attrs.solid || attrs.solid === '') return 'solid';
 | 
						|
  if (attrs.outline || attrs.outline === '') return 'outline';
 | 
						|
  if (attrs.faded || attrs.faded === '') return 'faded';
 | 
						|
  if (attrs.link || attrs.link === '') return 'link';
 | 
						|
  if (attrs.ghost || attrs.ghost === '') return 'ghost';
 | 
						|
  return 'solid'; // Default variant
 | 
						|
});
 | 
						|
 | 
						|
const computedColor = computed(() => {
 | 
						|
  if (props.color) return props.color;
 | 
						|
  if (attrs.blue || attrs.blue === '') return 'blue';
 | 
						|
  if (attrs.ruby || attrs.ruby === '') return 'ruby';
 | 
						|
  if (attrs.amber || attrs.amber === '') return 'amber';
 | 
						|
  if (attrs.slate || attrs.slate === '') return 'slate';
 | 
						|
  if (attrs.teal || attrs.teal === '') return 'teal';
 | 
						|
  return 'blue'; // Default color
 | 
						|
});
 | 
						|
 | 
						|
const computedSize = computed(() => {
 | 
						|
  if (props.size) return props.size;
 | 
						|
  if (attrs.xs || attrs.xs === '') return 'xs';
 | 
						|
  if (attrs.sm || attrs.sm === '') return 'sm';
 | 
						|
  if (attrs.md || attrs.md === '') return 'md';
 | 
						|
  if (attrs.lg || attrs.lg === '') return 'lg';
 | 
						|
  return 'md';
 | 
						|
});
 | 
						|
 | 
						|
const computedJustify = computed(() => {
 | 
						|
  if (props.justify) return props.justify;
 | 
						|
  if (attrs.start || attrs.start === '') return 'start';
 | 
						|
  if (attrs.center || attrs.center === '') return 'center';
 | 
						|
  if (attrs.end || attrs.end === '') return 'end';
 | 
						|
 | 
						|
  return 'center';
 | 
						|
});
 | 
						|
 | 
						|
const STYLE_CONFIG = {
 | 
						|
  colors: {
 | 
						|
    blue: {
 | 
						|
      solid:
 | 
						|
        'bg-n-brand text-white hover:enabled:brightness-110 focus-visible:brightness-110 outline-transparent',
 | 
						|
      faded:
 | 
						|
        'bg-n-brand/10 text-n-blue-text hover:enabled:bg-n-brand/20 focus-visible:bg-n-brand/20 outline-transparent',
 | 
						|
      outline: 'text-n-blue-text outline-n-brand',
 | 
						|
      ghost:
 | 
						|
        'text-n-blue-text hover:enabled:bg-n-alpha-2 focus-visible:bg-n-alpha-2 outline-transparent',
 | 
						|
      link: 'text-n-blue-text hover:enabled:underline focus-visible:underline outline-transparent',
 | 
						|
    },
 | 
						|
    ruby: {
 | 
						|
      solid:
 | 
						|
        'bg-n-ruby-9 text-white hover:enabled:bg-n-ruby-10 focus-visible:bg-n-ruby-10 outline-transparent',
 | 
						|
      faded:
 | 
						|
        'bg-n-ruby-9/10 text-n-ruby-11 hover:enabled:bg-n-ruby-9/20 focus-visible:bg-n-ruby-9/20 outline-transparent',
 | 
						|
      outline:
 | 
						|
        'text-n-ruby-11 hover:enabled:bg-n-ruby-9/10 focus-visible:bg-n-ruby-9/10 outline-n-ruby-8',
 | 
						|
      ghost:
 | 
						|
        'text-n-ruby-11 hover:enabled:bg-n-alpha-2 focus-visible:bg-n-alpha-2 outline-transparent',
 | 
						|
      link: 'text-n-ruby-9 dark:text-n-ruby-11 hover:enabled:underline focus-visible:underline outline-transparent',
 | 
						|
    },
 | 
						|
    amber: {
 | 
						|
      solid:
 | 
						|
        'bg-n-amber-9 text-white hover:enabled:bg-n-amber-10 focus-visible:bg-n-amber-10 outline-transparent',
 | 
						|
      faded:
 | 
						|
        'bg-n-amber-9/10 text-n-slate-12 hover:enabled:bg-n-amber-9/20 focus-visible:bg-n-amber-9/20 outline-transparent',
 | 
						|
      outline:
 | 
						|
        'text-n-amber-11 hover:enabled:bg-n-amber-9/10 focus-visible:bg-n-amber-9/10 outline-n-amber-9',
 | 
						|
      link: 'text-n-amber-9 hover:enabled:underline focus-visible:underline outline-transparent',
 | 
						|
      ghost:
 | 
						|
        'text-n-amber-9 hover:enabled:bg-n-alpha-2 focus-visible:bg-n-alpha-2 outline-transparent',
 | 
						|
    },
 | 
						|
    slate: {
 | 
						|
      solid:
 | 
						|
        'bg-n-solid-3 dark:hover:enabled:bg-n-solid-2 dark:focus-visible:bg-n-solid-2 hover:enabled:bg-n-alpha-2 focus-visible:bg-n-alpha-2 text-n-slate-12 outline-n-container',
 | 
						|
      faded:
 | 
						|
        'bg-n-slate-9/10 text-n-slate-12 hover:enabled:bg-n-slate-9/20 focus-visible:bg-n-slate-9/20 outline-transparent',
 | 
						|
      outline:
 | 
						|
        'text-n-slate-11 outline-n-strong hover:enabled:bg-n-slate-9/10 focus-visible:bg-n-slate-9/10',
 | 
						|
      link: 'text-n-slate-11 hover:enabled:text-n-slate-12 focus-visible:text-n-slate-12 hover:enabled:underline focus-visible:underline outline-transparent',
 | 
						|
      ghost:
 | 
						|
        'text-n-slate-12 hover:enabled:bg-n-alpha-2 focus-visible:bg-n-alpha-2 outline-transparent',
 | 
						|
    },
 | 
						|
    teal: {
 | 
						|
      solid:
 | 
						|
        'bg-n-teal-9 text-white hover:enabled:bg-n-teal-10 focus-visible:bg-n-teal-10 outline-transparent',
 | 
						|
      faded:
 | 
						|
        'bg-n-teal-9/10 text-n-slate-12 hover:enabled:bg-n-teal-9/20 focus-visible:bg-n-teal-9/20 outline-transparent',
 | 
						|
      outline:
 | 
						|
        'text-n-teal-11 hover:enabled:bg-n-teal-9/10 focus-visible:bg-n-teal-9/10 outline-n-teal-9',
 | 
						|
      link: 'text-n-teal-9 hover:enabled:underline focus-visible:underline outline-transparent',
 | 
						|
      ghost:
 | 
						|
        'text-n-teal-9 hover:enabled:bg-n-alpha-2 focus-visible:bg-n-alpha-2 outline-transparent',
 | 
						|
    },
 | 
						|
  },
 | 
						|
  sizes: {
 | 
						|
    regular: {
 | 
						|
      xs: 'h-6 px-2',
 | 
						|
      sm: 'h-8 px-3',
 | 
						|
      md: 'h-10 px-4',
 | 
						|
      lg: 'h-12 px-5',
 | 
						|
    },
 | 
						|
    iconOnly: {
 | 
						|
      xs: 'h-6 w-6 p-0',
 | 
						|
      sm: 'h-8 w-8 p-0',
 | 
						|
      md: 'h-10 w-10 p-0',
 | 
						|
      lg: 'h-12 w-12 p-0',
 | 
						|
    },
 | 
						|
    link: {
 | 
						|
      xs: 'p-0',
 | 
						|
      sm: 'p-0',
 | 
						|
      md: 'p-0',
 | 
						|
      lg: 'p-0',
 | 
						|
    },
 | 
						|
  },
 | 
						|
  fontSize: {
 | 
						|
    xs: 'text-xs',
 | 
						|
    sm: 'text-sm',
 | 
						|
    md: 'text-sm font-medium',
 | 
						|
    lg: 'text-base',
 | 
						|
  },
 | 
						|
  justify: {
 | 
						|
    start: 'justify-start',
 | 
						|
    center: 'justify-center',
 | 
						|
    end: 'justify-end',
 | 
						|
  },
 | 
						|
  base: 'inline-flex items-center min-w-0 gap-2 transition-all duration-200 ease-in-out border-0 rounded-lg outline-1 outline disabled:opacity-50',
 | 
						|
};
 | 
						|
 | 
						|
const variantClasses = computed(() => {
 | 
						|
  const variantMap = {
 | 
						|
    ghost: `${STYLE_CONFIG.colors[computedColor.value].ghost}`,
 | 
						|
    link: `${STYLE_CONFIG.colors[computedColor.value].link} p-0 font-medium underline-offset-2`,
 | 
						|
    outline: STYLE_CONFIG.colors[computedColor.value].outline,
 | 
						|
    faded: STYLE_CONFIG.colors[computedColor.value].faded,
 | 
						|
    solid: STYLE_CONFIG.colors[computedColor.value].solid,
 | 
						|
  };
 | 
						|
 | 
						|
  return variantMap[computedVariant.value];
 | 
						|
});
 | 
						|
 | 
						|
const isIconOnly = computed(() => !props.label && !slots.default);
 | 
						|
const isLink = computed(() => computedVariant.value === 'link');
 | 
						|
 | 
						|
const buttonClasses = computed(() => {
 | 
						|
  const sizeConfig = isIconOnly.value ? 'iconOnly' : 'regular';
 | 
						|
  const classes = [
 | 
						|
    variantClasses.value,
 | 
						|
    computedVariant.value !== 'link' &&
 | 
						|
      STYLE_CONFIG.sizes[sizeConfig][computedSize.value],
 | 
						|
  ].filter(Boolean);
 | 
						|
 | 
						|
  return classes.join(' ');
 | 
						|
});
 | 
						|
 | 
						|
const linkButtonClasses = computed(() => {
 | 
						|
  const classes = [
 | 
						|
    variantClasses.value,
 | 
						|
    STYLE_CONFIG.sizes.link[computedSize.value],
 | 
						|
  ].filter(Boolean);
 | 
						|
 | 
						|
  return classes.join(' ');
 | 
						|
});
 | 
						|
</script>
 | 
						|
 | 
						|
<template>
 | 
						|
  <button
 | 
						|
    v-bind="filteredAttrs"
 | 
						|
    :class="{
 | 
						|
      [STYLE_CONFIG.base]: true,
 | 
						|
      [isLink ? linkButtonClasses : buttonClasses]: true,
 | 
						|
      [STYLE_CONFIG.fontSize[computedSize]]: true,
 | 
						|
      [STYLE_CONFIG.justify[computedJustify]]: true,
 | 
						|
      'flex-row-reverse': trailingIcon && !isIconOnly,
 | 
						|
    }"
 | 
						|
  >
 | 
						|
    <slot v-if="(icon || $slots.icon) && !isLoading" name="icon">
 | 
						|
      <Icon :icon="icon" class="flex-shrink-0" />
 | 
						|
    </slot>
 | 
						|
 | 
						|
    <Spinner v-if="isLoading" class="!w-5 !h-5 flex-shrink-0" />
 | 
						|
 | 
						|
    <slot v-if="label || $slots.default" name="default">
 | 
						|
      <span v-if="label" class="min-w-0 truncate">{{ label }}</span>
 | 
						|
    </slot>
 | 
						|
  </button>
 | 
						|
</template>
 |