i was wondering if there is support or if its planned to make it possible to have materials or somehow can use shader for the Background images in Visual Elements…
i came up with a soultion but this one is kind of “Static”
public UIDocument uiDoc;
public VisualElement imageHolder;
public Material material;
public Texture2D src;
public Texture2D tex;
// Start is called before the first frame update
IEnumerator Start()
{
imageHolder = uiDoc.rootVisualElement.Q<VisualElement>("image");
tex = new Texture2D(512,512);
imageHolder.style.backgroundImage = new StyleBackground(tex);
float t = 0;
while (true)
{
t += Time.deltaTime;
material.SetFloat("_Grey", Mathf.Sin(t));
RenderTexture r = new RenderTexture(512, 512, 16);
Graphics.Blit(src, r, material);
RenderTexture.active = r;
tex.ReadPixels(new Rect(0, 0, r.width, r.height), 0, 0);
tex.Apply();
yield return new WaitForSeconds(1);
}
}
because everytime (in this exampe every second) i update the texture the visual element flickers (it seems to try to resize?!) and if i put the blit in the update the visual element basically stays in flickermode.
Any Ideas or Information when this will be possible?
I think the flashing you see is currently being worked on, but we have a workaround if it is the case.
The flashing you see is because UI toolkit uses the active RenderTexture to do its layout. You will need to back it up at the beginning of your redraw cycle and restor it afterwards :
var backup = RenderTexture.active
...
RenderTexture.active = backup
Let me know if this solves your problem.
Then, I am going to take the liberty of doing a few other comments regarding your code:
You don’t need to use a Texture2d. The ReadPixels is expensive as it needs to synchronize with the graphic card to read back to the CPU what is on the active render texture. If you don’t need to manually edit or save the image, it can stay just on the GPU by assigning the RenderTexture to the style.
Graphic.blit set RenderTexture.active to the destination and leave it like this. So event if you remove the tex2d, you still have to do backup and restore workaround.
Keep in mind that RenderTexutre and Texture2d have a native resource associated. To have a clear lifecycle, I would add a private field containing the generated textures and destroy it with the object in OnDestroy for example.
To finish answering your question, there is currently no way to specify a custom shader and for your use case I think you are better with the way you are handling things.
Don’t hesitate to ask for clarification or post back your code if you want more feedback!
oh thanks for the detailed Answer and explaining how it works behind the scenes! I’ll try this later today, because this one actually is regrading my private side project ;D … And thank you really for the code review … i mean this was just a quick and dirty proof of concept type of code from my side and some of your comments i already had in mind but i actually didn’t realize that i could assign the render texture to the style directly!
Okay thx again … with the backup and restore workaround the flicker is gone !
But what i dont get is how i would get rid of the texture2D ? … i cannot assign the render texture directly to the style ?!
new StyleBackground(tex);
StyleBackground only takes Tex2D or Vectorimage?! am i missing something here?
How did you manage to set it up? The StyleBackground does not take Image, only it’s .vectorImage field, but the texture is assigned to the .image field. I’m kinda confused here.
Would you mind sharing your solution? I can’t seem to make this work. Thanks for any info!
Nevermind, I figured it out. If anyone is as confused as me, the talk is about Image - an Element.
Despite explicitly stated, because it is not available directly through builder, I thought they meant the Image Component.
Nothing has changed in this regard. Unlocking this kind of scenarios is important to us, and we will get there eventually, but it likely won’t be through custom shaders or materials, at least not the way it’s done like the UGUI system.
@SimonDufour Hi, a while back I followed this thread to display a shader/material in the Image Element and it worked perfectly. After some time and the editor upgrade (currently using 2022.1 beta), this stopped working. No errors, but the shader does not display. Has anything significant changed that invalidates this approach? Thanks for any info!
m_Texture2D = Texture2D.blackTexture;
m_RankWidgetRenderTexture = new RenderTexture(200, 200, 24, RenderTextureFormat.ARGB32);
WidgetMaterialImage.image = m_RankWidgetRenderTexture;
m_RankWidgetMaterial = new Material(materialReference);
Graphics.Blit(m_Texture2D, m_RankWidgetRenderTexture, m_RankWidgetMaterial);
Nothing has changed regarding this. But I have seen some change related to the camera in HDRP.
Can you verify that the result that you are expecting is written in the rendertexture?
If you are able to provide us (by doing a bug report in the editor) with a project working in a previous version of unity that does not work in a newer version of the editor, we would investigate what is happening.
I am not very familiar with RenderTexture and Graphics.Blit, so I might be doing something wrong, however for sure it used to work (don’t remember the exact Unity version). I tested the result texture (had to use style.background = Background.FromRenderTexture(…)) so that the created texture would show up in the UITK Debugger. But the texture looks like a solid blue square.
So I tested in a new URP project (same editor version - 2022.1.0b3) - created just a simple version without UITK.
public class Tester_RenderTextureWidget : MonoBehaviour
{
public RenderTexture targetRenderTexture;
public Material materialReference;
[ReadOnly] public Material createdMaterial;
[Button]
public void TestRenderTexture()
{
var _Texture2D = Texture2D.redTexture;
createdMaterial = new Material(materialReference);
Graphics.Blit(_Texture2D, targetRenderTexture, createdMaterial);
// Graphics.Blit(_Texture2D, targetRenderTexture);
}
}
The result is the same: the texture turns blue (viewed only through inspector of the RT asset). If using the commented line, the texture turns red, as expected. The shader and the material look OK.
@SimonDufour Hi, just checking whether there was any development on your side concerning this issue. Has the attached test project helped you identify any leads? Do you have a suggestion how should I improve or change the test?
Thanks for any info!
No progress yet, I had a few errors with odin + URP, so I left the tab open and figure I would come to it later.
Usually the forum provide the quicker feedback, but we are more systematic with the bug reported in the editor. (Help menu->report a bug) We also have more feedback on the version you are using.
Based on your comments, it seems like the issue is with URP as UI toolkit has no involvement with the raw call to the following lines:
You can check that the targetRenderTexture contain the expected result by selecting the asset in the project browser without any involvement of UI toolkit.
I am thinking maybe the default Graphics.Blit use a material/shader incompatible with URP. But this is outside my area of expertise.
If you can do a repro project containing only the minimum to reproduce (I think a source texture, a c# script and destination texture?) and report in the editor with clear reproduction steps, you should have an answer.
Any news on using custom materials for the background image? Basically I have some icons that match the player colors in a local multi-player game, and I am using materials made with the AllIn1SpriteShader asset which lets me have 1 sprite and change one (or even more) of the colors of a sprite for another one of my choice (the player colors). And would be glad to use it in UIToolkit too. Currently it is critical for my project. Otherwise would need to go back to the old-school ui system. Repeating assets with different colors is off the table, and I can’t just create a white asset because it has multiple colors on it