Reinitialize libass every time the font is changed in the style editor

libass doesn't free any font data it requests until the ass_renderer is
destroyed, so recreate it every time the font is changed to avoid
holding on to extremely large amounts of data when the user quickly
switches between fonts.

Closes #1320.
This commit is contained in:
Thomas Goyne 2015-02-08 09:13:53 -08:00
parent 9f196adc2e
commit acb9b8adac
3 changed files with 20 additions and 3 deletions

View File

@ -49,6 +49,7 @@ public:
virtual ~SubtitlesProvider() = default; virtual ~SubtitlesProvider() = default;
void LoadSubtitles(AssFile *subs, int time = -1); void LoadSubtitles(AssFile *subs, int time = -1);
virtual void DrawSubtitles(VideoFrame &dst, double time)=0; virtual void DrawSubtitles(VideoFrame &dst, double time)=0;
virtual void Reinitialize() { }
}; };
namespace agi { class BackgroundRunner; } namespace agi { class BackgroundRunner; }

View File

@ -74,6 +74,9 @@ SubtitlesPreview::~SubtitlesPreview() {
} }
void SubtitlesPreview::SetStyle(AssStyle const& new_style) { void SubtitlesPreview::SetStyle(AssStyle const& new_style) {
if (provider && style->font != new_style.font)
provider->Reinitialize();
*style = new_style; *style = new_style;
style->name = "Default"; style->name = "Default";
style->alignment = 5; style->alignment = 5;

View File

@ -109,6 +109,8 @@ class LibassSubtitlesProvider final : public SubtitlesProvider {
return shared->renderer; return shared->renderer;
auto block = [&] { auto block = [&] {
if (shared->ready)
return;
br->Run([=](agi::ProgressSink *ps) { br->Run([=](agi::ProgressSink *ps) {
ps->SetTitle(from_wx(_("Updating font index"))); ps->SetTitle(from_wx(_("Updating font index")));
ps->SetMessage(from_wx(_("This may take several minutes"))); ps->SetMessage(from_wx(_("This may take several minutes")));
@ -136,6 +138,17 @@ public:
} }
void DrawSubtitles(VideoFrame &dst, double time) override; void DrawSubtitles(VideoFrame &dst, double time) override;
void Reinitialize() override {
// No need to reinit if we're not even done with the initial init
if (!shared->ready)
return;
ass_renderer_done(shared->renderer);
shared->renderer = ass_renderer_init(library);
ass_set_font_scale(shared->renderer, 1.);
ass_set_fonts(shared->renderer, nullptr, "Sans", 1, CONFIG_PATH, true);
}
}; };
LibassSubtitlesProvider::LibassSubtitlesProvider(agi::BackgroundRunner *br) LibassSubtitlesProvider::LibassSubtitlesProvider(agi::BackgroundRunner *br)