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 {