225 lines
7.1 KiB
C++
225 lines
7.1 KiB
C++
// Copyright (c) 2006-2007, Rodrigo Braz Monteiro, Evgeniy Stepanov
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright notice,
|
|
// this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above copyright notice,
|
|
// this list of conditions and the following disclaimer in the documentation
|
|
// and/or other materials provided with the distribution.
|
|
// * Neither the name of the Aegisub Group nor the names of its contributors
|
|
// may be used to endorse or promote products derived from this software
|
|
// without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
// POSSIBILITY OF SUCH DAMAGE.
|
|
//
|
|
// Aegisub Project http://www.aegisub.org/
|
|
|
|
/// @file subtitles_provider_libass.cpp
|
|
/// @brief libass-based subtitle renderer
|
|
/// @ingroup subtitle_rendering
|
|
///
|
|
|
|
#include "subtitles_provider_libass.h"
|
|
|
|
#include "compat.h"
|
|
#include "include/aegisub/subtitles_provider.h"
|
|
#include "video_frame.h"
|
|
|
|
#include <libaegisub/background_runner.h>
|
|
#include <libaegisub/dispatch.h>
|
|
#include <libaegisub/exception.h>
|
|
#include <libaegisub/log.h>
|
|
#include <libaegisub/make_unique.h>
|
|
#include <libaegisub/util.h>
|
|
|
|
#include <atomic>
|
|
#include <boost/gil/gil_all.hpp>
|
|
#include <memory>
|
|
#include <mutex>
|
|
|
|
#include <wx/intl.h>
|
|
#include <wx/thread.h>
|
|
|
|
extern "C" {
|
|
#include <ass/ass.h>
|
|
}
|
|
|
|
namespace {
|
|
std::unique_ptr<agi::dispatch::Queue> cache_queue;
|
|
ASS_Library *library;
|
|
|
|
void msg_callback(int level, const char *fmt, va_list args, void *) {
|
|
if (level >= 7) return;
|
|
char buf[1024];
|
|
#ifdef _WIN32
|
|
vsprintf_s(buf, sizeof(buf), fmt, args);
|
|
#else
|
|
vsnprintf(buf, sizeof(buf), fmt, args);
|
|
#endif
|
|
|
|
if (level < 2) // warning/error
|
|
LOG_I("subtitle/provider/libass") << buf;
|
|
else // verbose
|
|
LOG_D("subtitle/provider/libass") << buf;
|
|
}
|
|
|
|
// Stuff used on the cache thread, owned by a shared_ptr in case the provider
|
|
// gets deleted before the cache finishing updating
|
|
struct cache_thread_shared {
|
|
ASS_Renderer *renderer = nullptr;
|
|
std::atomic<bool> ready{false};
|
|
~cache_thread_shared() { if (renderer) ass_renderer_done(renderer); }
|
|
};
|
|
|
|
class LibassSubtitlesProvider final : public SubtitlesProvider {
|
|
agi::BackgroundRunner *br;
|
|
std::shared_ptr<cache_thread_shared> shared;
|
|
ASS_Track* ass_track = nullptr;
|
|
|
|
ASS_Renderer *renderer() {
|
|
if (shared->ready)
|
|
return shared->renderer;
|
|
|
|
auto block = [&] {
|
|
if (shared->ready)
|
|
return;
|
|
agi::util::sleep_for(250);
|
|
if (shared->ready)
|
|
return;
|
|
br->Run([=](agi::ProgressSink *ps) {
|
|
ps->SetTitle(from_wx(_("Updating font index")));
|
|
ps->SetMessage(from_wx(_("This may take several minutes")));
|
|
ps->SetIndeterminate();
|
|
while (!shared->ready && !ps->IsCancelled())
|
|
agi::util::sleep_for(250);
|
|
});
|
|
};
|
|
|
|
if (wxThread::IsMain())
|
|
block();
|
|
else
|
|
agi::dispatch::Main().Sync(block);
|
|
return shared->renderer;
|
|
}
|
|
|
|
public:
|
|
LibassSubtitlesProvider(agi::BackgroundRunner *br);
|
|
~LibassSubtitlesProvider();
|
|
|
|
void LoadSubtitles(const char *data, size_t len) override {
|
|
if (ass_track) ass_free_track(ass_track);
|
|
ass_track = ass_read_memory(library, const_cast<char *>(data), len, nullptr);
|
|
if (!ass_track) throw agi::InternalError("libass failed to load subtitles.");
|
|
}
|
|
|
|
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, nullptr, true);
|
|
}
|
|
};
|
|
|
|
LibassSubtitlesProvider::LibassSubtitlesProvider(agi::BackgroundRunner *br)
|
|
: br(br)
|
|
, shared(std::make_shared<cache_thread_shared>())
|
|
{
|
|
auto state = shared;
|
|
cache_queue->Async([state] {
|
|
auto ass_renderer = ass_renderer_init(library);
|
|
if (ass_renderer) {
|
|
ass_set_font_scale(ass_renderer, 1.);
|
|
ass_set_fonts(ass_renderer, nullptr, "Sans", 1, nullptr, true);
|
|
}
|
|
state->renderer = ass_renderer;
|
|
state->ready = true;
|
|
});
|
|
}
|
|
|
|
LibassSubtitlesProvider::~LibassSubtitlesProvider() {
|
|
if (ass_track) ass_free_track(ass_track);
|
|
}
|
|
|
|
#define _r(c) ((c)>>24)
|
|
#define _g(c) (((c)>>16)&0xFF)
|
|
#define _b(c) (((c)>>8)&0xFF)
|
|
#define _a(c) ((c)&0xFF)
|
|
|
|
void LibassSubtitlesProvider::DrawSubtitles(VideoFrame &frame,double time) {
|
|
ass_set_frame_size(renderer(), frame.width, frame.height);
|
|
|
|
ASS_Image* img = ass_render_frame(renderer(), ass_track, int(time * 1000), nullptr);
|
|
|
|
// libass actually returns several alpha-masked monochrome images.
|
|
// Here, we loop through their linked list, get the colour of the current, and blend into the frame.
|
|
// This is repeated for all of them.
|
|
|
|
using namespace boost::gil;
|
|
auto dst = interleaved_view(frame.width, frame.height, (bgra8_pixel_t*)frame.data.data(), frame.width * 4);
|
|
if (frame.flipped)
|
|
dst = flipped_up_down_view(dst);
|
|
|
|
for (; img; img = img->next) {
|
|
unsigned int opacity = 255 - ((unsigned int)_a(img->color));
|
|
unsigned int r = (unsigned int)_r(img->color);
|
|
unsigned int g = (unsigned int)_g(img->color);
|
|
unsigned int b = (unsigned int)_b(img->color);
|
|
|
|
auto srcview = interleaved_view(img->w, img->h, (gray8_pixel_t*)img->bitmap, img->stride);
|
|
auto dstview = subimage_view(dst, img->dst_x, img->dst_y, img->w, img->h);
|
|
|
|
transform_pixels(dstview, srcview, dstview, [=](const bgra8_pixel_t frame, const gray8_pixel_t src) -> bgra8_pixel_t {
|
|
unsigned int k = ((unsigned)src) * opacity / 255;
|
|
unsigned int ck = 255 - k;
|
|
|
|
bgra8_pixel_t ret;
|
|
ret[0] = (k * b + ck * frame[0]) / 255;
|
|
ret[1] = (k * g + ck * frame[1]) / 255;
|
|
ret[2] = (k * r + ck * frame[2]) / 255;
|
|
ret[3] = 0;
|
|
return ret;
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
namespace libass {
|
|
std::unique_ptr<SubtitlesProvider> Create(std::string const&, agi::BackgroundRunner *br) {
|
|
return agi::make_unique<LibassSubtitlesProvider>(br);
|
|
}
|
|
|
|
void CacheFonts() {
|
|
// Initialize the cache worker thread
|
|
cache_queue = agi::dispatch::Create();
|
|
|
|
// Initialize libass
|
|
library = ass_library_init();
|
|
ass_set_message_cb(library, msg_callback, nullptr);
|
|
|
|
// Initialize a renderer to force fontconfig to update its cache
|
|
cache_queue->Async([] {
|
|
auto ass_renderer = ass_renderer_init(library);
|
|
ass_set_fonts(ass_renderer, nullptr, "Sans", 1, nullptr, true);
|
|
ass_renderer_done(ass_renderer);
|
|
});
|
|
}
|
|
}
|