Refactor WebRTC and Webcam Feature into Vue Mixin

* Move all WebRTC and webcam-related functionality from App.vue into an
  external Vue mixin.
* Centralize the stand-alone functions videoIconClass and
  videoButtonClass into the new WebRTC mixin, which is called by
  WhoListRow and MessageBox.
This commit is contained in:
Noah 2025-03-30 13:50:01 -07:00
parent f1d55abb66
commit 16924a5ff5
4 changed files with 1906 additions and 1851 deletions

File diff suppressed because it is too large Load Diff

View File

@ -3,6 +3,7 @@ import EmojiPicker from 'vue3-emoji-picker';
import LocalStorage from '../lib/LocalStorage'; import LocalStorage from '../lib/LocalStorage';
import ScamDetection from './ScamDetection.vue'; import ScamDetection from './ScamDetection.vue';
import VideoFlag from '../lib/VideoFlag'; import VideoFlag from '../lib/VideoFlag';
import WebRTC from '../lib/WebRTC';
import 'vue3-emoji-picker/css'; import 'vue3-emoji-picker/css';
export default { export default {
@ -118,45 +119,10 @@ export default {
// TODO: make DRY, copied from WhoListRow. // TODO: make DRY, copied from WhoListRow.
videoButtonClass() { videoButtonClass() {
let result = ""; return WebRTC.videoButtonClass(this.user, this.isVideoNotAllowed);
// 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() { videoButtonTitle() {
// Mouse-over title text for the video button. return WebRTC.videoButtonTitle(this.user);
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: { methods: {

View File

@ -1,5 +1,6 @@
<script> <script>
import VideoFlag from '../lib/VideoFlag'; import VideoFlag from '../lib/VideoFlag';
import WebRTC from '../lib/WebRTC';
export default { export default {
props: { props: {
@ -48,45 +49,10 @@ export default {
return ""; return "";
}, },
videoButtonClass() { videoButtonClass() {
let result = ""; return WebRTC.videoButtonClass(this.user, this.isVideoNotAllowed);
// 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() { videoButtonTitle() {
// Mouse-over title text for the video button. return WebRTC.videoButtonTitle(this.user);
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("; ");
}, },
avatarURL() { avatarURL() {
if (this.user.avatar) { if (this.user.avatar) {

1890
src/lib/WebRTC.js Normal file

File diff suppressed because it is too large Load Diff