mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-11-03 20:48:07 +00:00 
			
		
		
		
	### Old behavior https://github.com/user-attachments/assets/5d7524c1-b1ca-454f-8e89-843ca25d017e ### New behavior https://github.com/user-attachments/assets/49814190-d6b4-4553-bced-3987be3583e2 --------- Co-authored-by: Sivin Varghese <64252451+iamsivin@users.noreply.github.com> Co-authored-by: iamsivin <iamsivin@gmail.com>
		
			
				
	
	
		
			229 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			229 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<script>
 | 
						|
/**
 | 
						|
 * Thumbnail Component
 | 
						|
 * Src - source for round image
 | 
						|
 * Size - Size of the thumbnail
 | 
						|
 * Badge - Chat source indication { fb / telegram }
 | 
						|
 * Username - Username for avatar
 | 
						|
 */
 | 
						|
import Avatar from './Avatar.vue';
 | 
						|
import { removeEmoji } from 'shared/helpers/emoji';
 | 
						|
 | 
						|
export default {
 | 
						|
  components: {
 | 
						|
    Avatar,
 | 
						|
  },
 | 
						|
  props: {
 | 
						|
    src: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    size: {
 | 
						|
      type: String,
 | 
						|
      default: '40px',
 | 
						|
    },
 | 
						|
    badge: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    username: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    status: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    hasBorder: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
    shouldShowStatusAlways: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
    title: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    variant: {
 | 
						|
      type: String,
 | 
						|
      default: 'circle',
 | 
						|
    },
 | 
						|
  },
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      hasImageLoaded: false,
 | 
						|
      imgError: false,
 | 
						|
    };
 | 
						|
  },
 | 
						|
  computed: {
 | 
						|
    userNameWithoutEmoji() {
 | 
						|
      return removeEmoji(this.username);
 | 
						|
    },
 | 
						|
    showStatusIndicator() {
 | 
						|
      if (this.shouldShowStatusAlways) return true;
 | 
						|
      return this.status === 'online' || this.status === 'busy';
 | 
						|
    },
 | 
						|
    avatarSize() {
 | 
						|
      return Number(this.size.replace(/\D+/g, ''));
 | 
						|
    },
 | 
						|
    badgeSrc() {
 | 
						|
      return {
 | 
						|
        instagram_direct_message: 'instagram-dm',
 | 
						|
        facebook: 'messenger',
 | 
						|
        'twitter-tweet': 'twitter-tweet',
 | 
						|
        'twitter-dm': 'twitter-dm',
 | 
						|
        whatsapp: 'whatsapp',
 | 
						|
        sms: 'sms',
 | 
						|
        'Channel::Line': 'line',
 | 
						|
        'Channel::Telegram': 'telegram',
 | 
						|
        'Channel::WebWidget': '',
 | 
						|
      }[this.badge];
 | 
						|
    },
 | 
						|
    badgeStyle() {
 | 
						|
      const size = Math.floor(this.avatarSize / 3);
 | 
						|
      const badgeSize = `${size + 2}px`;
 | 
						|
      const borderRadius = `${size / 2}px`;
 | 
						|
      return { width: badgeSize, height: badgeSize, borderRadius };
 | 
						|
    },
 | 
						|
    statusStyle() {
 | 
						|
      const statusSize = `${this.avatarSize / 4}px`;
 | 
						|
      return { width: statusSize, height: statusSize };
 | 
						|
    },
 | 
						|
    thumbnailClass() {
 | 
						|
      const className = this.hasBorder
 | 
						|
        ? 'border border-solid border-white dark:border-slate-700/50'
 | 
						|
        : '';
 | 
						|
      const variant =
 | 
						|
        this.variant === 'circle' ? 'thumbnail-rounded' : 'thumbnail-square';
 | 
						|
      return `user-thumbnail ${className} ${variant}`;
 | 
						|
    },
 | 
						|
    thumbnailBoxClass() {
 | 
						|
      const boxClass = this.variant === 'circle' ? 'is-rounded' : '';
 | 
						|
      return `user-thumbnail-box ${boxClass}`;
 | 
						|
    },
 | 
						|
    shouldShowImage() {
 | 
						|
      if (!this.src) {
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
      if (this.hasImageLoaded) {
 | 
						|
        return !this.imgError;
 | 
						|
      }
 | 
						|
      return false;
 | 
						|
    },
 | 
						|
  },
 | 
						|
  watch: {
 | 
						|
    src(value, oldValue) {
 | 
						|
      if (value !== oldValue && this.imgError) {
 | 
						|
        this.imgError = false;
 | 
						|
      }
 | 
						|
    },
 | 
						|
  },
 | 
						|
  methods: {
 | 
						|
    onImgError() {
 | 
						|
      this.imgError = true;
 | 
						|
    },
 | 
						|
    onImgLoad() {
 | 
						|
      this.hasImageLoaded = true;
 | 
						|
    },
 | 
						|
  },
 | 
						|
};
 | 
						|
</script>
 | 
						|
 | 
						|
<template>
 | 
						|
  <div
 | 
						|
    :class="thumbnailBoxClass"
 | 
						|
    :style="{ height: size, width: size }"
 | 
						|
    :title="title"
 | 
						|
  >
 | 
						|
    <!-- Using v-show instead of v-if to avoid flickering as v-if removes dom elements.  -->
 | 
						|
    <slot>
 | 
						|
      <img
 | 
						|
        v-show="shouldShowImage"
 | 
						|
        :src="src"
 | 
						|
        draggable="false"
 | 
						|
        :class="thumbnailClass"
 | 
						|
        @load="onImgLoad"
 | 
						|
        @error="onImgError"
 | 
						|
      />
 | 
						|
      <Avatar
 | 
						|
        v-show="!shouldShowImage"
 | 
						|
        :username="userNameWithoutEmoji"
 | 
						|
        :class="thumbnailClass"
 | 
						|
        :size="avatarSize"
 | 
						|
      />
 | 
						|
    </slot>
 | 
						|
    <img
 | 
						|
      v-if="badgeSrc"
 | 
						|
      class="source-badge z-20"
 | 
						|
      :style="badgeStyle"
 | 
						|
      :src="`/integrations/channels/badges/${badgeSrc}.png`"
 | 
						|
      alt="Badge"
 | 
						|
    />
 | 
						|
    <div
 | 
						|
      v-if="showStatusIndicator"
 | 
						|
      class="z-20"
 | 
						|
      :class="`source-badge user-online-status user-online-status--${status}`"
 | 
						|
      :style="statusStyle"
 | 
						|
    />
 | 
						|
  </div>
 | 
						|
</template>
 | 
						|
 | 
						|
<style lang="scss" scoped>
 | 
						|
.user-thumbnail-box {
 | 
						|
  flex: 0 0 auto;
 | 
						|
  max-width: 100%;
 | 
						|
  position: relative;
 | 
						|
 | 
						|
  &.is-rounded {
 | 
						|
    border-radius: 50%;
 | 
						|
  }
 | 
						|
 | 
						|
  .user-thumbnail {
 | 
						|
    border-radius: 50%;
 | 
						|
    &.thumbnail-square {
 | 
						|
      border-radius: var(--border-radius-large);
 | 
						|
    }
 | 
						|
    height: 100%;
 | 
						|
    width: 100%;
 | 
						|
    box-sizing: border-box;
 | 
						|
    object-fit: cover;
 | 
						|
    vertical-align: initial;
 | 
						|
  }
 | 
						|
 | 
						|
  .source-badge {
 | 
						|
    border-radius: var(--border-radius-small);
 | 
						|
    bottom: var(--space-minus-micro);
 | 
						|
    box-shadow: var(--shadow-small);
 | 
						|
    height: var(--space-slab);
 | 
						|
    padding: var(--space-micro);
 | 
						|
    position: absolute;
 | 
						|
    right: 0;
 | 
						|
    width: var(--space-slab);
 | 
						|
    @apply bg-white dark:bg-slate-900;
 | 
						|
  }
 | 
						|
 | 
						|
  .user-online-status {
 | 
						|
    border-radius: 50%;
 | 
						|
    bottom: var(--space-micro);
 | 
						|
 | 
						|
    &:after {
 | 
						|
      content: ' ';
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  .user-online-status--online {
 | 
						|
    @apply bg-green-400 dark:bg-green-400;
 | 
						|
  }
 | 
						|
 | 
						|
  .user-online-status--busy {
 | 
						|
    @apply bg-yellow-500 dark:bg-yellow-500;
 | 
						|
  }
 | 
						|
 | 
						|
  .user-online-status--offline {
 | 
						|
    @apply bg-slate-500 dark:bg-slate-500;
 | 
						|
  }
 | 
						|
}
 | 
						|
</style>
 |