diff --git a/src/App.vue b/src/App.vue index 4250186..ff949f5 100644 --- a/src/App.vue +++ b/src/App.vue @@ -5004,7 +5004,8 @@ export default {
+ height="24" alt="" + :class="{ 'offline-avatar': isUserOffline(c.name) }">
@@ -5247,13 +5248,32 @@ export default {
- +
@@ -5548,4 +5568,9 @@ export default { .forcibly-single-line { white-space: nowrap; } + +/* Grey avatar for offline user on your DMs */ +.offline-avatar { + filter: grayscale(); +} diff --git a/src/components/MessageBox.vue b/src/components/MessageBox.vue index 54c985d..4597d41 100644 --- a/src/components/MessageBox.vue +++ b/src/components/MessageBox.vue @@ -2,6 +2,7 @@ import EmojiPicker from 'vue3-emoji-picker'; import LocalStorage from '../lib/LocalStorage'; import ScamDetection from './ScamDetection.vue'; +import VideoFlag from '../lib/VideoFlag'; import 'vue3-emoji-picker/css'; export default { @@ -18,9 +19,14 @@ export default { reactions: Object, // emoji reactions on the message reportEnabled: Boolean, // Report Message webhook is available position: Number, // position of the message (0 to n), for the emoji menu to know which side to pop + totalCount: Number, // total count of messages isDm: Boolean, // is in a DM thread (hide DM buttons) isOp: Boolean, // current user is Operator (always show takeback button) noButtons: Boolean, // hide all message buttons (e.g. for Report Modal) + + // User webcam settings + isVideoNotAllowed: Boolean, + videoIconClass: String, }, components: { EmojiPicker, @@ -28,10 +34,12 @@ export default { }, data() { return { + VideoFlag: VideoFlag, + // Emoji picker visible showEmojiPicker: false, - // Message menu (compact displays) + // Message menu (compact displays, and overflow button on card layout) menuVisible: false, // Favorite emojis @@ -96,6 +104,60 @@ export default { } return 'auto'; }, + + // Unique ID component for dropdown menus. + uniqueID() { + // Messages sent by users always have a msgID, use that. + if (this.message.msgID) { + return this.message.msgID; + } + + // Others (e.g. ChatServer messages), return something unique. + return `${this.position}-${this.user.username}-${this.message.at}-${Math.random()*99999}`; + }, + + // TODO: make DRY, copied from WhoListRow. + videoButtonClass() { + let result = ""; + + // VIP background if their cam is set to VIPs only + if ((this.user.video & VideoFlag.Active) && (this.user.video & VideoFlag.VipOnly)) { + result = "has-background-vip "; + } + + // Colors and/or cursors. + if ((this.user.video & VideoFlag.Active) && (this.user.video & VideoFlag.NSFW)) { + result += "is-danger is-outlined"; + } else if ((this.user.video & VideoFlag.Active) && !(this.user.video & VideoFlag.NSFW)) { + result += "is-link is-outlined"; + } else if (this.isVideoNotAllowed) { + result += "cursor-notallowed"; + } + + return result; + }, + videoButtonTitle() { + // Mouse-over title text for the video button. + let parts = ["Open video stream"]; + + if (this.user.video & VideoFlag.MutualRequired) { + parts.push("mutual video sharing required"); + } + + if (this.user.video & VideoFlag.MutualOpen) { + parts.push("will auto-open your video"); + } + + if (this.user.video & VideoFlag.VipOnly) { + parts.push(`${this.vipConfig.Name} only`); + } + + if (this.user.video & VideoFlag.NonExplicit) { + parts.push("prefers non-explicit video"); + } + + return parts.join("; "); + }, }, methods: { openProfile() { @@ -108,6 +170,10 @@ export default { }); }, + openVideo() { + this.$emit('open-video', this.user); + }, + muteUser() { this.$emit('mute-user', this.message.username); }, @@ -213,10 +279,10 @@ export default { {{ prettyDate(message.at) }} - + {{ nickname }} - (offline) - (@{{ message.username }}) + (@{{ message.username }}) {{ message.message }} @@ -231,40 +297,36 @@ export default {
-
-
+ + +
+ -
-
-
- +
+
+ - - {{ nickname }} + + {{ nickname }} - - (offline) - -
-
- {{ prettyDate(message.at) }} -
+ (offline) +
-