author | created on | last updated | issue id |
---|---|---|---|
Mike Griese @zadjii-msft |
2019-11-13 |
2019-12-05 |
Oftentimes, users have some common settings that they'd like applied to all of their profiles, without needing to manually edit the settings of each of them. This doc will cover some of the many proposals on how to expose that functionality to the user in our JSON settings model. In this first document, we'll examine a number of proposed solutions, as well as state our finalized design.
During the course of the pull request review on #3369, the original pull request for this feature's implementation, it became apparent that the entire team has differing opinions on how this feature should be exposed to the user. This doc is born from that discussion.
The following are a number of different proposals of different ways to achieve the proposed functionality:
defaultSettings
Profile object in the global settings__default__
Profile object in the user's profiles- Change
profiles
to an object with alist
of profiles and adefaults
object inheritFrom
in profiles
{
"$schema": "https://aka.ms/terminal-profiles-schema",
"defaultProfile": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"defaultSettings":
{
"useAcrylic": true,
"acrylicOpacity": 0.1,
"fontFace": "Cascadia Code",
"fontSize": 10
},
"requestedTheme" : "dark",
"showTabsInTitlebar" : true,
"profiles":
[
{
"guid": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"name": "Windows PowerShell",
"commandline": "powershell.exe",
"hidden": false
},
{
"guid": "{0caa0dad-35be-5f56-a8ff-afceeeaa6101}",
"name": "cmd",
"commandline": "cmd.exe",
"hidden": false
}
],
"schemes": [],
"keybindings": []
}
Puts all the default profiles settings in one object. It's immediately obvious when scanning the file where the defaults are.
There's one object that applies to all the subsequent profiles, and that
object is the defaultSettings
object.
People were concerned about the naming of this property. No one has a name that we're quite happy with:
defaultSettings
: This kinda seems to conflict conceptually with "defaults.json". It's different, but is that obvious?defaultProfileSettings
: Implies "settings of the default profile"defaults
: This kinda seems to conflict conceptually with "defaults.json"baseProfileSettings
: not the worst, but not terribly intuitive- Others considered with less enthusiasm
profiles.defaults
: people don't love the idea of a.
, but hey, VsCode does it.inheritedSettings
rootSettings
globalSettings
: again maybe conflicts a bit with other concepts/propertiesprofileSettings
profilePrototype
Users may be confused about the purpose of this random Profile
that's in the
globals. What's that profile doing there? Is it the default profile?
{
"$schema": "https://aka.ms/terminal-profiles-schema",
"defaultProfile": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"requestedTheme" : "dark",
"showTabsInTitlebar" : true,
"profiles":
[
{
"guid": "__default__",
"useAcrylic": true,
"acrylicOpacity": 0.1,
"fontFace": "Cascadia Code",
"fontSize": 10
},
{
"guid": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"name": "Windows PowerShell",
"commandline": "powershell.exe",
"hidden": false
},
{
"guid": "{0caa0dad-35be-5f56-a8ff-afceeeaa6101}",
"name": "cmd",
"commandline": "cmd.exe",
"hidden": false
}
],
"schemes": [],
"keybindings": []
}
Puts all the default profiles settings in one object. Probably not as clear as proposal 1, since it could be anywhere in the list of profiles.
In this proposal, the default profile is grouped into the same list of objects
as the other profiles. All the profiles, and the defaults are all under the
"profiles"
object. Makes sense.
The only way to definitively identify that this profile is special is by giving it a constant string. This string is not a guid, which again, would be obvious.
Adding a profile that has a mysterious guid
value use that profile as the
"defaults" is very unintuitive. Nothing aside from documentation would
indicate to the user "hey, add this magic profile blob to use as defaults across
all your profiles".
It might be unintuitive that one profile from the list of profiles affects all the others.
{
"$schema": "https://aka.ms/terminal-profiles-schema",
"defaultProfile": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"requestedTheme" : "dark",
"showTabsInTitlebar" : true,
"profiles":
{
"defaults": {
"useAcrylic": true,
"acrylicOpacity": 0.1,
"fontFace": "Cascadia Code",
"fontSize": 10
},
"list":[
{
"guid": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"name": "Windows PowerShell",
"commandline": "powershell.exe",
"hidden": false
},
{
"guid": "{0caa0dad-35be-5f56-a8ff-afceeeaa6101}",
"name": "cmd",
"commandline": "cmd.exe",
"hidden": false
}
]
},
"schemes": [],
"keybindings": []
}
In this proposal, the default profile is grouped into the same object as the
list of profiles. All the profiles, and the defaults are all under the
"profiles"
object. Makes sense.
Fortunately, we can add this functionality without breaking the existing schema. With Jsoncpp, we can determine at runtime if an object is an array or an object. If it's an array, we can fall back to the current behavior, safe in our knowledge that there's no defaults object. If the object is an array however, we can then dig into the object to find the default profile and the list of profiles.
This is a pretty big delta to the settings schema. Instead of using profiles
as a list of Profile
objects, it instead becomes an object, with a list inside
it.
As noted above, we could gracefully upgrade this. If the profiles
object is a
list, then we can assume there's no defaults
. This ensures that user's current
settings files don't break. This is not a major problem.
Some people just hate having things indented this much. 4 layers of indentation is quite a lot.
{
"$schema": "https://aka.ms/terminal-profiles-schema",
"defaultProfile": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"requestedTheme" : "dark",
"showTabsInTitlebar" : true,
"profiles":
[
{
"guid": "{11111111-1111-1111-1111-111111111111}",
"hidden": true,
"useAcrylic": true,
"acrylicOpacity": 0.1,
"fontFace": "Cascadia Code",
"fontSize": 10
},
{
"guid": "{61c54bbd-c2c6-5271-96e7-009a87ff44bf}",
"inheritFrom": "{11111111-1111-1111-1111-111111111111}",
"name": "Windows PowerShell",
"commandline": "powershell.exe",
"hidden": false
},
{
"guid": "{0caa0dad-35be-5f56-a8ff-afceeeaa6101}",
"inheritFrom": "{11111111-1111-1111-1111-111111111111}",
"name": "cmd",
"commandline": "cmd.exe",
"hidden": false
},
{
"guid": "{0caa0dad-ffff-5f56-a8ff-afceeeaa6101}",
"inheritFrom": "{0caa0dad-35be-5f56-a8ff-afceeeaa6101}",
"name": "This is another CMD",
"commandline": "cmd.exe /c myCoolScript.bat",
"hidden": false
}
],
"schemes": [],
"keybindings": []
}
Simply adding a new property to Profile
would not majorly alter the structure
of the file.
inheritFrom
is very unique relative to other keys we already have.
This lets the user have potentially many layers of settings grouping. These layers would let the user separate out common settings however they like, without forcing them to a single "default" profile. They could potentially have many "default" profiles, e.g.
- one that's used for all their WSL profiles, with
startingDirectory
set to~
andfontFace
set to "Ubuntu Mono" - One that's used for all their powershell profiles
etc.
Using the guid in the inheritFrom
field is the only way to be sure we're
uniquely identifying profiles. However, guids are notoriously un-friendly. The
above example manually uses "{11111111-1111-1111-1111-111111111111}"
as the
guid of the "default" profile, but inheriting from other profiles with "real"
GUIDs would be less understandable. Consider the "This is another CMD" case,
where it's inheriting from the "cmd" profile. That "inheritFrom"
value does
not mean at a quick glance "cmd".
This is mostly a technical challenge, but this does make the implementation a bit trickier.
When the user edits settings for a profile with the UI, do we only place the changes in the top-most profile?
How do we communicate in the UI that a profile is inheriting settings from other profiles?
Maybe not as easy to mentally picture how one profile inherits from another. The user would probably need to manually build the tree of profile inheritance in their own head to understand how a profile gets its settings.
After discussion the available options, the team has settled on proposal 3. The major selling points being:
- It groups the new "default profile settings" with the rest of the profile settings
- While being a schema change, it's not a breaking schema change.
- When looking at the settings, it's easy to understand how they're related
We also like the idea of proposal 4, but felt that it was too heavy-handed of an approach for this relatively simple feature. It's been added to the backlog of terminal features, tracked in #3818.