// Copyright (c) 2006, 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 // /////////// // Headers #include "subtitle_format_prs.h" #include "ass_file.h" #include "ass_dialogue.h" #include "avisynth_wrap.h" #include "video_box.h" #include "video_display.h" #include "video_provider.h" #include "main.h" #include "frame_main.h" #include "vfr.h" #include "../prs/prs_file.h" #include "../prs/prs_image.h" #include "../prs/prs_display.h" #include "../prs/prs_vsfilter_reader.h" ////////////////////// // Can write to file? bool PRSSubtitleFormat::CanWriteFile(wxString filename) { #ifdef __WINDOWS__ return (filename.Right(4).Lower() == _T(".prs")); #else return false; #endif } ////////////// // Write file void PRSSubtitleFormat::WriteFile(wxString filename,wxString encoding) { #ifdef __WINDOWS__ // Video loaded? VideoDisplay *display = ((AegisubApp*)wxTheApp)->frame->videoBox->videoDisplay; // Create file PRSFile file; // Open two Avisynth environments AviSynthWrapper avs1,avs2; IScriptEnvironment *env1 = avs1.GetEnv(); IScriptEnvironment *env2 = avs2.GetEnv(); // Prepare environments wxString val = wxString::Format(_T("BlankClip(pixel_type=\"RGB32\",length=%i,width=%i,height=%i,fps=%f"),display->provider->GetFrameCount(),display->provider->GetWidth(),display->provider->GetHeight(),display->provider->GetFPS()); AVSValue script1 = env1->Invoke("Eval",AVSValue(wxString(val + _T(",color=$000000)")).mb_str(wxConvUTF8))); AVSValue script2 = env2->Invoke("Eval",AVSValue(wxString(val + _T(",color=$FFFFFF)")).mb_str(wxConvUTF8))); char temp[512]; strcpy(temp,display->GetTempWorkFile().mb_str(wxConvLocal)); AVSValue args1[2] = { script1.AsClip(), temp }; AVSValue args2[2] = { script2.AsClip(), temp }; try { script1 = env1->Invoke("TextSub", AVSValue(args1,2)); script2 = env2->Invoke("TextSub", AVSValue(args2,2)); } catch (AvisynthError &err) { throw _T("AviSynth error: ") + wxString(err.msg,wxConvLocal); } // Loop through subtitles in file AssFile *ass = AssFile::top; AssDialogue *diag = NULL; PClip clip1 = script1.AsClip(); PClip clip2 = script1.AsClip(); int id = 0; for (entryIter cur=ass->Line.begin();cur!=ass->Line.end();cur++) { diag = AssEntry::GetAsDialogue(*cur); // Dialogue found if (diag) { // Read its image int framen = VFR_Output.GetFrameAtTime(diag->Start.GetMS(),true); PVideoFrame frame1 = clip1->GetFrame(framen,env1); PVideoFrame frame2 = clip2->GetFrame(framen,env2); // Create PRSImage PRSImage *img = new PRSImage; img->id = id; img->dataLen = 0; img->data = NULL; // Create PRSDisplay PRSDisplay *display = new PRSDisplay; display->start = diag->Start.GetMS(); display->end = diag->End.GetMS(); display->id = id; display->x = 0; display->y = 0; display->alpha = 255; display->blend = 0; // Insert into list file.AddEntry(img); file.AddEntry(display); id++; } } // Save file file.Save(filename.mb_str(wxConvLocal)); #endif }