Aegisub/aegilib/src/formats/format_ass_style.cpp

246 lines
6.3 KiB
C++
Raw Normal View History

// 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/GORGONSUB
//
// Website: http://www.aegisub.net
// Contact: mailto:amz@aegisub.net
//
#include "format_ass.h"
#include "tokenizer.h"
#include "utils.h"
using namespace Gorgonsub;
////////////////
// Constructors
StyleASS::StyleASS()
{
formatVersion = 1;
}
StyleASS::StyleASS(String data,int version)
{
// Try parsing with all different versions
bool valid = false;
for (int count=0;!valid && count < 3;count++) {
valid = Parse(data,version);
version++;
if (version > 2) version = 0;
}
if (!valid) throw Exception(Exception::Parse_Error);
}
/////////
// Parse
bool StyleASS::Parse(String data,int version)
{
try {
// Tokenize
wxString temp;
Tokenizer tkn(data.Trim(false).Mid(6),_T(","));
// Read name, font name and size
name = tkn.GetString(true);
font = tkn.GetString(true);
fontSize = tkn.GetFloat();
// Read colours
for (int i=0;i<5;i++) {
if ((i == 4 && version == 0) || (i == 2 && version != 0)) colour[i] = colour[i-1];
else colour[i].Parse(tkn.GetString(),true);
}
// Read bold and italics
bold = tkn.GetBool();
italic = tkn.GetBool();
// ASS
if (version != 0) {
// Read underline, strikeout, scale, spacing and angle
underline = tkn.GetBool();
strikeout = tkn.GetBool();
scalex = tkn.GetFloat();
scaley = tkn.GetFloat();
spacing = tkn.GetFloat();
angle = tkn.GetFloat();
}
else {
// SSA defaults
underline = false;
strikeout = false;
scalex = 100;
scaley = 100;
spacing = 0;
angle = 0.0;
}
// Read border style and widths
borderStyle = tkn.GetInt();
outline_w = tkn.GetFloat();
shadow_w = tkn.GetFloat();
// Read alignment
alignment = tkn.GetInt();
if (version == 0) alignment = AlignSSAtoASS(alignment);
// Read margins
for (int i=0;i<4;i++) {
if (i == 3 && version < 2) margin[i] = margin[i-1];
else margin[i] = tkn.GetInt();
}
// Read and discard alpha level on SSA
// TODO: do something with this?
if (version == 0) tkn.GetString();
// Read encoding
encoding = tkn.GetInt();
// Read relative to
relativeTo = 0;
if (version == 2) relativeTo = tkn.GetInt();
// Read it all?
if (tkn.HasMore()) return false;
// Done
formatVersion = version;
return true;
}
catch (...) {
return false;
}
}
////////////////////////////////
// Convert SSA alignment to ASS
int StyleASS::AlignSSAtoASS(int align) const
{
switch(align) {
case 1: return 1;
case 2: return 2;
case 3: return 3;
case 5: return 7;
case 6: return 8;
case 7: return 9;
case 9: return 4;
case 10: return 5;
case 11: return 6;
default: return 2;
}
}
////////////////////////////////
// Convert ASS alignment to SSA
int StyleASS::AlignASStoSSA(int align) const
{
switch (align) {
case 1: return 1;
case 2: return 2;
case 3: return 3;
case 4: return 9;
case 5: return 10;
case 6: return 11;
case 7: return 5;
case 8: return 6;
case 9: return 7;
default: return 2;
}
}
/////////////
// Serialize
String StyleASS::ToText(int version) const
{
// Final string
wxString final;
// Calculate colour offset
int cOff = 0;
if (version >= 1) cOff = 1;
// Calculate alignment
int align = alignment;
if (version == 0) align = AlignASStoSSA(align);
// Name, font, fontsize, colours, bold, italics
final = wxString::Format(_T("Style: %s,%s,%s,%s,%s,%s,%s,%i,%i,"),
name.c_str(), font.c_str(), PrettyFloatD(fontSize).c_str(),
colour[0].GetVBHex(true,true,false).c_str(), colour[1].GetVBHex(true,true,false).c_str(),
colour[2+cOff].GetVBHex(true,true,false).c_str(), colour[3+cOff].GetVBHex(true,true,false).c_str(),
(bold? -1 : 0), (italic ? -1 : 0));
// ASS-only
final += wxString::Format(_T("%i,%i,%s,%s,%s,%s,"),
(underline?-1:0),(strikeout?-1:0),PrettyFloatD(scalex).c_str(),PrettyFloatD(scaley).c_str(),
PrettyFloatD(spacing).c_str(),PrettyFloatD(angle).c_str());
// Borderstyle, outline width, shadow width, alignment, first three margins
final += wxString::Format(_T("%i,%s,%s,%i,%i,%i,%i,"),
borderStyle,PrettyFloatD(outline_w).c_str(),PrettyFloatD(shadow_w).c_str(),
align,margin[0],margin[1],margin[2]);
// Fourth margin for ASS2 only
if (version == 2) final += wxString::Format(_T("%i,"),margin[3]);
// Alpha level for SSA only
// TODO: write something relevant?
if (version == 0) final += wxString::Format(_T("%i,"),0);
// Encoding
final += wxString::Format(_T("%i"),encoding);
// Relative-to for ASS2 only
if (version == 2) final += wxString::Format(_T(",%i"),relativeTo);
// Done
return final;
}
/////////////////////
// Get default group
String StyleASS::GetDefaultGroup() const
{
switch (formatVersion) {
case 0: return L"V4 Events";
case 1: return L"V4+ Events";
default: return L"V4++ Events";
}
}