2008-01-21 04:58:14 +01:00
|
|
|
// Copyright (c) 2008, Rodrigo Braz Monteiro
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
// Website: http://aegisub.cellosoft.com
|
|
|
|
// Contact: mailto:zeratul@cellosoft.com
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
|
|
///////////
|
|
|
|
// Headers
|
|
|
|
#include "subtitle_format_dvd.h"
|
|
|
|
#include "video_provider_dummy.h"
|
|
|
|
#include "subtitles_provider.h"
|
|
|
|
#include "ass_dialogue.h"
|
|
|
|
#include "ass_file.h"
|
2008-01-21 06:46:52 +01:00
|
|
|
#ifdef _OPENMP
|
|
|
|
#include <omp.h>
|
|
|
|
#endif
|
2008-01-21 04:58:14 +01:00
|
|
|
|
|
|
|
|
|
|
|
///////////////
|
|
|
|
// Format name
|
|
|
|
wxString DVDSubtitleFormat::GetName() {
|
|
|
|
return _T("DVD Subpictures");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////
|
|
|
|
// Extensions
|
|
|
|
wxArrayString DVDSubtitleFormat::GetWriteWildcards() {
|
|
|
|
wxArrayString results;
|
|
|
|
results.Add(_T("sup"));
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////
|
|
|
|
// Can write
|
|
|
|
bool DVDSubtitleFormat::CanWriteFile(wxString filename) {
|
|
|
|
return (filename.Lower().EndsWith(_T(".sup")));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////
|
2008-01-21 06:46:52 +01:00
|
|
|
// Get subpicture list
|
|
|
|
void DVDSubtitleFormat::GetSubPictureList(std::vector<SubPicture> &pics) {
|
2008-01-21 04:58:14 +01:00
|
|
|
// Create video frame
|
|
|
|
int w = 720;
|
|
|
|
int h = 480;
|
|
|
|
VideoProvider *video = new DummyVideoProvider(10,1,w,h,wxColour(255,0,0),false);
|
2008-01-21 06:46:52 +01:00
|
|
|
AegiVideoFrame srcFrame = video->GetFrame(0);
|
2008-01-21 04:58:14 +01:00
|
|
|
delete video;
|
|
|
|
|
|
|
|
// Prepare subtitles
|
|
|
|
CreateCopy();
|
|
|
|
SortLines();
|
|
|
|
//Merge(true,true,true);
|
|
|
|
|
2008-01-21 06:46:52 +01:00
|
|
|
// Count and index lines
|
2008-01-21 04:58:14 +01:00
|
|
|
using std::list;
|
2008-01-21 06:46:52 +01:00
|
|
|
int count = 0;
|
|
|
|
std::vector<AssDialogue*> diags;
|
2008-01-21 04:58:14 +01:00
|
|
|
for (list<AssEntry*>::iterator cur=Line->begin();cur!=Line->end();cur++) {
|
|
|
|
AssDialogue *current = AssEntry::GetAsDialogue(*cur);
|
|
|
|
if (current) {
|
2008-01-21 06:46:52 +01:00
|
|
|
diags.push_back(current);
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pics.resize(count);
|
|
|
|
|
|
|
|
SubtitlesProvider *provider = NULL;
|
|
|
|
provider = SubtitlesProviderFactory::GetProvider();
|
|
|
|
provider->LoadSubtitles(GetAssFile());
|
|
|
|
|
|
|
|
// Write lines
|
|
|
|
int i;
|
|
|
|
#pragma omp parallel for shared(diags,pics,provider) private(i)
|
|
|
|
for (i=0;i<count;i++) {
|
|
|
|
// Dialogue
|
|
|
|
AssDialogue *current = diags[i];
|
|
|
|
|
|
|
|
// Get the image
|
|
|
|
AegiVideoFrame dst;
|
|
|
|
dst.CopyFrom(srcFrame);
|
2008-01-21 07:52:01 +01:00
|
|
|
double time = (current->Start.GetMS()/1000.0 + current->End.GetMS()/1000.0)/2.0;
|
2008-01-21 06:46:52 +01:00
|
|
|
#pragma omp critical
|
|
|
|
{
|
2008-01-21 07:52:01 +01:00
|
|
|
provider->DrawSubtitles(dst,time);
|
2008-01-21 06:46:52 +01:00
|
|
|
}
|
|
|
|
wxImage img = dst.GetImage();
|
|
|
|
dst.Clear();
|
|
|
|
|
|
|
|
// Perform colour reduction on image
|
|
|
|
unsigned char r,g,b;
|
|
|
|
unsigned char *data = img.GetData();
|
|
|
|
const unsigned char *dataRead = data;
|
|
|
|
unsigned char *dataWrite = data;
|
|
|
|
int startY = 0;
|
|
|
|
int endY = 0;
|
|
|
|
int startX = w;
|
|
|
|
int endX = 0;
|
|
|
|
|
|
|
|
// For each line
|
|
|
|
for (int y=h;--y>=0;) {
|
|
|
|
bool hasData = false;
|
|
|
|
int lineStartX = 0;
|
|
|
|
int lineEndX;
|
|
|
|
|
|
|
|
// Scan line
|
|
|
|
for (int x=w;--x>=0;) {
|
|
|
|
// Read
|
|
|
|
r = *(dataRead++);
|
|
|
|
g = *(dataRead++);
|
|
|
|
b = *(dataRead++);
|
|
|
|
|
|
|
|
// Background
|
|
|
|
if (r > 127 && g < 20) {
|
2008-01-21 07:52:01 +01:00
|
|
|
r = 200;
|
2008-01-21 06:46:52 +01:00
|
|
|
g = 0;
|
|
|
|
b = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Text
|
|
|
|
else {
|
|
|
|
// Mark coordinates
|
|
|
|
hasData = true;
|
|
|
|
if (lineStartX == 0) lineStartX = w-x-1;
|
|
|
|
lineEndX = w-x-1;
|
|
|
|
|
|
|
|
// Set colour
|
|
|
|
if (r > 170 && g > 170) {
|
2008-01-21 04:58:14 +01:00
|
|
|
r = 255;
|
2008-01-21 06:46:52 +01:00
|
|
|
g = 255;
|
|
|
|
b = 255;
|
|
|
|
}
|
|
|
|
else if (r > 85 && g > 85) {
|
|
|
|
r = 127;
|
|
|
|
g = 127;
|
|
|
|
b = 127;
|
2008-01-21 04:58:14 +01:00
|
|
|
}
|
|
|
|
else {
|
2008-01-21 06:46:52 +01:00
|
|
|
r = 0;
|
|
|
|
g = 0;
|
|
|
|
b = 0;
|
2008-01-21 04:58:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-21 06:46:52 +01:00
|
|
|
// Write
|
|
|
|
*(dataWrite++) = r;
|
|
|
|
*(dataWrite++) = g;
|
|
|
|
*(dataWrite++) = b;
|
2008-01-21 04:58:14 +01:00
|
|
|
}
|
|
|
|
|
2008-01-21 06:46:52 +01:00
|
|
|
// Mark as last found so far
|
|
|
|
if (hasData) {
|
|
|
|
if (startY == 0) startY = h-y-1;
|
|
|
|
endY = h-y-1;
|
|
|
|
if (lineStartX < startX) startX = lineStartX;
|
|
|
|
if (lineEndX > endX) endX = lineEndX;
|
|
|
|
}
|
2008-01-21 04:58:14 +01:00
|
|
|
}
|
2008-01-21 06:46:52 +01:00
|
|
|
|
|
|
|
// Save image
|
|
|
|
if (startX > endX) endX = startX;
|
|
|
|
if (startY > endY) endY = startY;
|
2008-01-21 07:52:01 +01:00
|
|
|
int sw = endX-startX+1;
|
|
|
|
int sh = endY-startY+1;
|
2008-01-21 06:46:52 +01:00
|
|
|
pics[i].x = startX;
|
|
|
|
pics[i].y = startY;
|
2008-01-21 07:52:01 +01:00
|
|
|
pics[i].w = sw;
|
|
|
|
pics[i].h = sh;
|
|
|
|
|
|
|
|
// RLE to memory
|
|
|
|
for (int j=0;j<2;j++) {
|
|
|
|
int curCol = -1;
|
|
|
|
int col;
|
|
|
|
int temp;
|
|
|
|
int len = 0;
|
|
|
|
dataRead = data + ((startY+j)*w+startX)*3;
|
|
|
|
std::vector<RLEGroup> groups;
|
|
|
|
groups.reserve(1024);
|
|
|
|
|
|
|
|
// Read this scanline
|
|
|
|
for (int y=startY+j;y<=endY;y+=2) {
|
|
|
|
for (int x=startX;x<=endX;x++) {
|
|
|
|
// Read current pixel colour
|
|
|
|
temp = *dataRead;
|
|
|
|
if (temp == 200) col = 0;
|
|
|
|
else if (temp == 255) col = 1;
|
|
|
|
else if (temp == 0) col = 2;
|
|
|
|
else col = 3;
|
|
|
|
|
|
|
|
// See if it matches
|
|
|
|
if (col == curCol) {
|
|
|
|
len++;
|
|
|
|
if (len == 255) {
|
|
|
|
if (len) groups.push_back(RLEGroup(curCol,len,false));
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (len) groups.push_back(RLEGroup(curCol,len,false));
|
|
|
|
len = 1;
|
|
|
|
curCol = col;
|
|
|
|
}
|
|
|
|
|
|
|
|
dataRead += 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flush
|
|
|
|
if (len) groups.push_back(RLEGroup(curCol,0,true));
|
|
|
|
else {
|
|
|
|
groups.back().len = 0;
|
|
|
|
groups.back().eol = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Advance
|
|
|
|
dataRead += (2*w-sw)*3;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encode into subpicture format
|
|
|
|
int nibble[4];
|
|
|
|
nibble[3] = 0;
|
|
|
|
bool off = false;
|
|
|
|
std::vector<unsigned char> &data = pics[i].data[j];
|
|
|
|
unsigned char last = 0;
|
|
|
|
for (size_t m=0;m<groups.size();m++) {
|
|
|
|
int nibbles;
|
|
|
|
if (groups[m].eol) nibbles = 4;
|
|
|
|
else nibbles = groups[m].len >> 2;
|
|
|
|
nibble[0] = groups[m].col | ((groups[m].len & 0x3) << 2);
|
|
|
|
nibble[1] = (groups[m].len & 0x3C) >> 2;
|
|
|
|
nibble[2] = (groups[m].len & 0xC0) >> 6;
|
|
|
|
for (int n=0;n<nibbles;n++) {
|
|
|
|
if (!off) {
|
|
|
|
last = nibble[nibbles-n-1] << 4;
|
|
|
|
data.push_back(last);
|
|
|
|
}
|
|
|
|
else data.back() = data.back() | last;
|
|
|
|
off = !off;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
data.resize(data.size());
|
|
|
|
}
|
2008-01-21 04:58:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Clean up
|
|
|
|
delete provider;
|
|
|
|
srcFrame.Clear();
|
|
|
|
}
|
2008-01-21 06:46:52 +01:00
|
|
|
|
|
|
|
|
|
|
|
///////////////////////
|
|
|
|
// Actually write them
|
|
|
|
void DVDSubtitleFormat::WriteFile(wxString filename,wxString encoding) {
|
|
|
|
// Get subpictures
|
|
|
|
std::vector<SubPicture> pics;
|
|
|
|
GetSubPictureList(pics);
|
|
|
|
}
|