Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

FEAT(client): Settings Profiles #6681

Draft
wants to merge 2 commits into
base: master
Choose a base branch
from

Conversation

Hartmnt
Copy link
Member

@Hartmnt Hartmnt commented Jan 5, 2025

This is very much WIP and I need to add a proper description, but it is functional on a fundamental level.

Feedback for both UI and settings refactor welcome

sneak_peak.mp4

Closes #1264

@Hartmnt Hartmnt added client feature-request This issue or PR deals with a new feature labels Jan 5, 2025
Copy link
Member

@Krzmbrzl Krzmbrzl left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For reference, this implements #1264

One thing we need to think about/discuss is whether we believe whether all settings are created equal or whether there might be some settings that should be profile-dependent and some which are global-independent.
In the latter category, one might have things like the ask-on-quit setting.

If we believe that some settings should be independent of profiles, then the entire feature would become much more complex as this dependence would also have to be reflected in the UI in order to not make users guess whether a given setting depends on the profile or not 👀

@@ -401,6 +446,8 @@ std::size_t qHash(const ChannelTarget &target) {
return qHash(target.channelID);
}

const QString Profiles::s_default_profile_name = QLatin1String("default");
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not use std::string for this?

Comment on lines +192 to +193
QString activeProfileName = s_default_profile_name;
QMap< QString, Settings > allProfiles = {};
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where (reasonably) possible new code should prefer the use of std containers

@Hartmnt
Copy link
Member Author

Hartmnt commented Jan 7, 2025

One thing we need to think about/discuss is whether we believe whether all settings are created equal or whether there might be some settings that should be profile-dependent and some which are global-independent.
In the latter category, one might have things like the ask-on-quit setting.

I am 100% in favor of "all settings are created equal".

  1. The implementation complexity (and maintainability) would be a nightmare. We would need 2 settings versions and keep track of global vs non-global ourselves.
  2. The user should decide what settings are important per profile. Maybe they want Ask-on-Quit profile specific 😆 And I imagine the outcry for "I can't set this setting per profile" will be much harder than "why don't you save this globally?".
  3. The "Add" will actually duplicate the current profile, which should make it pretty easy to deal with this UX-wise from the user side. And in the worst case where you have to change every profile for a single setting: I don't see the benefit in making this so much more complex to implement just because the user will have to set like 5 checkboxes in that scenario...

@Krzmbrzl One thing that I am unsure about as of yet is "settings_version". That should be included in the individual profiles, but when we leave it out of the JSON root, we might - in rare, error caused circumstances - have users that might not be able to start Mumble... I/We have to come up with a nice solution for this

@Krzmbrzl
Copy link
Member

Krzmbrzl commented Jan 8, 2025

I am 100% in favor of "all settings are created equal".

I'm also like 80% or 90% sure that this is likely the correct thing to do, so let's stick with that 👍

One thing that I am unsure about as of yet is "settings_version". That should be included in the individual profiles, but when we leave it out of the JSON root, we might - in rare, error caused circumstances - have users that might not be able to start Mumble... I/We have to come up with a nice solution for this

settings_version is meant as a version number for the format in which we store the settings. Therefore, it has to be a global property rather than a profile-specific one (unless you want to allow different profiles to use different syntax/semantics in which they are stored on disk, but that would likely just be a nightmare all around).

It is meant for scenarios such as

// Compatibility layer for overtaking the old (now deprecated) settings
// This block should only be called once at the first start of the new Mumble version
// As echo cancellation was not available on macOS before, we don't have to run this compatibility
// code on macOS (instead simply use the new default as set in the constructor).
if (settings_ptr->contains("audio/echo")) {
bool deprecatedEcho = false;
bool deprecatedEchoMulti = false;
LOAD(deprecatedEcho, "audio/echo");
LOAD(deprecatedEchoMulti, "audio/echomulti");
if (deprecatedEcho) {
if (deprecatedEchoMulti) {
echoOption = EchoCancelOptionID::SPEEX_MULTICHANNEL;
} else {
echoOption = EchoCancelOptionID::SPEEX_MIXED;
}
} else {
echoOption = EchoCancelOptionID::DISABLED;
}
}

So with settings_version you could simply check the version and decide based on that how you have to parse the settings file, without having to do this awkward "always do the new format but also always the old in case the file is not yet updated" procedure. Similarly, if we rename a setting, this could cause the version to increase, though that is not (always) necessary as we have
void migrateSettings(nlohmann::json &json, int settingsVersion) {
// Perform conversions required to transform the given JSON into the format applicable to be read out by the most
// recent standards
// Check if the old ask_on_quit key exists and the new one does not exist within the json file
if (json.contains("ask_on_quit")
&& (!json.contains(static_cast< const char * >(SettingsKeys::QUIT_BEHAVIOR_KEY)))) {
if (!json.at("ask_on_quit").get< bool >()) {
json[SettingsKeys::QUIT_BEHAVIOR_KEY] = QuitBehavior::ALWAYS_QUIT;
} else {
json[SettingsKeys::QUIT_BEHAVIOR_KEY] = QuitBehavior::ALWAYS_ASK;
}
}
if (json.contains("play_transmit_cue")
&& (!json.contains(static_cast< const char * >(SettingsKeys::TRANSMIT_CUE_WHEN_PTT_KEY)))) {
json[SettingsKeys::TRANSMIT_CUE_WHEN_PTT_KEY] = json.at("play_transmit_cue").get< bool >();
}
(void) settingsVersion;
}


Finally, if not already implemented, we will probably want a hotkey to switch to a specific settings profile (and one to cycle through them?).

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
client feature-request This issue or PR deals with a new feature
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Feature request: Audio profiles
2 participants