mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2026-04-30 09:32:05 -04:00
- adds log.info/error/debug/warn/fatal - adds ability to write logs to any file - add CLI options in addition to env to set log levels
534 lines
18 KiB
QML
534 lines
18 KiB
QML
pragma Singleton
|
|
pragma ComponentBehavior: Bound
|
|
|
|
import QtQuick
|
|
import Quickshell
|
|
import Quickshell.Io
|
|
import Quickshell.Bluetooth
|
|
import qs.Services
|
|
|
|
Singleton {
|
|
id: root
|
|
|
|
readonly property BluetoothAdapter adapter: Bluetooth.defaultAdapter
|
|
readonly property bool available: adapter !== null
|
|
readonly property bool enabled: (adapter && adapter.enabled) ?? false
|
|
readonly property bool discovering: (adapter && adapter.discovering) ?? false
|
|
readonly property var devices: adapter ? adapter.devices : null
|
|
readonly property bool enhancedPairingAvailable: DMSService.dmsAvailable && DMSService.apiVersion >= 9 && DMSService.capabilities.includes("bluetooth")
|
|
readonly property bool connected: {
|
|
if (!adapter || !adapter.devices) {
|
|
return false;
|
|
}
|
|
|
|
let isConnected = false;
|
|
adapter.devices.values.forEach(dev => {
|
|
if (dev.connected)
|
|
isConnected = true;
|
|
});
|
|
return isConnected;
|
|
}
|
|
readonly property var pairedDevices: {
|
|
if (!adapter || !adapter.devices) {
|
|
return [];
|
|
}
|
|
|
|
return adapter.devices.values.filter(dev => {
|
|
return dev && (dev.paired || dev.trusted);
|
|
});
|
|
}
|
|
readonly property var allDevicesWithBattery: {
|
|
if (!adapter || !adapter.devices) {
|
|
return [];
|
|
}
|
|
|
|
return adapter.devices.values.filter(dev => {
|
|
return dev && dev.batteryAvailable && dev.battery > 0;
|
|
});
|
|
}
|
|
|
|
function sortDevices(devices) {
|
|
return devices.sort((a, b) => {
|
|
const aName = a.name || a.deviceName || "";
|
|
const bName = b.name || b.deviceName || "";
|
|
const aAddr = a.address || "";
|
|
const bAddr = b.address || "";
|
|
|
|
const aHasRealName = aName.includes(" ") && aName.length > 3;
|
|
const bHasRealName = bName.includes(" ") && bName.length > 3;
|
|
|
|
if (aHasRealName && !bHasRealName)
|
|
return -1;
|
|
if (!aHasRealName && bHasRealName)
|
|
return 1;
|
|
|
|
if (aHasRealName && bHasRealName) {
|
|
return aName.localeCompare(bName);
|
|
}
|
|
|
|
return aAddr.localeCompare(bAddr);
|
|
});
|
|
}
|
|
|
|
function getDeviceIcon(device) {
|
|
if (!device) {
|
|
return "bluetooth";
|
|
}
|
|
|
|
const name = (device.name || device.deviceName || "").toLowerCase();
|
|
const icon = (device.icon || "").toLowerCase();
|
|
|
|
const audioKeywords = ["headset", "audio", "headphone", "airpod", "arctis"];
|
|
if (audioKeywords.some(keyword => icon.includes(keyword) || name.includes(keyword))) {
|
|
return "headset";
|
|
}
|
|
|
|
if (icon.includes("mouse") || name.includes("mouse")) {
|
|
return "mouse";
|
|
}
|
|
|
|
if (icon.includes("keyboard") || name.includes("keyboard")) {
|
|
return "keyboard";
|
|
}
|
|
|
|
const phoneKeywords = ["phone", "iphone", "android", "samsung"];
|
|
if (phoneKeywords.some(keyword => icon.includes(keyword) || name.includes(keyword))) {
|
|
return "smartphone";
|
|
}
|
|
|
|
if (icon.includes("watch") || name.includes("watch")) {
|
|
return "watch";
|
|
}
|
|
|
|
if (icon.includes("speaker") || name.includes("speaker")) {
|
|
return "speaker";
|
|
}
|
|
|
|
if (icon.includes("display") || name.includes("tv")) {
|
|
return "tv";
|
|
}
|
|
|
|
return "bluetooth";
|
|
}
|
|
|
|
function canConnect(device) {
|
|
if (!device) {
|
|
return false;
|
|
}
|
|
|
|
return !device.paired && !device.pairing && !device.blocked;
|
|
}
|
|
|
|
function getSignalStrength(device) {
|
|
if (!device || device.signalStrength === undefined || device.signalStrength <= 0) {
|
|
return "Unknown";
|
|
}
|
|
|
|
const signal = device.signalStrength;
|
|
if (signal >= 80) {
|
|
return "Excellent";
|
|
}
|
|
if (signal >= 60) {
|
|
return "Good";
|
|
}
|
|
if (signal >= 40) {
|
|
return "Fair";
|
|
}
|
|
if (signal >= 20) {
|
|
return "Poor";
|
|
}
|
|
|
|
return "Very Poor";
|
|
}
|
|
|
|
function getSignalIcon(device) {
|
|
if (!device || device.signalStrength === undefined || device.signalStrength <= 0) {
|
|
return "signal_cellular_null";
|
|
}
|
|
|
|
const signal = device.signalStrength;
|
|
if (signal >= 80) {
|
|
return "signal_cellular_4_bar";
|
|
}
|
|
if (signal >= 60) {
|
|
return "signal_cellular_3_bar";
|
|
}
|
|
if (signal >= 40) {
|
|
return "signal_cellular_2_bar";
|
|
}
|
|
if (signal >= 20) {
|
|
return "signal_cellular_1_bar";
|
|
}
|
|
|
|
return "signal_cellular_0_bar";
|
|
}
|
|
|
|
function isDeviceBusy(device) {
|
|
if (!device) {
|
|
return false;
|
|
}
|
|
return device.pairing || device.state === BluetoothDeviceState.Disconnecting || device.state === BluetoothDeviceState.Connecting;
|
|
}
|
|
|
|
function connectDeviceWithTrust(device) {
|
|
if (!device) {
|
|
return;
|
|
}
|
|
|
|
device.trusted = true;
|
|
device.connect();
|
|
}
|
|
|
|
function pairDevice(device, callback) {
|
|
if (!device) {
|
|
if (callback)
|
|
callback({
|
|
error: "Invalid device"
|
|
});
|
|
return;
|
|
}
|
|
|
|
// The DMS backend actually implements a bluez agent, so we can pair anything
|
|
if (enhancedPairingAvailable) {
|
|
const devicePath = getDevicePath(device);
|
|
DMSService.bluetoothPair(devicePath, callback);
|
|
return;
|
|
}
|
|
|
|
// Quickshell does not implement a bluez agent, so we can try to pair but only with devices that don't require a passcode
|
|
device.trusted = true;
|
|
device.connect();
|
|
if (callback)
|
|
callback({
|
|
success: true
|
|
});
|
|
}
|
|
|
|
function getCardName(device) {
|
|
if (!device) {
|
|
return "";
|
|
}
|
|
return `bluez_card.${device.address.replace(/:/g, "_")}`;
|
|
}
|
|
|
|
function getDevicePath(device) {
|
|
if (!device || !device.address) {
|
|
return "";
|
|
}
|
|
const adapterPath = adapter ? "/org/bluez/hci0" : "/org/bluez/hci0";
|
|
return `${adapterPath}/dev_${device.address.replace(/:/g, "_")}`;
|
|
}
|
|
|
|
function isAudioDevice(device) {
|
|
if (!device) {
|
|
return false;
|
|
}
|
|
const icon = getDeviceIcon(device);
|
|
return icon === "headset" || icon === "speaker";
|
|
}
|
|
|
|
function getCodecInfo(codecName) {
|
|
const codec = codecName.replace(/-/g, "_").toUpperCase();
|
|
|
|
const codecMap = {
|
|
"LDAC": {
|
|
"name": "LDAC",
|
|
"description": "Highest quality • Higher battery usage",
|
|
"qualityColor": "#4CAF50"
|
|
},
|
|
"APTX_HD": {
|
|
"name": "aptX HD",
|
|
"description": "High quality • Balanced battery",
|
|
"qualityColor": "#FF9800"
|
|
},
|
|
"APTX": {
|
|
"name": "aptX",
|
|
"description": "Good quality • Low latency",
|
|
"qualityColor": "#FF9800"
|
|
},
|
|
"AAC": {
|
|
"name": "AAC",
|
|
"description": "Balanced quality and battery",
|
|
"qualityColor": "#2196F3"
|
|
},
|
|
"SBC_XQ": {
|
|
"name": "SBC-XQ",
|
|
"description": "Enhanced SBC • Better compatibility",
|
|
"qualityColor": "#2196F3"
|
|
},
|
|
"SBC": {
|
|
"name": "SBC",
|
|
"description": "Basic quality • Universal compatibility",
|
|
"qualityColor": "#9E9E9E"
|
|
},
|
|
"MSBC": {
|
|
"name": "mSBC",
|
|
"description": "Modified SBC • Optimized for speech",
|
|
"qualityColor": "#9E9E9E"
|
|
},
|
|
"CVSD": {
|
|
"name": "CVSD",
|
|
"description": "Basic speech codec • Legacy compatibility",
|
|
"qualityColor": "#9E9E9E"
|
|
}
|
|
};
|
|
|
|
return codecMap[codec] || {
|
|
"name": codecName,
|
|
"description": "Unknown codec",
|
|
"qualityColor": "#9E9E9E"
|
|
};
|
|
}
|
|
|
|
property var deviceCodecs: ({})
|
|
|
|
function updateDeviceCodec(deviceAddress, codec) {
|
|
deviceCodecs[deviceAddress] = codec;
|
|
deviceCodecsChanged();
|
|
}
|
|
|
|
function refreshDeviceCodec(device) {
|
|
if (!device || !device.connected || !isAudioDevice(device)) {
|
|
return;
|
|
}
|
|
|
|
const cardName = getCardName(device);
|
|
codecQueryProcess.cardName = cardName;
|
|
codecQueryProcess.deviceAddress = device.address;
|
|
codecQueryProcess.availableCodecs = [];
|
|
codecQueryProcess.parsingTargetCard = false;
|
|
codecQueryProcess.detectedCodec = "";
|
|
codecQueryProcess.running = true;
|
|
}
|
|
|
|
function getCurrentCodec(device, callback) {
|
|
if (!device || !device.connected || !isAudioDevice(device)) {
|
|
callback("");
|
|
return;
|
|
}
|
|
|
|
const cardName = getCardName(device);
|
|
codecQueryProcess.cardName = cardName;
|
|
codecQueryProcess.callback = callback;
|
|
codecQueryProcess.availableCodecs = [];
|
|
codecQueryProcess.parsingTargetCard = false;
|
|
codecQueryProcess.detectedCodec = "";
|
|
codecQueryProcess.running = true;
|
|
}
|
|
|
|
function getAvailableCodecs(device, callback) {
|
|
if (!device || !device.connected || !isAudioDevice(device)) {
|
|
callback([], "");
|
|
return;
|
|
}
|
|
|
|
const cardName = getCardName(device);
|
|
codecFullQueryProcess.cardName = cardName;
|
|
codecFullQueryProcess.callback = callback;
|
|
codecFullQueryProcess.availableCodecs = [];
|
|
codecFullQueryProcess.parsingTargetCard = false;
|
|
codecFullQueryProcess.detectedCodec = "";
|
|
codecFullQueryProcess.running = true;
|
|
}
|
|
|
|
function switchCodec(device, profileName, callback) {
|
|
if (!device || !isAudioDevice(device)) {
|
|
callback(false, "Invalid device");
|
|
return;
|
|
}
|
|
|
|
const cardName = getCardName(device);
|
|
codecSwitchProcess.cardName = cardName;
|
|
codecSwitchProcess.profile = profileName;
|
|
codecSwitchProcess.callback = callback;
|
|
codecSwitchProcess.running = true;
|
|
}
|
|
|
|
Process {
|
|
id: codecQueryProcess
|
|
|
|
property string cardName: ""
|
|
property string deviceAddress: ""
|
|
property var callback: null
|
|
property bool parsingTargetCard: false
|
|
property string detectedCodec: ""
|
|
property var availableCodecs: []
|
|
|
|
command: ["pactl", "list", "cards"]
|
|
|
|
onExited: (exitCode, exitStatus) => {
|
|
if (exitCode === 0 && detectedCodec) {
|
|
if (deviceAddress) {
|
|
root.updateDeviceCodec(deviceAddress, detectedCodec);
|
|
}
|
|
if (callback) {
|
|
callback(detectedCodec);
|
|
}
|
|
} else if (callback) {
|
|
callback("");
|
|
}
|
|
|
|
parsingTargetCard = false;
|
|
detectedCodec = "";
|
|
availableCodecs = [];
|
|
deviceAddress = "";
|
|
callback = null;
|
|
}
|
|
|
|
stdout: SplitParser {
|
|
splitMarker: "\n"
|
|
onRead: data => {
|
|
let line = data.trim();
|
|
|
|
if (line.includes(`Name: ${codecQueryProcess.cardName}`)) {
|
|
codecQueryProcess.parsingTargetCard = true;
|
|
return;
|
|
}
|
|
|
|
if (codecQueryProcess.parsingTargetCard && line.startsWith("Name: ") && !line.includes(codecQueryProcess.cardName)) {
|
|
codecQueryProcess.parsingTargetCard = false;
|
|
return;
|
|
}
|
|
|
|
if (codecQueryProcess.parsingTargetCard) {
|
|
if (line.startsWith("Active Profile:")) {
|
|
let profile = line.split(": ")[1] || "";
|
|
let activeCodec = codecQueryProcess.availableCodecs.find(c => {
|
|
return c.profile === profile;
|
|
});
|
|
if (activeCodec) {
|
|
codecQueryProcess.detectedCodec = activeCodec.name;
|
|
}
|
|
return;
|
|
}
|
|
if (line.includes("codec") && line.includes("available: yes")) {
|
|
let parts = line.split(": ");
|
|
if (parts.length >= 2) {
|
|
let profile = parts[0].trim();
|
|
let description = parts[1];
|
|
let codecMatch = description.match(/codec ([^\)\s]+)/i);
|
|
let codecName = codecMatch ? codecMatch[1].toUpperCase() : "UNKNOWN";
|
|
let codecInfo = root.getCodecInfo(codecName);
|
|
if (codecInfo && !codecQueryProcess.availableCodecs.some(c => {
|
|
return c.profile === profile;
|
|
})) {
|
|
let newCodecs = codecQueryProcess.availableCodecs.slice();
|
|
newCodecs.push({
|
|
"name": codecInfo.name,
|
|
"profile": profile,
|
|
"description": codecInfo.description,
|
|
"qualityColor": codecInfo.qualityColor
|
|
});
|
|
codecQueryProcess.availableCodecs = newCodecs;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: codecFullQueryProcess
|
|
|
|
property string cardName: ""
|
|
property var callback: null
|
|
property bool parsingTargetCard: false
|
|
property string detectedCodec: ""
|
|
property var availableCodecs: []
|
|
|
|
command: ["pactl", "list", "cards"]
|
|
|
|
onExited: function (exitCode, exitStatus) {
|
|
if (callback) {
|
|
callback(exitCode === 0 ? availableCodecs : [], exitCode === 0 ? detectedCodec : "");
|
|
}
|
|
parsingTargetCard = false;
|
|
detectedCodec = "";
|
|
availableCodecs = [];
|
|
callback = null;
|
|
}
|
|
|
|
stdout: SplitParser {
|
|
splitMarker: "\n"
|
|
onRead: data => {
|
|
let line = data.trim();
|
|
|
|
if (line.includes(`Name: ${codecFullQueryProcess.cardName}`)) {
|
|
codecFullQueryProcess.parsingTargetCard = true;
|
|
return;
|
|
}
|
|
|
|
if (codecFullQueryProcess.parsingTargetCard && line.startsWith("Name: ") && !line.includes(codecFullQueryProcess.cardName)) {
|
|
codecFullQueryProcess.parsingTargetCard = false;
|
|
return;
|
|
}
|
|
|
|
if (codecFullQueryProcess.parsingTargetCard) {
|
|
if (line.startsWith("Active Profile:")) {
|
|
let profile = line.split(": ")[1] || "";
|
|
let activeCodec = codecFullQueryProcess.availableCodecs.find(c => {
|
|
return c.profile === profile;
|
|
});
|
|
if (activeCodec) {
|
|
codecFullQueryProcess.detectedCodec = activeCodec.name;
|
|
}
|
|
return;
|
|
}
|
|
if (line.includes("codec") && line.includes("available: yes")) {
|
|
let parts = line.split(": ");
|
|
if (parts.length >= 2) {
|
|
let profile = parts[0].trim();
|
|
let description = parts[1];
|
|
let codecMatch = description.match(/codec ([^\)\s]+)/i);
|
|
let codecName = codecMatch ? codecMatch[1].toUpperCase() : "UNKNOWN";
|
|
let codecInfo = root.getCodecInfo(codecName);
|
|
if (codecInfo && !codecFullQueryProcess.availableCodecs.some(c => {
|
|
return c.profile === profile;
|
|
})) {
|
|
let newCodecs = codecFullQueryProcess.availableCodecs.slice();
|
|
newCodecs.push({
|
|
"name": codecInfo.name,
|
|
"profile": profile,
|
|
"description": codecInfo.description,
|
|
"qualityColor": codecInfo.qualityColor
|
|
});
|
|
codecFullQueryProcess.availableCodecs = newCodecs;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: codecSwitchProcess
|
|
|
|
property string cardName: ""
|
|
property string profile: ""
|
|
property var callback: null
|
|
|
|
command: ["pactl", "set-card-profile", cardName, profile]
|
|
|
|
onExited: function (exitCode, exitStatus) {
|
|
if (callback) {
|
|
callback(exitCode === 0, exitCode === 0 ? "Codec switched successfully" : "Failed to switch codec");
|
|
}
|
|
|
|
// If successful, refresh the codec for this device
|
|
if (exitCode === 0) {
|
|
if (root.adapter && root.adapter.devices) {
|
|
root.adapter.devices.values.forEach(device => {
|
|
if (device && root.getCardName(device) === cardName) {
|
|
Qt.callLater(() => root.refreshDeviceCodec(device));
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
callback = null;
|
|
}
|
|
}
|
|
}
|