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

Add ability to change background dim colour in grayscale #30391

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

Conversation

Uncomfy
Copy link

@Uncomfy Uncomfy commented Oct 22, 2024

Previously discussed here: #12983

Depends on:

Allows to change which colour the background is dimmed to. Currently only supports grayscale.

This is not a full implementation, as it doesn't support storyboards and videos.

The main reason I'm asking for comments so early is that this implementation handles dimming in two different classes, which leads to code duplication, so I wonder if there is a cleaner way to do this.

osu_bg_dim_colour.mp4

Here are also a few examples of how it could look with colour picker instead of a grayscale slider, using SettingsColour, but I'm unsure if it fits the design, as colour pickers weren't used in Settings/Visual Settings before. Also it doesn't work properly in Visual Settings yet, because it doesn't have a PopoverContainer in its hierarchy, and I'm not really sure how to handle it.

settings
settings_colour_picker_open
visual_settings_1

Uncomfy added 30 commits October 9, 2024 18:43
…avoid confusion with existing BeatmapBackground class
…ed BeatmapBackground to DimmableBeatmapBackground in BackgroundScreenBeatmap
@jhk2601
Copy link
Contributor

jhk2601 commented Oct 22, 2024

Just a quick thing, "0%-100%" doesn't really make sense in the context of a slider changing how light or dark the dim is. This would be fixed in the color picker implementation anyway though (which I personally think makes more sense, though maybe hiding the UI element behind an option that's off by default?)

[BackgroundDependencyLoader]
private void load(ShaderManager shaders)
{
TextureShader = shaders.Load(VertexShaderDescriptor.TEXTURE_2, "BeatmapBackground");
Copy link
Member

@frenzibyte frenzibyte Oct 22, 2024

Choose a reason for hiding this comment

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

Isn't TextureShader responsible for rendering the framebuffer texture of the buffered container? If so, isn't that pointless? Isn't it enough to apply the custom "BeatmapBackground" shader on the underlying beatmap background sprite? Would that cause issues with blurring? I can't imagine it will.

In other words, do Sprite.TextureShader = "BeatmapBackground" instead (alongside the extra logic for the uniform buffer). That will save exposing TextureShader framework-side and also possibly "save" shader computation because the overall effect will become buffered (don't quote me on this).

Copy link
Author

Choose a reason for hiding this comment

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

Yes, but it will require the redraw of the framebuffer texture on each DimLevel change, which will also force the blur to be redrawn, and I feared that this might introduce unnecessary overhead during (for example) breaks, if player has background blur enabled. If that is not a concern - yeah, keeping the custom shader only on the sprite makes complete sense.

Copy link
Author

Choose a reason for hiding this comment

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

You're right, it's pointless, at least in the current implementation.
Changing the dim parameters requires DrawNode invalidation, which forces the framebuffer to be redrawn, so there are no benefits. This does mean that DimLevel transitions are way more expensive now (at least on my rather old and slow hardware).

Here are the results of my benchmarks for average frametime during different parts of the beatmap.
With user blur set to 0%:

Gameplay Breaks Fade in Fade out
Baseline 4.25 ms 4.29 ms 4.42 ms 4.08 ms
Coloured dim 4.26 ms 4.33 ms 4.92 ms 4.55 ms
Relative increase 0.2% 0.9% 11.3% 11.5%

With user blur set to 100%:

Gameplay Breaks Fade in Fade out
Baseline 4.20 ms 4.24 ms 4.39 ms 4.02 ms
Coloured dim 4.22 ms 4.30 ms 5.18 ms 4.66 ms
Relative increase 0.5% 1.4% 18% 15.9%

"Baseline" is osu! 2024.1009.1
"Coloured dim" is commit c5867b6

Details about the benchmark

Made a map with 64 gameplay sections and 63 breaks for this benchmark, total duration is 3:09.

"Fade in" refers to frames rendered during gameplay-to-break transition, "Fade out" refers to break-to-gameplay transition.
"Gameplay" and "Breaks" columns are just for sanity-checking, as they should be same.

Total amount of recorded frames:

Blur Gameplay Breaks Fade in Fade out
Baseline 0% 12068 8176 11018 12069
Baseline 100% 12193 8253 11089 12236
Coloured dim 0% 12090 8105 9810 10689
Coloured dim 100% 12154 8137 9448 10512

osu! settings:

  • Compiled in release mode
  • Single-threaded mode
  • "OpenGL (Experimental)" renderer
  • Windowed screen mode
  • Resolution: 1564x964
  • Frame limiter: Basically unlimited

System info:

  • OS: Kubuntu 24.04.1 LTS x86_64
  • CPU: Intel i3-6100U @ 2.300GHz
  • GPU: Intel HD Graphics 520
  • RAM: 19907MiB

So I guess now there are two ways to go - ditch the shader replacement on the BufferedContainer and have slower transitions, or modify the BufferedContainer somehow to avoid redrawing the framebuffer when shader parameters are changed.

Which one is more preferable?

Copy link
Member

Choose a reason for hiding this comment

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

Fair enough, this is a bit of a pesky situation. You can control whether the framebuffer gets redrawn by overriding GetDrawVersion() and make that point to a custom invalidation ID that is normally incremented but not when dim properties are changed. See Path in osu!framework, it might actually be doing exactly what you're looking for.

Copy link
Author

Choose a reason for hiding this comment

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

Added a test case for it, but solving it requires a change in the framework: ppy/osu-framework#6404

With that change applied benchmarks look a lot better.

With user blur set to 0%:

Implementation Gameplay Breaks Fade in Fade out
Baseline 4.15 ms 4.20 ms 4.34 ms 3.99 ms
Coloured dim 4.19 ms 4.23 ms 4.39 ms 4.00 ms
Relative increase 1.0% 0.7% 1.2% 0.3%

With user blur set to 100%:

Implementation Gameplay Breaks Fade in Fade out
Baseline 4.11 ms 4.16 ms 4.31 ms 3.93 ms
Coloured dim 4.05 ms 4.11 ms 4.25 ms 3.85 ms
Relative increase -1.5% -1.2% -1.4% -2.0%

"Baseline" is osu! 2024.1009.1
"Coloured dim" is commit 93c3fac with ppy/osu-framework#6404 applied

Copy link
Member

Choose a reason for hiding this comment

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

I don't think these benchmark numbers represent the true impact because as far as I read you're measuring over a long period of time, while when the blur re-renders it's going to be very detrimental only to the local frames (so the overall "increase" is much smaller than the actual perceived difference to the user).

Of course, as long as the numbers are going down this doesn't matter so much, so carry on I guess.

A better way of benchmarking would be to force the redraw every frame then just take a reading of the average frame time.

Copy link
Author

@Uncomfy Uncomfy Nov 14, 2024

Choose a reason for hiding this comment

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

I've double checked, and without ppy/osu-framework#6404 my implementation triggered a blur redraw on every frame during transition, but didn't affect frames during gameplay or break.
This means that the stats computed in the "Fade in" and "Fade out" columns only take into account frames where redraw happened.

A better way of benchmarking would be to force the redraw every frame then just take a reading of the average frame time.

My goal was to show that extra redraws were triggered and that it affected the performance during specific parts of the beatmap, so I couldn't use this approach

@Uncomfy
Copy link
Author

Uncomfy commented Oct 22, 2024

Just a quick thing, "0%-100%" doesn't really make sense in the context of a slider changing how light or dark the dim is. This would be fixed in the color picker implementation anyway though (which I personally think makes more sense, though maybe hiding the UI element behind an option that's off by default?)

I'm not sure what would be better than 0%-100% for a slider. Or do you mean ditching the slider completely?
Would renaming the setting to background dim brightness make more sense?

@jhk2601
Copy link
Contributor

jhk2601 commented Oct 23, 2024

Just a quick thing, "0%-100%" doesn't really make sense in the context of a slider changing how light or dark the dim is. This would be fixed in the color picker implementation anyway though (which I personally think makes more sense, though maybe hiding the UI element behind an option that's off by default?)

I'm not sure what would be better than 0%-100% for a slider. Or do you mean ditching the slider completely? Would renaming the setting to background dim brightness make more sense?

yes that would be better, if I saw a slider labeled "background dim color" and it was a percent I wouldn't know what the number represented, "brightness" works better.

@peppy
Copy link
Member

peppy commented Oct 23, 2024

It might make more sense to rename "Background dim" to "Background opacity" and have 100% be a fully visible image. Although this will go against user expectations coming from stable so maybe it's too late to make a change like this.

@Uncomfy
Copy link
Author

Uncomfy commented Nov 12, 2024

I wonder if it is fine to just add an overlay (a quad with Colour == DimColour, Alpha == DimLevel and a size that covers an entire background) to achieve same effect for storyboards/videos.
It's a way simpler solution than changing shaders of all of the osu.Game.Storyboards.Drawables. It might cause performance to drop a bit, though I'm unsure how noticeable it would be with storyboards/videos enabled.

@frenzibyte
Copy link
Member

I can only understand doing this for storyboards since they are not buffered (and I would question how it's supposed to behave, because blending the entire storyboard behind some "background dim colour" overlay might end up looking silly). For videos, this can still totally be done by shaders with some changes framework-side.

@Uncomfy
Copy link
Author

Uncomfy commented Nov 12, 2024

Currently coloured dimming in the shader is done using a mix function (mix(texel, m_DimColour, m_DimLevel)), which can be expanded as texel * (1 - m_DimLevel) + m_DimColour * m_DimLevel

Since all colours of all storyboard drawables are already multiplied by 1 - DimLevel, the entire effect can be reproduced by just adding an overlay quad with additive blending and Colour == DimColour && Alpha == DimLevel

Also I just realized that in the mix function I'm also interpolating alpha, need to fix that.

@frenzibyte
Copy link
Member

"Just adding an overlay quad with..." is exactly the 1x overhead problem that we would like to avoid here by doing better.

@Uncomfy
Copy link
Author

Uncomfy commented Nov 13, 2024

Yeah, true
I just wondered if it's going to be fine at least with storyboards, given that they're pretty heavy already (though it depends on individual storyboards), so the performance hit might not be as noticeable
But ok, no shortcuts

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants