-
Notifications
You must be signed in to change notification settings - Fork 280
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
Rework Tool properties architecture #1906
Open
MrStevns
wants to merge
35
commits into
pencil2d:master
Choose a base branch
from
MrStevns:task/rework-toolproperties
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Conversation
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
- Get rid of tool specific functions in ToolManager. In this version the GUI is still not setup correctly.
As the Bucket tool doesn't really make use of any of its properties.. at least not in a meaningful way.
Tools themselves from now on should handle updates, eg. camera will call updateFrame when updating showPath rather than scribblearea doing it. This creates much less coupling to individual tools.
4edbe44
to
57074fa
Compare
Which would be used for both settings related to selection tool and move tool.
57074fa
to
d7090cd
Compare
which caused me some confusion...
I believe this PR is ready to be reviewed now 🚀 |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Currently all our tool properties are exposed to BaseTool which means that whenever BaseTool is accessed from somewhere else all its properties are also "leaked". Right now we don't have a lot of properties so it's not that big of a deal but it doesn't scale that well... Also, it's been bothering so much that I decided to put my other Floating Selection PR on hold again until I found a better solution for that.
My proposal for solving this is called ToolSettings. The ToolSettings struct controls the settings of all our tool properties and allows extensibility through inheritance.
ToolSettings
ToolSettings controls the following:
The most important method to call is
load
which should be called by each respective tool as part of the initialization. The load method will then setup the ToolSettings model with an identifier, an instance of QSettings and a QHash with tool properties being added.The identifier is currently set to the name of the tool but this could easily become layer type + tool name.
To declare the properties of a tool, you must specify the min, max and default properties like so:
Here's an example what that could look like:
Internally the values are stored in a union, meaning that it only stores memory for the given type it's created with.
You might be wondering... what about the base value? we used to fetch that from QSettings, sometimes from the widget, other times from the tools
load
method. This is now handled by the ToolSettings::load method which Internally carries the responsibility of either fetching the value we have stored via QSettings or using the values we provide through the QHash above.ToolSettings::Type Key
Right now each ToolSettings must have its own
Type
enum with a start and end value. I've picked an arbitrary 100 numbers range which should be sufficient for never having to change the bounds but i'm open to other suggestions.I also thought about having a big enum like we had with
ToolPropertyType
, the naming would be something like:The biggest reason I can think of for going in that direction would be to be able to use the enum as a key rather than an integer.
If people like that better, i'm open to change it to that instead. For now though there are assertions in various places to make it easy to spot when you try to access, load or save a value on the wrong model and that's has been sufficient so far.
Setting baseValue
Updating a ToolSettings base value is done through
ToolSettings::setBaseValue
which takes in a integer as key and a PropertyInfo as value.mSettings->setBaseValue(StrokeSettings::WIDTH_VALUE, 1.0);
As opposed to
mSettings->setBaseValue(100, PropertyInfo(1.0));
I thought about creating a different ToolSetting model for all our different Stroke tools but decided against it for sake of the size of the PR. I did want to see whether my model could actually deliver on potential extensibility needs and as such created PolylineSettings to prove that specifically. So if we were to decouple say EraserTool from StrokeSettings, then following the setup of PolylineSettings would be the way to go.
Tool property name alignment
Prior to this PR, there was no naming convention for tool settings, meaning that whatever came to mind could be used a setting and there was no correlation between the setting and the tool except the name it shared eg "brush".
I've tried to align this by creating the ToolSettings::identifier. Granted, someone still has to write it but it's now tied to the specific ToolSettings struct and not some arbitrary combination of names. You won't be able to reuse the same setting for a different tool like we currently have in some places.
When the setting is saved now it will be stored in a group based on an identifier, meaning you should be able to find a toolsetting like so: brushTool/Width where brushTool is the ToolSettings identifier and /Width is the name of the specific setting identifier.
GUI Communication and Responsibility
Rather than going through ToolManager to update tool properties and requiring other listeners to know about ToolPropertyType, I decided to make a tighter coupling between a tool and it's respective tool widget. You can see the result of that in for example BucketOptionsWidget.
The only responsibility of the widget now is to add GUI elements, their labeling and things that's generally UI bound. The rest now comes from ToolSettings. The widget no longer has to nor should be allowed to update its stored/QSettings settings, nor should it set base, min or max values itself. This data now comes from the ToolSettings model and is setup prior to calling the widget and updating any property requires going through the model.
Updating a GUI elements in relation to a UI change builds on a UI -> Model -> UI update cycle. similar to MVVM but we don't have ViewModels so ignoring that part it's just:
UI -> Model
Model -> UI
UI visibility
Another thing the tool options widget no longer needs to be concerned with, is the relation between what the tool can do for the given layer type. Rather than having a fixed map of which tool properties are enabled in general, each tool now declare what propreties are
usable
as part ofBaseTool::loadSettings
like so:For the widget this means that the visibility logic becomes as simple as this:
I haven't quite decided who will be responsible for notifying about layer change and tool change updates to the UI. For now it's the main ToolOptions widget but that's still open for debate.
What's missing or still in the air: