SFML community forums

Help => Graphics => Topic started by: GetterSetter on April 27, 2024, 02:28:16 pm

Title: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: GetterSetter on April 27, 2024, 02:28:16 pm
The problem: I have a set of textures, but they are not of the sf::Texture type. However, sometimes I need to set them to sf::RectangleShape and render them. The idea is as follows:
sf::RenderTexture rt;
rt.create(...);
rt.clear();
for(auto &texture:non_sfml_textures)
{
  sf::Texture t=convertion_function(texture);
  sf::RectangleShape rect;
  rect.setTexture(&t);

  //Setting position of rect, etc...

  rt.draw(rect);
}
rt.display();
 
Is it safe to do so? By 'safe' I mean that all the textures will be rendered correctly after calling rt.disaplay().
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: eXpl0it3r on April 27, 2024, 03:44:21 pm
No, textures have to exist for as long as you use them and a "use" counts until you call display().
For anything else, SFML doesn't provide any guarantees.
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: GetterSetter on April 27, 2024, 06:27:55 pm
Hm, what if put rt.display() inside the loop and don't clear rt? Does it have any side effects? From my perspective, it looks like all the rendered rectangles will just stack in the texture, and that's definitely what I want, but I'm not sure if it works as I expect. No, it doesn't work like this.
The problem is that I don't want to create a whole set of texture copies just to render them, because it requires a lot of memory.
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: eXpl0it3r on April 28, 2024, 10:34:04 pm
What other kind of textures do you have anyways?

Creating a texture on the fly isn't a cheap operation.
If it's an OpenGL texture, you could also just bind it, instead of creating a copy.
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: GetterSetter on April 29, 2024, 06:55:39 am
I have tgui::Textures. Would you mind explaining to me, please, how I can use the bind function, because it's not clear from the documentation?
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: eXpl0it3r on April 29, 2024, 08:26:56 am
If you have tgui::Texture can't you just call texture.getData().backendTexture->getInternalTexture() which should return the sf::Texture instance?

https://tgui.eu/documentation/latest-stable/classtgui_1_1Texture.html#a749d82057ff2ac73060ad1c6bbd8b970
https://tgui.eu/documentation/latest-stable/TextureData_8hpp_source.html
https://tgui.eu/documentation/latest-stable/classtgui_1_1BackendTextureSFML.html
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: texus on April 29, 2024, 08:33:53 am
If you need a texture that you use for drawing in SFML and you also use for TGUI then I actually recommend loading it twice (once as sf::Texture and once as tgui::Texture), because TGUI makes no guarantees about how its texture is stored internally and whether it will be compatible with sf::Texture.

That said, you can get access to the internal SFML texture with the methods that eXpl0it3r mentioned:
const sf::Texture* pTex = &std::static_pointer_cast<tgui::BackendTextureSFML>(texture.getData()->backendTexture)->getInternalTexture();
rect.setTexture(pTex);
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: GetterSetter on April 29, 2024, 10:46:15 am
Thank you, texus and eXpl0it3r! The only question I have now is whether we should use dynamic_pointer_cast instead of static_pointer_cast because we are performing a downcast.
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: texus on April 29, 2024, 12:40:43 pm
dynamic_pointer_cast will return a nullptr if the backend texture is of a different type, while static_pointer_cast assumes the type is correct and has undefined behavior if the backend texture is of a different type.

As long as you use the SFML_GRAPHICS backend, you can be certain that the backend texture always has type BackendTextureSFML, so the cast will always succeed and a static_pointer_cast is fine. Also, there is no point in using dynamic_pointer_cast if you aren't going to check whether the return value is a nullptr or not.
Title: Re: Is it safe to overwrite sf::Texture after calling render_texture.draw()?
Post by: GetterSetter on April 29, 2024, 01:11:30 pm
Got it, thank you!