# vue-advanced-chat **Repository Path**: mirrors_antoine92190/vue-advanced-chat ## Basic Information - **Project Name**: vue-advanced-chat - **Description**: No description available - **Primary Language**: Unknown - **License**: MIT - **Default Branch**: main - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 1 - **Forks**: 1 - **Created**: 2022-01-06 - **Last Updated**: 2024-01-19 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README

# vue-advanced-chat ![Demo Image](demo/src/assets/web_mobile.png) ## Sponsors
ChatKitty sponsor
Easy to use chat API/Server with scalable infrastructure
## Features - Compatible with all Javascript frameworks (Vue, Angular, React, etc.) or no framework at all - Customizeable realtime chat messaging - Backend agnostic - Images, videos, files, voice messages & emojis - Edit messages & reply to other messages - Tag users & emojis shortcut suggestions - UI elements for seen, new, deleted, typing and system messages - Text formatting - bold, italic, strikethrough, underline, code, multiline - Online / Offline users status - Flexible options and slots - Light and dark theme modes - Firestore example - Typescript, PWA, Web Component support ## [Demo](https://advanced-chat.github.io/vue-advanced-chat) Enjoy :smile: ## [Real World Example](https://vue-advanced-chat-app.netlify.app/) A Progressive Web Application showcasing all the features of `vue-advanced-chat` component.
Built with Firestore, Vuetify, and Push Notifications. If you wish to get premium access to the real world example source code, please contact me by email. You will get a fully working chat application for web and mobile: - UI and backend integration - Email, Facebook and Google authentication - Real-time messaging, browser push notifications, images optimization (Firebase Cloud Functions to compress avatars) - UI/UX components for alerts (errors, information), dialogs, etc. - Add existing users to a room using their email - Send email invitations to non-existing users - Edition of profile and rooms - Addition and deletion of room users - Optimised firestore implementation to reduce bandwidth and costs as much as possible - State management using vuex - Internationalisation (i18n) - Google Analytics - Support to help you get the chat up and running
## Table of Contents - [Installation](#installation) - [Usage](#example) - [Props API](#props-api) - [Props data structure](#props-data-structure) - [Events API](#events-api) - [Named Slots](#named-slots) - [Using with Firestore](#using-with-firestore)
## Installation ```bash # Using npm npm install --save vue-advanced-chat # Using yarn yarn add vue-advanced-chat # Using CDN ``` ### Vue Register `vue-advanced-chat` and `emoji-picker` as web components in your [config file](https://vuejs.org/guide/extras/web-components.html#example-vite-config): ```javascript compilerOptions: { isCustomElement: tagName => { return tagName === 'vue-advanced-chat' || tagName === 'emoji-picker' } } ``` Demo: https://github.com/advanced-chat/vue-advanced-chat-sandbox/tree/main ### React Demo: https://github.com/advanced-chat/vue-advanced-chat-sandbox/tree/react ### Angular / Ionic Demo: https://github.com/advanced-chat/vue-advanced-chat-sandbox/tree/angular
## Usage #### You can import it as a custom component: ```javascript ``` ### Important notes `vue-advanced-chat` component is performance oriented, hence you have to follow specific rules to make it work properly. - Use array assignement instead of `push` method ```javascript // DO THIS const rooms = [] for (let i = 0; i < res.length; i++) { rooms.push(res) } this.rooms = rooms // DON'T DO THIS for (let i = 0; i < res.length; i++) { this.rooms.push(res) } ``` ```javascript // DO THIS this.rooms[i].typingUsers = [...this.rooms[i].typingUsers, typingUserId] // DON'T DO THIS this.rooms[i].typingUsers.push(typingUserId) ``` - To add or replace an item inside an array, use spread operator ```javascript // DO THIS this.rooms[roomIndex] = room this.rooms = [...this.rooms] // DON'T DO THIS this.rooms[roomIndex] = room // AND DON'T DO THIS this.rooms.push(room) ``` - Follow the UI loading pattern by updating `messagesLoaded` prop every time a new room is fetched ```javascript fetchMessages({ room, options }) { this.messagesLoaded = false // use timeout to imitate async server fetched data setTimeout(() => { this.messages = [] this.messagesLoaded = true }) } ```
## Props API ### Important notes If you are using Vue 3, you can pass Array and Object props normally: [Passing DOM Properties in Vue 3](https://vuejs.org/guide/extras/web-components.html#passing-dom-properties)
Otherwise, you need to pass those props as strings. For example: `[messages]="JSON.stringify(messages)"` |
Prop
| Type | Required | Default | | ----------------------------------- | ---------------- | -------- | ----------------------------------------------------------------------------------------------------------------- | | `height` | String | - | `600px` | | `current-user-id`(1) | String | `true` | - | | `rooms` | [String, Array] | - | `[]` | | `rooms-order` | `desc` / `asc` | - | `desc` | | `loading-rooms`(2) | Boolean | - | `false` | | `rooms-loaded`(3) | Boolean | - | `false` | | `room-id`(4) | String | - | `null` | | `load-first-room`(5) | Boolean | - | `true` | | `rooms-list-opened` | Boolean | - | `true` | | `custom-search-room-enabled` (6) | Boolean | - | `false` | | `messages` | [String, Array] | - | `[]` | | `room-message`(7) | String | - | `null` | | `username-options` (8) | [String, Object] | - | `{minUsers: 3, currentUser: false}` | | `messages-loaded`(9) | Boolean | - | `false` | | `room-actions`(10) | [String, Array] | - | `[]` | | `menu-actions`(11) | [String, Array] | - | `[]` | | `message-actions`(12) | [String, Array] | - | (11) | | `message-selection-actions`(13) | [String, Array] | - | (13) | | `templates-text`(14) | [String, Array] | - | `null` | | `auto-scroll`(15) | [String, Object] | - | `{ send: { new: true, newAfterScrollUp: true }, receive: { new: true, newAfterScrollUp: false } }` | | `show-search` | Boolean | - | `true` | | `show-add-room` | Boolean | - | `true` | | `show-send-icon` | Boolean | - | `true` | | `show-files` | Boolean | - | `true` | | `show-audio` | Boolean | - | `true` | | `audio-bit-rate` | Number | - | `128` | | `audio-sample-rate` | Number | - | `44100` | | `show-emojis` | Boolean | - | `true` | | `show-reaction-emojis` | Boolean | - | `true` | | `show-new-messages-divider`(16) | Boolean | - | `true` | | `show-footer`(17) | Boolean | - | `true` | | `text-messages`(18) | [String, Object] | - | `null` | | `text-formatting`(19) | [String, Object] | - | ` {disabled: false, italic: '_', bold: '*', strike: '~', underline: '°', multilineCode: '```', inlineCode: ' `'}` | | `link-options`(20) | [String, Object] | - | `{ disabled: false, target: '_blank', rel: null }` | | `room-info-enabled` (21) | Boolean | - | `false` | | `textarea-action-enabled`(22) | Boolean | - | `false` | | `textarea-auto-focus` | Boolean | - | `true` | | `user-tags-enabled` | Boolean | - | `true` | | `emojis-suggestion-enabled` | Boolean | - | `true` | | `media-preview-enabled` | Boolean | - | `true` | | `responsive-breakpoint`(23) | Number | - | `900` | | `single-room`(24) | Boolean | - | `false` | | `scroll-distance`(25) | Number | - | `60` | | `theme`(26) | `light` / `dark` | - | `light` | | `accepted-files`(27) | String | - | `*` | | `capture-files`(28) | String | - | `''` | | `multiple-files`(29) | Boolean | - | `true` | | `styles`(30) | [String, Object] | - | (26) | | `show-audio`(31) | Boolean | - | `true` | | `emoji-data-source` | String | - | `https://cdn.jsdelivr.net/npm/emoji-picker-element-data@%5E1/en/emojibase/data.json` | **(1)** `current-user-id` is required to display UI and trigger actions according to the user using the chat (ex: messages position on the right, etc.) **(2)** `loading-rooms` can be used to show/hide a spinner icon while rooms are loading **(3)** `rooms-loaded` must be set to `true` when all rooms have been loaded. Meaning the user cannot scroll to load more paginated rooms **(4)** `room-id` can be used to load a specific room at any time **(5)** `load-first-room` can be used to remove the default behaviour of opening the first room at initialization **(6)** `custom-search-room-enabled` can be used to use the [search-room](#events-api) event to call your own method to filter out rooms **(7)** `room-message` can be used to add a default textarea value **(8)** `username-options` can be used to show/hide room messages username according to the minimum number of users `minUsers` inside a room, and if the message user is the current user `currentUser` **(9)** `messages-loaded` must be set to `true` when all messages of a conversation have been loaded. Meaning the user cannot scroll on top to load more paginated messages **(10)** `room-actions` can be used to display your own buttons when clicking the dropdown icon of each room inside the rooms list.
You can then use the [room-action-handler](#events-api) event to call your own action after clicking a button. Ex: ```javascript room-actions="[ { name: 'archiveRoom', title: 'Archive Room' } ]" ``` **(11)** `menu-actions` can be used to display your own buttons when clicking the vertical dots icon inside a room.
You can then use the [menu-action-handler](#events-api) event to call your own action after clicking a button. Ex: ```javascript menu-actions="[ { name: 'inviteUser', title: 'Invite User' }, { name: 'removeUser', title: 'Remove User' }, { name: 'deleteRoom', title: 'Delete Room' } ]" ``` **(12)** `message-actions` can be used to display your own buttons when clicking the dropdown icon inside a message.
You can then use the [message-action-handler](#events-api) event to call your own action after clicking a button. Ex: ```javascript message-actions="[ { name: 'addMessageToFavorite', title: 'Add To Favorite' }, { name: 'shareMessage', title: 'Share Message' } ]" ``` You can use built-in `message-actions` names to trigger specific UI modifications when clicked.
Currently, `replyMessage`, `editMessage` and `deleteMessage` action names are available.
If `messageActions` is not set, it will use the default values below.
If you don't want to display this `messageActions` menu, you can pass it an empty array. ```javascript messageActions="[ { name: 'replyMessage', title: 'Reply' }, { name: 'editMessage', title: 'Edit Message', onlyMe: true }, { name: 'deleteMessage', title: 'Delete Message', onlyMe: true }, { name: 'selectMessages', title: 'Select' } ]" ``` **(13)** `message-selection-actions` is related to the above `selectMessages` message action. You can use it to display custom action buttons in the room header when selecting a message. Ex: ``` messageActions="[ { name: 'deleteMessages', title: 'Delete' }, { name: 'forwardMessages', title: 'Forward' } ]" ``` **(14)** `templates-text` can be used to add autocomplete templates text when typing `/` in the room textarea. Ex: ```javascript templatesText="[ { tag: 'help', text: 'This is the help' }, { tag: 'action', text: 'This is the action' } ]" ``` **(15)** `auto-scroll` can be used to change the auto scroll behaviour when sending/receiving a message. Ex: ```javascript auto-scroll="{ send: { new: true, // will scroll down after sending a message newAfterScrollUp: false // will not scroll down after sending a message when scrolled up }, receive: { new: false, // will not scroll down when receiving a message newAfterScrollUp: true // will scroll down when receiving a message when scrolled up } }" ``` **(16)** `show-new-messages-divider` can be used to show/hide the blue line divider between seen and unseen messages. **(17)** `show-footer` can be used to hide the room footer. For example to prevent users to send any message or media. **(18)** `text-messages` can be used to replace default i18n texts. Ex: ```javascript text-messages="{ ROOMS_EMPTY: 'Aucune conversation', ROOM_EMPTY: 'Aucune conversation sélectionnée', NEW_MESSAGES: 'Nouveaux messages', MESSAGE_DELETED: 'Ce message a été supprimé', MESSAGES_EMPTY: 'Aucun message', CONVERSATION_STARTED: 'La conversation a commencée le :', TYPE_MESSAGE: 'Tapez votre message', SEARCH: 'Rechercher', IS_ONLINE: 'est en ligne', LAST_SEEN: 'dernière connexion ', IS_TYPING: 'est en train de taper...', CANCEL_SELECT_MESSAGE: 'Annuler Sélection' }" ``` **(19)** `text-formatting` can be used to add text formatting. Bold, italic, strikethrough, underline, inline code and multiline code formatting are currently available and can be used in conjonction. - You can disable text formatting by passing the prop `:text-formatting="{disabled: true}"`. - You can change markdown characters, for example: `:text-formatting="{italic: '^'}"` - You can disable a specific markdown character, for example: `:text-formatting="{bold: null}"` | Style | Syntax | Example | Output | | ----------------- | --------------- | -------------------------------------- | -------------------------------------- | | Bold | `* *` | `*This is bold text*` | **This is bold text** | | Italic | `_ _` | `_This text is italicized_` | _This text is italicized_ | | Strikethrough | `~ ~` | `~This was mistaken text~` | ~~This was mistaken text~~ | | Underline | `° °` | `°This text is underlined°` | This text is underlined | | Nested formatting | `* *` and `_ _` | `*This text is _extremely_ important*` | **This text is _extremely_ important** | **Inline Code** Example: \`This is inline code\` Output: `This is inline code` **Multiline Code** Example: \```This is multiline code\``` Output: ```bash This is multiline code ``` **(20)** `link-options` can be used to disable url links in messages, or change urls target. Ex: ```javascript :link-options="{ disabled: true, target: '_self', rel: null }" ``` **(21)** `room-info-enabled` can be used to trigger an event after clicking the room header component.
You can then use the [room-info](#events-api) event to call your own action after clicking the header. **(22)** `textarea-action-enabled` can be used to add an extra icon on the right of the textarea
You can then use the [textarea-action-handler](#events-api) event to call your own action after clicking the icon. **(23)** `responsive-breakpoint` can be used to collapse the rooms list on the left when then viewport size goes below the specified width. **(24)** `single-room` can be used if you never want to show the rooms list on the left. You still need to pass the `rooms` prop as an array with a single element. **(25)** `scroll-distance` can be used to change the number of pixels before `fetchMessages` event is triggered when scrolling up to load more messages, or `fetchMoreRooms` event is triggered when scrolling down to load more rooms. **(26)** `theme` can be used to change the chat theme. Currently, only `light` and `dark` are available. **(27)** `accepted-files` can be used to set specifics file types allowed in chat. By default, all file types are allowed: `"*"`. Example: set `accepted-files="image/png, image/jpeg, application/pdf"` to allow `JPG` `PNG` and `PDF` files only **(28)** `capture-files` can be used to enable direct capturing of photos and videos on mobile browsers, as opposed to just uploading existing photos and videos which are already on the device. See [here](https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/capture) for more information and recognized values. By default, the attribute is omitted and mobile browsers will only offer the gallery to choose photos and videos. Note: this only affects file attachments. Audio messages are always recorded using the device's microphone. **(29)** `multiple-files` can be used to define whether multiple file selections will be accepted. By default this is true. **(30)** `styles` can be used to customize your own theme. You can find the full list [here](src/themes/index.js) **(31)** `show-audio` can be used to enable or disable audio icon ```javascript styles="{ general: { color: '#0a0a0a', colorSpinner: '#333', borderStyle: '1px solid #e1e4e8' }, footer: { background: '#f8f9fa', backgroundReply: 'rgba(0, 0, 0, 0.08)' }, icons: { search: '#9ca6af' } }" ``` ## Props data structure Your props must follow a specific structure to display rooms and messages correctly: ### Rooms prop ```javascript rooms="[ { roomId: '1', roomName: 'Room 1', avatar: 'assets/imgs/people.png', unreadCount: 4, index: 3, lastMessage: { _id: 'xyz', content: 'Last message received', senderId: '1234', username: 'John Doe', timestamp: '10:20', saved: true, distributed: false, seen: false, new: true }, users: [ { _id: '1234', username: 'John Doe', avatar: 'assets/imgs/doe.png', status: { state: 'online', lastChanged: 'today, 14:30' } }, { _id: '4321', username: 'John Snow', avatar: 'assets/imgs/snow.png', status: { state: 'offline', lastChanged: '14 July, 20:00' } } ], typingUsers: [ 4321 ] } ]" ``` - If you add the `index` property, your rooms will be ordered using this value. `index` can be any sortable value, like a `string`, `datetime`, `timestamp`, etc. - For each room user, you can add the `status` property, which can hold the `state` and `lastChanged` properties: - `state` can be `'online'` or `'offline'` - `lastChanged` is the date when `state` was last modified. - `typingUsers` is an array of all the users who are currently writing a message ### Messages prop Message objects are rendered differently depending on their type. Text, emoji, image, video and file types are supported.

Each message object has a `senderId` field which holds the id of the corresponding agent. If `senderId` matches the `currentUserId` prop, specific UI and actions will be implemented.

Notes: - `username` will be displayed on each message of corresponding agents if at least 3 users are in the room - `system` is used to show messages with a specific centered display - `indexId` can be used if you need to change a message ID that is already displayed in a room, this preventing an animation glitch. For example, when you don't know in advance the message ID your backend will create. Message states: - `saved: true` one checkmark - `distributed: true` two checkmarks - `seen: true` two blue checkmarks - `deleted: true` grey background with deleted message text - `failure: true` red clickable failure icon ```javascript messages="[ { _id: '7890', indexId: 12092, content: 'Message 1', senderId: '1234', username: 'John Doe', avatar: 'assets/imgs/doe.png', date: '13 November', timestamp: '10:20', system: false, saved: true, distributed: true, seen: true, deleted: false, failure: true, disableActions: false, disableReactions: false, files: [ { name: 'My File', size: 67351, type: 'png', audio: true, duration: 14.4, url: 'https://firebasestorage.googleapis.com/...', preview: '...', progress: 88 } ], reactions: { 😁: [ '1234', // USER_ID '4321' ], 🥰: [ '1234' ] }, replyMessage: { content: 'Reply Message', senderId: '4321', files: [ { name: 'My Replied File', size: 67351, type: 'png', audio: true, duration: 14.4, url: 'https://firebasestorage.googleapis.com/...', preview: '...' } ] }, } ]" ```
## Events API |
Event
| Params | Fires when a user | | ------------------------------------- | -------------------------------------------------------------------------- | ----------------------------------------------- | | `fetch-messages`(1) | `{ room, options }` | Scrolled on top to load more messages | | `fetch-more-rooms`(2) | - | Scrolled to load more rooms | | `send-message` | `{ roomId, content, files(11), replyMessage(12), usersTag }` | Sent a message | | `edit-message` | `{ roomId, messageId, newContent, files(11), replyMessage(12) ,usersTag }` | Edited a message | | `delete-message` | `{ roomId, message }` | Deleted a message | | `open-file` | `{ message, file }` | Clicked to view or download a file | | `open-user-tag`(3) | `{ user }` | Clicked on a user tag inside a message | | `open-failed-message` | `{ roomId, message }` | Clicked on the failure icon next to a message | | `add-room` | - | Clicked on the plus icon next to searchbar | | `search-room`(4) | `{ roomId, value }` | Typed a character in the room search input | | `room-action-handler`(5) | `{ roomId, action }` | Clicked on the vertical dots icon inside a room | | `menu-action-handler`(6) | `{ roomId, action }` | Clicked on the vertical dots icon inside a room | | `message-action-handler`(7) | `{ roomId, action, message }` | Clicked on the dropdown icon inside a message | | `message-selection-action-handler`(8) | `{ roomId, action, messages }` | Clicked on the select button inside a message | | `send-message-reaction` | `{ roomId, messageId, reaction, remove }` | Clicked on the emoji icon inside a message | | `room-info` (9) | `room` | Clicked the room header bar | | `toggle-rooms-list` | `{ opened }` | Clicked on the toggle icon inside a room header | | `textarea-action-handler`(10) | `{ roomId, message }` | Clicked on custom icon inside the footer | | `typing-message` | `{ roomId, message }` | Started typing a message | **(1)** `fetch-messages` is triggered every time a room is opened. If the room is opened for the first time, the `options` param will hold `reset: true`.
**(1)** `fetch-messages` should be a method implementing a pagination system. Its purpose is to load older messages of a conversation when the user scroll on top. **(2)** `fetch-more-rooms` is triggered when scrolling down the rooms list, and should be a method implementing a pagination system. **(3)** `open-user-tag` is triggered when clicking a user tag inside a message. When creating a user tag by typing `@` in the footer textarea and sending the message, the tag will be identified with the below pattern: ```javascript TAGGED_USER_ID ``` This will make the tag clickable inside a message. Ex: [message tag content](#messages-collection-inside-a-room-document)
`send-message` and `edit-message` events will handle that pattern for you and pass it in the `content` param. **(4)** `room-action-handler` is the result of the [`room-actions`](#props-api) prop.
When clicking a button from your `room-actions` array, `room-action-handler` will give you the name of the button that was click. Then you can do anything you want with it. Ex: ```javascript menuActionHandler({ roomId, action }) { switch (action.name) { case 'archiveRoom': // call a method to archive the room } } ``` **(5)** `menu-action-handler` is the result of the [`menu-actions`](#props-api) prop.
When clicking a button from your `menu-actions` array, `menu-action-handler` will give you the name of the button that was click. Then you can do anything you want with it. Ex: ```javascript menuActionHandler({ roomId, action }) { switch (action.name) { case 'inviteUser': // call a method to invite a user to the room case 'removeUser': // call a method to remove a user from the room case 'deleteRoom': // call a method to delete the room } } ``` **(6)** `search-room` can be enabled by using [`custom-search-room-enabled`](#props-api) prop. This will allow you to call your own method to filter out searched rooms. **(7)** `message-action-handler` is the result of the [`message-actions`](#props-api) prop.
When clicking a button from your `message-actions` array, `message-action-handler` will give you the name of the button that was click and the corresponding message data. Then you can do anything you want with it. Ex: ```javascript messageActionHandler({ roomId, action, message }) { switch (action.name) { case 'addMessageToFavorite': // call a method to add a message to the favorite list case 'shareMessage': // call a method to share the message with another user } } ``` **(8)** `message-selection-action-handler` is the result of the [`message-selection-actions`](#props-api) prop.
When clicking a button from your `message-selection-actions` array, `message-selection-action-handler` will give you the name of the button that was click and the corresponding selected messages data. Then you can do anything you want with it. Ex: ```javascript messageSelectionActionHandler({ roomId, action, message }) { switch (action.name) { case 'deleteMessages': // call a method to delete selected messaged case 'shareMessage': // call a method to share selected messaged with another user } } ``` **(9)** `room-info` is the result of the [`room-info-enabled`](#props-api) prop.
**(10)** `textarea-action-handler` is the result of the [`textarea-action-enabled`](#props-api) prop.
**(11)** Array of files where each file contain: `{ blob, localUrl, name, size, type, extension }` **(12)** `replyMessage` object is available when the user replied to another message by clicking the corresponding icon, and contains the message information that was clicked.
## Named Slots Example: ```javascript
This is a new room header
System message: {{ message.content }}
Normal message: {{ message.content }}
New Avatar
``` |
Slot
| Action | | ------------------------------------------- | ----------------------------------------------------------------- | | `custom-action-icon` | Add a custom icon inside the footer | | `rooms-header` | Add a template on top of rooms list (above the search bar) | | `room-list-item_{{ROOM_ID}}` | Replace the template of the room list items | | `room-list-info_{{ROOM_ID}}` | Replace the info of room list items | | `room-list-avatar_{{ROOM_ID}}` | Replace the avatar of room list items | | `room-list-options_{{ROOM_ID}}` | Replace the template of the list room options | | `room-list-options-icon_{{ROOM_ID}}` | Replace the room list options dropdown icon | | `rooms-header` | Replace the content above the search bar | | `rooms-list-search` | Replace the search bar | | `room-header` | Replace the template of the room header | | `room-header-avatar` | Replace the template of the room header avatar | | `room-header-info` | Replace the template of the room header text | | `room-options` | Replace the template of the room options | | `message_{{MESSAGE_ID}}` | Replace the template of the message (and system message) box | | `message-avatar_{{MESSAGE_ID}}` | Replace the template of the message avatar | | `message-failure_{{MESSAGE_ID}}` | Replace the message failure icon | | `messages-empty` | Replace the empty message template | | `rooms-empty` | Replace the empty rooms template | | `no-room-selected` | Replace the no room selected template | | `menu-icon` | Replace the room menu icon | | `toggle-icon` | Replace the toggle room list icon | | `spinner-icon-rooms` | Replace the loading spinner icon in the rooms list | | `spinner-icon-infinite-rooms` | Replace the loading spinner icon to load more rooms | | `spinner-icon-messages` | Replace the loading spinner icon in a room | | `spinner-icon-infinite-messages` | Replace the loading spinner icon to load more messages | | `spinner-icon-room-file` | Replace the loading spinner icon to load heavy files | | `spinner-icon-message-file_{{MESSAGE_ID}}` | Replace the loading spinner icon in a message containing an image | | `scroll-icon` | Replace the scroll to newest message icon | | `reply-close-icon` | Replace the reply close icon | | `image-close-icon` | Replace the image close icon | | `file-icon` | Replace the file icon | | `files-close-icon` | Replace the file close icon | | `edit-close-icon` | Replace the edit close icon | | `preview-close-icon` | Replace the media preview close icon | | `emoji-picker-icon` | Replace the emoji picker icon | | `emoji-picker-reaction-icon_{{MESSAGE_ID}}` | Replace the emoji picker reaction icon (in the message box) | | `paperclip-icon` | Replace the paperclip icon | | `send-icon` | Replace the message send icon | | `eye-icon_{{MESSAGE_ID}}` | Replace the eye icon (image message) | | `document-icon_{{MESSAGE_ID}}` | Replace the document icon | | `pencil-icon_{{MESSAGE_ID}}` | Replace the pencil icon | | `checkmark-icon_{{MESSAGE_ID}}` | Replace the checkmark icon of a message | | `checkmark-icon_{{ROOM_ID}}` | Replace the checkmark icon of a message from the rooms list | | `deleted-icon_{{MESSAGE_ID}}` | Replace the deleted icon of a message | | `deleted-icon_{{ROOM_ID}}` | Replace the deleted icon of a message from the rooms list | | `microphone-icon_{{ROOM_ID}}` | Replace the microphone icon | | `dropdown-icon_{{MESSAGE_ID}}` | Replace the dropdown icon | | `search-icon` | Replace the search icon | | `add-icon` | Replace the add room icon | | `audio-stop-icon` | Replace the audio recorder stop icon | | `audio-check-icon` | Replace the audio recorder confirm icon | | `audio-pause-icon_{{MESSAGE_ID}}` | Replace the message audio pause icon | | `audio-play-icon_{{MESSAGE_ID}}` | Replace the message audio play icon |
## Using with Firestore ### Source code You can find the source code to implement a full featured chat app using Firebase/Firestore inside the `demo` folder.
To test it using your own Firebase project: - Setup Cloud Firestore (to store users and rooms) and Realtime Database (to store users online status) - Clone this repository: `git clone https://github.com/advanced-chat/vue-advanced-chat.git` - Inside `demo/src/database/index.js` file, replace the line `const config = ...` by your own Firebase config - Go inside `demo` folder and run `npm run serve` ### Data structure If you decide to use the same code as in the `demo` folder to create your chat app, you need to have a specific Firestore data structure.
To help you get started, I added in `demo/src/App.vue` a method `addData` to initialize some data on your Firestore database. #### Users collection ```javascript users: { USER_ID_1: { _id: '1', username: 'User 1' }, USER_ID_2: { _id: '2', username: 'User 2' }, USER_ID_3: { _id: '3', username: 'User 2' } } ``` #### Rooms collection ```javascript chatRooms: { ROOM_ID_1: { users: ['1', '3'] }, ROOM_ID_2: { users: ['1', '2', '3'] } } ``` #### Messages collection inside a room document ```javascript messages: { MESSAGE_ID_1: { content: 'My first message to John', senderId: '2', timestamp: 'December 11, 2019 at 4:00:00 PM', seen: true } } ``` #### Notes - You need to create a composite index to order rooms by last message received. The easiest way to do it is to create a room, then click the error message url in the browser debugging console.
## [Contributing](https://github.com/advanced-chat/vue-advanced-chat/blob/master/.github/CONTRIBUTING.md) Your help is always appreciated :rocket: ## License This project is licensed under [MIT License](http://en.wikipedia.org/wiki/MIT_License)