3a069b7f60
Remove some unused or constant arguments and simplify some overly convoluted logic. Check for whether timecodes are open rather than whether video is open to determine if by-frame mode is enabled. Operate on a project context rather than using VideoContext::Get(). Use non-event-generating setter methods rather than a boolean ready variable. Make all member variables private and add setters rather than relying on the client code calling Update when appropriate. Eliminate flickering in overwrite mode. Originally committed to SVN as r6056.
911 lines
29 KiB
C++
911 lines
29 KiB
C++
// Copyright (c) 2005, Rodrigo Braz Monteiro
|
|
// Copyright (c) 2010, Thomas Goyne <plorkyeran@aegisub.org>
|
|
// 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/
|
|
//
|
|
// $Id$
|
|
|
|
/// @file subs_edit_box.cpp
|
|
/// @brief Main subtitle editing area, including toolbars around the text control
|
|
/// @ingroup main_ui
|
|
|
|
#include "config.h"
|
|
|
|
#ifndef AGI_PRE
|
|
#ifdef _WIN32
|
|
#include <functional>
|
|
#else
|
|
#include <tr1/functional>
|
|
#endif
|
|
|
|
#include <wx/button.h>
|
|
#include <wx/checkbox.h>
|
|
#include <wx/colordlg.h>
|
|
#include <wx/combobox.h>
|
|
#include <wx/dcclient.h>
|
|
#include <wx/dcmemory.h>
|
|
#include <wx/fontdlg.h>
|
|
#include <wx/radiobut.h>
|
|
#include <wx/spinctrl.h>
|
|
#endif
|
|
|
|
#include "include/aegisub/hotkey.h"
|
|
|
|
#include "ass_dialogue.h"
|
|
#include "ass_file.h"
|
|
#include "ass_override.h"
|
|
#include "ass_style.h"
|
|
#include "audio_controller.h"
|
|
#include "dialog_colorpicker.h"
|
|
#include "dialog_search_replace.h"
|
|
#include "include/aegisub/context.h"
|
|
#include "libresrc/libresrc.h"
|
|
#include "main.h"
|
|
#include "subs_edit_box.h"
|
|
#include "subs_edit_ctrl.h"
|
|
#include "subs_grid.h"
|
|
#include "timeedit_ctrl.h"
|
|
#include "tooltip_manager.h"
|
|
#include "utils.h"
|
|
#include "validators.h"
|
|
#include "video_context.h"
|
|
|
|
enum {
|
|
BUTTON_BOLD = 1300,
|
|
BUTTON_ITALICS,
|
|
BUTTON_UNDERLINE,
|
|
BUTTON_STRIKEOUT,
|
|
BUTTON_FONT_NAME,
|
|
BUTTON_COLOR1,
|
|
BUTTON_COLOR2,
|
|
BUTTON_COLOR3,
|
|
BUTTON_COLOR4,
|
|
BUTTON_COMMIT,
|
|
BUTTON_LAST
|
|
};
|
|
enum {
|
|
BUTTON_FIRST = BUTTON_BOLD
|
|
};
|
|
|
|
template<class T>
|
|
struct field_setter : public std::binary_function<AssDialogue*, T, void> {
|
|
T AssDialogue::*field;
|
|
field_setter(T AssDialogue::*field) : field(field) { }
|
|
void operator()(AssDialogue* obj, T value) {
|
|
obj->*field = value;
|
|
}
|
|
};
|
|
|
|
/// @brief Get the selection from a text edit
|
|
/// @param[out] start Beginning of selection
|
|
/// @param[out] end End of selection
|
|
void get_selection(SubsTextEditCtrl *TextEdit, int &start, int &end) {
|
|
TextEdit->GetSelection(&start, &end);
|
|
int len = TextEdit->GetText().size();
|
|
start = mid(0,TextEdit->GetReverseUnicodePosition(start),len);
|
|
end = mid(0,TextEdit->GetReverseUnicodePosition(end),len);
|
|
}
|
|
|
|
/// @brief Get the value of a tag at a specified position in a line
|
|
/// @param line Line to get the value from
|
|
/// @param blockn Block number in the line
|
|
/// @param initial Value from style to use if the tag does not exist
|
|
/// @param tag Tag to get the value of
|
|
/// @param alt Alternate name of the tag, if any
|
|
template<class T>
|
|
static T get_value(AssDialogue const& line, int blockn, T initial, wxString tag, wxString alt = "") {
|
|
for (int i = blockn; i >= 0; i--) {
|
|
AssDialogueBlockOverride *ovr = dynamic_cast<AssDialogueBlockOverride*>(line.Blocks[i]);
|
|
if (!ovr) continue;
|
|
|
|
for (int j = (int)ovr->Tags.size() - 1; j >= 0; j--) {
|
|
if (ovr->Tags[j]->Name == tag || ovr->Tags[j]->Name == alt) {
|
|
return ovr->Tags[j]->Params[0]->Get<T>(initial);
|
|
}
|
|
}
|
|
}
|
|
return initial;
|
|
}
|
|
|
|
template<class Control>
|
|
struct FocusHandler : std::unary_function<wxFocusEvent &, void> {
|
|
wxString value;
|
|
wxString alt;
|
|
wxColor color;
|
|
Control *control;
|
|
void operator()(wxFocusEvent &event) const {
|
|
event.Skip();
|
|
|
|
if (control->GetValue() == alt) {
|
|
control->Freeze();
|
|
control->ChangeValue(value);
|
|
control->SetForegroundColour(color);
|
|
control->Thaw();
|
|
}
|
|
}
|
|
};
|
|
|
|
template<class Event, class T>
|
|
void bind_focus_handler(T *control, Event event, wxString value, wxString alt, wxColor color) {
|
|
FocusHandler<T> handler;
|
|
handler.value = value;
|
|
handler.alt = alt;
|
|
handler.color = color;
|
|
handler.control = control;
|
|
control->Bind(event, handler);
|
|
}
|
|
|
|
SubsEditBox::SubsEditBox(wxWindow *parent, agi::Context *context)
|
|
: wxPanel(parent, -1, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL | wxRAISED_BORDER, "SubsEditBox")
|
|
, line(NULL)
|
|
, splitLineMode(false)
|
|
, controlState(true)
|
|
, c(context)
|
|
{
|
|
// Top controls
|
|
wxArrayString styles;
|
|
styles.Add("Default");
|
|
CommentBox = new wxCheckBox(this,wxID_ANY,_("&Comment"));
|
|
StyleBox = new wxComboBox(this,wxID_ANY,"Default",wxDefaultPosition,wxSize(110,-1),styles,wxCB_READONLY | wxTE_PROCESS_ENTER);
|
|
ActorBox = new wxComboBox(this,wxID_ANY,"Actor",wxDefaultPosition,wxSize(110,-1),styles,wxCB_DROPDOWN | wxTE_PROCESS_ENTER);
|
|
Effect = new wxTextCtrl(this,wxID_ANY,"",wxDefaultPosition,wxSize(80,-1),wxTE_PROCESS_ENTER);
|
|
|
|
// Middle controls
|
|
Layer = new wxSpinCtrl(this,wxID_ANY,"",wxDefaultPosition,wxSize(50,-1),wxSP_ARROW_KEYS,0,0x7FFFFFFF,0);
|
|
StartTime = new TimeEdit(this, wxID_ANY, context, "", wxSize(75,-1));
|
|
EndTime = new TimeEdit(this, wxID_ANY, context, "", wxSize(75,-1), true);
|
|
Duration = new TimeEdit(this,wxID_ANY, context,"",wxSize(75,-1));
|
|
MarginL = new wxTextCtrl(this,wxID_ANY,"",wxDefaultPosition,wxSize(40,-1),wxTE_CENTRE | wxTE_PROCESS_ENTER,NumValidator());
|
|
MarginL->SetMaxLength(4);
|
|
MarginR = new wxTextCtrl(this,wxID_ANY,"",wxDefaultPosition,wxSize(40,-1),wxTE_CENTRE | wxTE_PROCESS_ENTER,NumValidator());
|
|
MarginR->SetMaxLength(4);
|
|
MarginV = new wxTextCtrl(this,wxID_ANY,"",wxDefaultPosition,wxSize(40,-1),wxTE_CENTRE | wxTE_PROCESS_ENTER,NumValidator());
|
|
MarginV->SetMaxLength(4);
|
|
|
|
// Middle-bottom controls
|
|
ToggableButtons.reserve(10);
|
|
int id = BUTTON_FIRST;
|
|
#define MAKE_BUTTON(img, tooltip) \
|
|
ToggableButtons.push_back(new wxBitmapButton(this, id++, GETIMAGE(img))); \
|
|
ToggableButtons.back()->SetToolTip(tooltip);
|
|
|
|
MAKE_BUTTON(button_bold_16, _("Bold"));
|
|
MAKE_BUTTON(button_italics_16, _("Italics"));
|
|
MAKE_BUTTON(button_underline_16, _("Underline"));
|
|
MAKE_BUTTON(button_strikeout_16, _("Strikeout"));
|
|
MAKE_BUTTON(button_fontname_16, _("Font Face"));
|
|
MAKE_BUTTON(button_color_one_16, _("Primary color"));
|
|
MAKE_BUTTON(button_color_two_16, _("Secondary color"));
|
|
MAKE_BUTTON(button_color_three_16, _("Outline color"));
|
|
MAKE_BUTTON(button_color_four_16, _("Shadow color"));
|
|
MAKE_BUTTON(button_audio_commit_16, _("Commits the text (Enter)"));
|
|
#undef MAKE_BUTTON
|
|
|
|
ByTime = new wxRadioButton(this,wxID_ANY,_("&Time"),wxDefaultPosition,wxDefaultSize,wxRB_GROUP);
|
|
ByFrame = new wxRadioButton(this,wxID_ANY,_("F&rame"));
|
|
ByFrame->Enable(false);
|
|
|
|
// Tooltips
|
|
CommentBox->SetToolTip(_("Comment this line out. Commented lines don't show up on screen."));
|
|
StyleBox->SetToolTip(_("Style for this line."));
|
|
ActorBox->SetToolTip(_("Actor name for this speech. This is only for reference, and is mainly useless."));
|
|
Effect->SetToolTip(_("Effect for this line. This can be used to store extra information for karaoke scripts, or for the effects supported by the renderer."));
|
|
Layer->SetToolTip(_("Layer number"));
|
|
StartTime->SetToolTip(_("Start time"));
|
|
EndTime->SetToolTip(_("End time"));
|
|
Duration->SetToolTip(_("Line duration"));
|
|
MarginL->SetToolTip(_("Left Margin (0 = default)"));
|
|
MarginR->SetToolTip(_("Right Margin (0 = default)"));
|
|
MarginV->SetToolTip(_("Vertical Margin (0 = default)"));
|
|
ByTime->SetToolTip(_("Time by h:mm:ss.cs"));
|
|
ByFrame->SetToolTip(_("Time by frame number"));
|
|
|
|
// Top sizer
|
|
TopSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
TopSizer->Add(CommentBox,0,wxRIGHT | wxALIGN_CENTER,5);
|
|
TopSizer->Add(StyleBox,2,wxRIGHT|wxALIGN_CENTER,5);
|
|
TopSizer->Add(ActorBox,2,wxRIGHT|wxALIGN_CENTER,5);
|
|
TopSizer->Add(Effect,3,wxALIGN_CENTER,5);
|
|
|
|
// Middle sizer
|
|
splitLineMode = true;
|
|
MiddleSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
MiddleSizer->Add(Layer,0,wxRIGHT|wxALIGN_CENTER,5);
|
|
MiddleSizer->Add(StartTime,0,wxRIGHT|wxALIGN_CENTER,0);
|
|
MiddleSizer->Add(EndTime,0,wxRIGHT|wxALIGN_CENTER,5);
|
|
MiddleSizer->Add(Duration,0,wxRIGHT|wxALIGN_CENTER,5);
|
|
MiddleSizer->Add(MarginL,0,wxALIGN_CENTER,0);
|
|
MiddleSizer->Add(MarginR,0,wxALIGN_CENTER,0);
|
|
MiddleSizer->Add(MarginV,0,wxALIGN_CENTER,0);
|
|
MiddleSizer->AddSpacer(5);
|
|
|
|
// Middle-bottom sizer
|
|
MiddleBotSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
for (size_t i = 0; i < ToggableButtons.size(); ++i) {
|
|
MiddleBotSizer->Add(ToggableButtons[i],0,wxALIGN_CENTER|wxEXPAND,0);
|
|
if (i == 4 || i == 8)
|
|
MiddleBotSizer->AddSpacer(5);
|
|
}
|
|
MiddleBotSizer->AddSpacer(10);
|
|
MiddleBotSizer->Add(ByTime,0,wxRIGHT | wxALIGN_CENTER | wxEXPAND,5);
|
|
MiddleBotSizer->Add(ByFrame,0,wxRIGHT | wxALIGN_CENTER | wxEXPAND,5);
|
|
|
|
// Text editor
|
|
TextEdit = new SubsTextEditCtrl(this, wxSize(300,50), wxBORDER_SUNKEN, c->subsGrid);
|
|
TextEdit->Bind(wxEVT_KEY_DOWN, &SubsEditBox::OnKeyDown, this);
|
|
TextEdit->SetUndoCollection(false);
|
|
BottomSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
BottomSizer->Add(TextEdit,1,wxEXPAND,0);
|
|
|
|
// Main sizer
|
|
MainSizer = new wxBoxSizer(wxVERTICAL);
|
|
MainSizer->Add(TopSizer,0,wxEXPAND | wxALL,3);
|
|
MainSizer->Add(MiddleSizer,0,wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM,3);
|
|
MainSizer->Add(MiddleBotSizer,0,wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM,3);
|
|
MainSizer->Add(BottomSizer,1,wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM,3);
|
|
|
|
// Set sizer
|
|
SetSizer(MainSizer);
|
|
MainSizer->SetSizeHints(this);
|
|
|
|
origBgColour = TextEdit->GetBackgroundColour();
|
|
disabledBgColour = GetBackgroundColour();
|
|
|
|
wxColor text = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT);
|
|
wxColor grey((text.Red() + origBgColour.Red()) / 2,
|
|
(text.Green() + origBgColour.Green()) / 2,
|
|
(text.Blue() + origBgColour.Blue()) / 2);
|
|
|
|
// Setup placeholders for effect and actor boxes
|
|
bind_focus_handler(Effect, wxEVT_SET_FOCUS, "", "Effect", text);
|
|
bind_focus_handler(Effect, wxEVT_KILL_FOCUS, "Effect", "", grey);
|
|
Effect->SetForegroundColour(grey);
|
|
|
|
bind_focus_handler(ActorBox, wxEVT_SET_FOCUS, "", "Actor", text);
|
|
bind_focus_handler(ActorBox, wxEVT_KILL_FOCUS, "Actor", "", grey);
|
|
ActorBox->SetForegroundColour(grey);
|
|
|
|
TextEdit->Bind(wxEVT_STC_MODIFIED, &SubsEditBox::OnChange, this);
|
|
TextEdit->SetModEventMask(wxSTC_MOD_INSERTTEXT | wxSTC_MOD_DELETETEXT);
|
|
|
|
Bind(wxEVT_COMMAND_RADIOBUTTON_SELECTED, &SubsEditBox::OnFrameTimeRadio, this, ByFrame->GetId());
|
|
Bind(wxEVT_COMMAND_RADIOBUTTON_SELECTED, &SubsEditBox::OnFrameTimeRadio, this, ByTime->GetId());
|
|
|
|
Bind(wxEVT_COMMAND_COMBOBOX_SELECTED, &SubsEditBox::OnStyleChange, this, StyleBox->GetId());
|
|
Bind(wxEVT_COMMAND_COMBOBOX_SELECTED, &SubsEditBox::OnActorChange, this, ActorBox->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnActorChange, this, ActorBox->GetId());
|
|
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnLayerEnter, this, Layer->GetId());
|
|
Bind(wxEVT_COMMAND_SPINCTRL_UPDATED, &SubsEditBox::OnLayerChange, this, Layer->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnStartTimeChange, this, StartTime->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnEndTimeChange, this, EndTime->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnDurationChange, this, Duration->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnMarginLChange, this, MarginL->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnMarginRChange, this, MarginR->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnMarginVChange, this, MarginV->GetId());
|
|
Bind(wxEVT_COMMAND_TEXT_UPDATED, &SubsEditBox::OnEffectChange, this, Effect->GetId());
|
|
Bind(wxEVT_COMMAND_CHECKBOX_CLICKED, &SubsEditBox::OnCommentChange, this, CommentBox->GetId());
|
|
|
|
Bind(wxEVT_SIZE, &SubsEditBox::OnSize, this);
|
|
|
|
for (int i = 0; i < 4; i++) {
|
|
Bind(wxEVT_COMMAND_BUTTON_CLICKED, &SubsEditBox::OnFlagButton, this, BUTTON_FIRST + i);
|
|
}
|
|
Bind(wxEVT_COMMAND_BUTTON_CLICKED, &SubsEditBox::OnFontButton, this, BUTTON_FONT_NAME);
|
|
for (int i = 5; i < 9; i++) {
|
|
Bind(wxEVT_COMMAND_BUTTON_CLICKED, &SubsEditBox::OnColorButton, this, BUTTON_FIRST + i);
|
|
}
|
|
Bind(wxEVT_COMMAND_BUTTON_CLICKED, &SubsEditBox::OnCommitButton, this, BUTTON_COMMIT);
|
|
|
|
wxSizeEvent evt;
|
|
OnSize(evt);
|
|
|
|
c->selectionController->AddSelectionListener(this);
|
|
file_changed_slot = c->ass->AddCommitListener(&SubsEditBox::Update, this);
|
|
context->videoController->AddTimecodesListener(&SubsEditBox::UpdateFrameTiming, this);
|
|
}
|
|
SubsEditBox::~SubsEditBox() {
|
|
c->selectionController->RemoveSelectionListener(this);
|
|
}
|
|
|
|
void SubsEditBox::Update(int type) {
|
|
SetEvtHandlerEnabled(false);
|
|
|
|
if (type == AssFile::COMMIT_NEW || type & AssFile::COMMIT_STYLES) {
|
|
StyleBox->Clear();
|
|
StyleBox->Append(c->ass->GetStyles());
|
|
}
|
|
|
|
if (type == AssFile::COMMIT_NEW) {
|
|
/// @todo maybe preserve selection over undo?
|
|
PopulateActorList();
|
|
|
|
TextEdit->SetSelection(0,0);
|
|
SetEvtHandlerEnabled(true);
|
|
return;
|
|
}
|
|
else if (type & AssFile::COMMIT_STYLES)
|
|
StyleBox->Select(StyleBox->FindString(line->Style));
|
|
|
|
if (!(type ^ AssFile::COMMIT_ORDER)) return;
|
|
|
|
SetControlsState(!!line);
|
|
if (!line) {
|
|
SetEvtHandlerEnabled(true);
|
|
return;
|
|
}
|
|
|
|
if (type & AssFile::COMMIT_DIAG_TIME) {
|
|
StartTime->SetTime(line->Start);
|
|
EndTime->SetTime(line->End);
|
|
Duration->SetTime(line->End-line->Start);
|
|
}
|
|
|
|
if (type & AssFile::COMMIT_DIAG_TEXT) {
|
|
TextEdit->SetTextTo(line->Text);
|
|
}
|
|
|
|
if (type & AssFile::COMMIT_DIAG_META) {
|
|
Layer->SetValue(line->Layer);
|
|
MarginL->ChangeValue(line->GetMarginString(0,false));
|
|
MarginR->ChangeValue(line->GetMarginString(1,false));
|
|
MarginV->ChangeValue(line->GetMarginString(2,false));
|
|
Effect->ChangeValue(line->Effect.empty() ? "Effect" : line->Effect);
|
|
CommentBox->SetValue(line->Comment);
|
|
StyleBox->Select(StyleBox->FindString(line->Style));
|
|
|
|
PopulateActorList();
|
|
ActorBox->ChangeValue(line->Actor.empty() ? "Actor" : line->Actor);
|
|
ActorBox->SetStringSelection(line->Actor);
|
|
}
|
|
|
|
SetEvtHandlerEnabled(true);
|
|
}
|
|
|
|
void SubsEditBox::PopulateActorList() {
|
|
std::set<wxString> actors;
|
|
for (entryIter it = c->ass->Line.begin(); it != c->ass->Line.end(); ++it) {
|
|
if (AssDialogue *diag = dynamic_cast<AssDialogue*>(*it))
|
|
actors.insert(diag->Actor);
|
|
}
|
|
#ifdef __APPLE__
|
|
// OSX doesn't like combo boxes that are empty.
|
|
actors.insert("Actor");
|
|
#endif
|
|
actors.erase("");
|
|
wxArrayString arrstr;
|
|
arrstr.reserve(actors.size());
|
|
copy(actors.begin(), actors.end(), std::back_inserter(arrstr));
|
|
|
|
ActorBox->Freeze();
|
|
bool evt_handler_was_enabled = GetEvtHandlerEnabled();
|
|
SetEvtHandlerEnabled(false);
|
|
long pos = ActorBox->GetInsertionPoint();
|
|
wxString value = ActorBox->GetValue();
|
|
|
|
ActorBox->Clear();
|
|
ActorBox->Append(arrstr);
|
|
ActorBox->ChangeValue(value);
|
|
ActorBox->SetStringSelection(value);
|
|
ActorBox->SetInsertionPoint(pos);
|
|
|
|
if (evt_handler_was_enabled)
|
|
SetEvtHandlerEnabled(true);
|
|
ActorBox->Thaw();
|
|
}
|
|
|
|
void SubsEditBox::OnActiveLineChanged(AssDialogue *new_line) {
|
|
SetEvtHandlerEnabled(false);
|
|
line = new_line;
|
|
|
|
Update(AssFile::COMMIT_DIAG_FULL);
|
|
|
|
/// @todo VideoContext should be doing this
|
|
if (c->videoController->IsLoaded()) {
|
|
bool sync;
|
|
if (Search.HasFocus()) sync = OPT_GET("Tool/Search Replace/Video Update")->GetBool();
|
|
else sync = OPT_GET("Video/Subtitle Sync")->GetBool();
|
|
|
|
if (sync) {
|
|
c->videoController->Stop();
|
|
c->videoController->JumpToTime(line->Start.GetMS());
|
|
}
|
|
}
|
|
SetEvtHandlerEnabled(true);
|
|
}
|
|
void SubsEditBox::OnSelectedSetChanged(const Selection &, const Selection &) {
|
|
sel = c->selectionController->GetSelectedSet();
|
|
}
|
|
|
|
void SubsEditBox::UpdateFrameTiming(agi::vfr::Framerate const& fps) {
|
|
if (fps.IsLoaded()) {
|
|
ByFrame->Enable(true);
|
|
}
|
|
else {
|
|
ByFrame->Enable(false);
|
|
ByTime->SetValue(true);
|
|
StartTime->SetByFrame(false);
|
|
EndTime->SetByFrame(false);
|
|
c->subsGrid->SetByFrame(false);
|
|
}
|
|
}
|
|
|
|
void SubsEditBox::OnKeyDown(wxKeyEvent &event) {
|
|
if (hotkey::check("Subtitle Edit Box", c, event.GetKeyCode(), event.GetUnicodeKey(), event.GetModifiers()))
|
|
return;
|
|
|
|
int key = event.GetKeyCode();
|
|
if (line && (key == WXK_RETURN || key == WXK_NUMPAD_ENTER)) {
|
|
NextLine();
|
|
}
|
|
else {
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
void SubsEditBox::OnCommitButton(wxCommandEvent &) {
|
|
if (line) NextLine();
|
|
}
|
|
|
|
void SubsEditBox::NextLine() {
|
|
AssDialogue *cur = line;
|
|
c->selectionController->NextLine();
|
|
if (line == cur) {
|
|
AssDialogue *newline = new AssDialogue;
|
|
newline->Start = cur->End;
|
|
newline->End = cur->End + OPT_GET("Timing/Default Duration")->GetInt();
|
|
newline->Style = cur->Style;
|
|
|
|
entryIter pos = find(c->ass->Line.begin(), c->ass->Line.end(), line);
|
|
c->ass->Line.insert(++pos, newline);
|
|
c->ass->Commit(_("line insertion"), AssFile::COMMIT_DIAG_ADDREM);
|
|
c->selectionController->NextLine();
|
|
}
|
|
}
|
|
|
|
void SubsEditBox::OnChange(wxStyledTextEvent &event) {
|
|
if (line && TextEdit->GetText() != line->Text) {
|
|
if (event.GetModificationType() & wxSTC_MOD_INSERTTEXT) {
|
|
CommitText(_("insert text"));
|
|
}
|
|
else {
|
|
CommitText(_("delete text"));
|
|
}
|
|
}
|
|
}
|
|
|
|
template<class T, class setter>
|
|
void SubsEditBox::SetSelectedRows(setter set, T value, wxString desc, int type, bool amend) {
|
|
for_each(sel.begin(), sel.end(), bind(set, std::tr1::placeholders::_1, value));
|
|
|
|
file_changed_slot.Block();
|
|
commitId = c->ass->Commit(desc, type, (amend && desc == lastCommitType) ? commitId : -1, sel.size() == 1 ? *sel.begin() : 0);
|
|
file_changed_slot.Unblock();
|
|
lastCommitType = desc;
|
|
}
|
|
|
|
template<class T>
|
|
void SubsEditBox::SetSelectedRows(T AssDialogue::*field, T value, wxString desc, int type, bool amend) {
|
|
SetSelectedRows(field_setter<T>(field), value, desc, type, amend);
|
|
}
|
|
|
|
void SubsEditBox::CommitText(wxString desc) {
|
|
SetSelectedRows(&AssDialogue::Text, TextEdit->GetText(), desc, AssFile::COMMIT_DIAG_TEXT, true);
|
|
}
|
|
|
|
void SubsEditBox::CommitTimes(TimeField field) {
|
|
Duration->SetTime(EndTime->GetTime() - StartTime->GetTime());
|
|
|
|
// Update lines
|
|
for (Selection::iterator cur = sel.begin(); cur != sel.end(); ++cur) {
|
|
switch (field) {
|
|
case TIME_START:
|
|
(*cur)->Start = StartTime->GetTime();
|
|
if ((*cur)->Start > (*cur)->End) (*cur)->End = (*cur)->Start;
|
|
break;
|
|
case TIME_END:
|
|
(*cur)->End = EndTime->GetTime();
|
|
if ((*cur)->Start > (*cur)->End) (*cur)->Start = (*cur)->End;
|
|
break;
|
|
case TIME_DURATION:
|
|
(*cur)->End = (*cur)->Start + Duration->GetTime();
|
|
break;
|
|
}
|
|
}
|
|
|
|
timeCommitId[field] = c->ass->Commit(_("modify times"), AssFile::COMMIT_DIAG_TIME, timeCommitId[field], sel.size() == 1 ? *sel.begin() : 0);
|
|
}
|
|
|
|
void SubsEditBox::OnSize(wxSizeEvent &evt) {
|
|
int topWidth = TopSizer->GetSize().GetWidth();
|
|
int midMin = MiddleSizer->GetMinSize().GetWidth();
|
|
int botMin = MiddleBotSizer->GetMinSize().GetWidth();
|
|
|
|
if (splitLineMode) {
|
|
if (topWidth >= midMin + botMin) {
|
|
MainSizer->Detach(MiddleBotSizer);
|
|
MiddleSizer->Add(MiddleBotSizer,0,wxALIGN_CENTER_VERTICAL);
|
|
splitLineMode = false;
|
|
}
|
|
}
|
|
else {
|
|
if (topWidth < midMin) {
|
|
MiddleSizer->Detach(MiddleBotSizer);
|
|
MainSizer->Insert(2,MiddleBotSizer,0,wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM,3);
|
|
splitLineMode = true;
|
|
}
|
|
}
|
|
|
|
evt.Skip();
|
|
}
|
|
|
|
void SubsEditBox::OnFrameTimeRadio(wxCommandEvent &event) {
|
|
bool byFrame = ByFrame->GetValue();
|
|
StartTime->SetByFrame(byFrame);
|
|
EndTime->SetByFrame(byFrame);
|
|
Duration->SetByFrame(byFrame);
|
|
c->subsGrid->SetByFrame(byFrame);
|
|
event.Skip();
|
|
}
|
|
|
|
void SubsEditBox::SetControlsState(bool state) {
|
|
if (state == controlState) return;
|
|
controlState = state;
|
|
|
|
// HACK: TextEdit workaround the stupid colour lock bug
|
|
TextEdit->SetReadOnly(!state);
|
|
if (state) TextEdit->SetBackgroundColour(origBgColour);
|
|
else TextEdit->SetBackgroundColour(disabledBgColour);
|
|
|
|
// Sets controls
|
|
StartTime->Enable(state);
|
|
EndTime->Enable(state);
|
|
Duration->Enable(state);
|
|
Layer->Enable(state);
|
|
MarginL->Enable(state);
|
|
MarginR->Enable(state);
|
|
MarginV->Enable(state);
|
|
Effect->Enable(state);
|
|
CommentBox->Enable(state);
|
|
StyleBox->Enable(state);
|
|
ActorBox->Enable(state);
|
|
ByTime->Enable(state);
|
|
for (size_t i = 0; i < ToggableButtons.size(); ++i)
|
|
ToggableButtons[i]->Enable(state);
|
|
|
|
if (!state) {
|
|
SetEvtHandlerEnabled(false);
|
|
TextEdit->SetTextTo("");
|
|
StartTime->SetTime(0);
|
|
EndTime->SetTime(0);
|
|
Duration->SetTime(0);
|
|
Layer->SetValue("");
|
|
MarginL->ChangeValue("");
|
|
MarginR->ChangeValue("");
|
|
MarginV->ChangeValue("");
|
|
Effect->ChangeValue("");
|
|
CommentBox->SetValue(false);
|
|
SetEvtHandlerEnabled(true);
|
|
}
|
|
}
|
|
|
|
|
|
void SubsEditBox::OnStyleChange(wxCommandEvent &) {
|
|
SetSelectedRows(&AssDialogue::Style, StyleBox->GetValue(), _("style change"), AssFile::COMMIT_DIAG_META);
|
|
}
|
|
|
|
void SubsEditBox::OnActorChange(wxCommandEvent &) {
|
|
SetSelectedRows(&AssDialogue::Actor, ActorBox->GetValue(), _("actor change"), AssFile::COMMIT_DIAG_META);
|
|
PopulateActorList();
|
|
}
|
|
|
|
void SubsEditBox::OnLayerChange(wxSpinEvent &event) {
|
|
OnLayerEnter(event);
|
|
}
|
|
|
|
void SubsEditBox::OnLayerEnter(wxCommandEvent &) {
|
|
SetSelectedRows(&AssDialogue::Layer, Layer->GetValue(), _("layer change"), AssFile::COMMIT_DIAG_META);
|
|
}
|
|
|
|
void SubsEditBox::OnStartTimeChange(wxCommandEvent &) {
|
|
if (StartTime->GetTime() > EndTime->GetTime()) EndTime->SetTime(StartTime->GetTime());
|
|
CommitTimes(TIME_START);
|
|
}
|
|
|
|
void SubsEditBox::OnEndTimeChange(wxCommandEvent &) {
|
|
if (StartTime->GetTime() > EndTime->GetTime()) StartTime->SetTime(EndTime->GetTime());
|
|
CommitTimes(TIME_END);
|
|
}
|
|
|
|
void SubsEditBox::OnDurationChange(wxCommandEvent &) {
|
|
EndTime->SetTime(StartTime->GetTime() + Duration->GetTime());
|
|
CommitTimes(TIME_DURATION);
|
|
}
|
|
void SubsEditBox::OnMarginLChange(wxCommandEvent &) {
|
|
SetSelectedRows(std::mem_fun(&AssDialogue::SetMarginString<0>), MarginL->GetValue(), _("MarginL change"), AssFile::COMMIT_DIAG_META);
|
|
if (line) MarginL->ChangeValue(line->GetMarginString(0, false));
|
|
}
|
|
|
|
void SubsEditBox::OnMarginRChange(wxCommandEvent &) {
|
|
SetSelectedRows(std::mem_fun(&AssDialogue::SetMarginString<1>), MarginR->GetValue(), _("MarginR change"), AssFile::COMMIT_DIAG_META);
|
|
if (line) MarginR->ChangeValue(line->GetMarginString(1, false));
|
|
}
|
|
|
|
static void set_margin_v(AssDialogue* diag, wxString value) {
|
|
diag->SetMarginString(value, 2);
|
|
diag->SetMarginString(value, 3);
|
|
}
|
|
|
|
void SubsEditBox::OnMarginVChange(wxCommandEvent &) {
|
|
SetSelectedRows(set_margin_v, MarginV->GetValue(), _("MarginV change"), AssFile::COMMIT_DIAG_META);
|
|
if (line) MarginV->ChangeValue(line->GetMarginString(2, false));
|
|
}
|
|
|
|
void SubsEditBox::OnEffectChange(wxCommandEvent &) {
|
|
SetSelectedRows(&AssDialogue::Effect, Effect->GetValue(), _("effect change"), AssFile::COMMIT_DIAG_META);
|
|
}
|
|
|
|
void SubsEditBox::OnCommentChange(wxCommandEvent &) {
|
|
SetSelectedRows(&AssDialogue::Comment, CommentBox->GetValue(), _("comment change"), AssFile::COMMIT_DIAG_META);
|
|
}
|
|
|
|
int SubsEditBox::BlockAtPos(wxString const& text, int pos) const {
|
|
int n = 0;
|
|
int max = text.size() - 1;
|
|
for (int i = 0; i <= pos && i <= max; ++i) {
|
|
if (i > 0 && text[i] == '{')
|
|
n++;
|
|
if (text[i] == '}' && i != max && i != pos && i != pos -1 && (i+1 == max || text[i+1] != '{'))
|
|
n++;
|
|
}
|
|
|
|
return n;
|
|
}
|
|
|
|
void SubsEditBox::SetTag(wxString tag, wxString value, bool atEnd) {
|
|
assert(line);
|
|
if (line->Blocks.empty())
|
|
line->ParseASSTags();
|
|
|
|
int selstart, selend;
|
|
get_selection(TextEdit, selstart, selend);
|
|
int start = atEnd ? selend : selstart;
|
|
int blockn = BlockAtPos(line->Text, start);
|
|
|
|
AssDialogueBlockPlain *plain = 0;
|
|
AssDialogueBlockOverride *ovr = 0;
|
|
while (blockn >= 0) {
|
|
AssDialogueBlock *block = line->Blocks[blockn];
|
|
if (dynamic_cast<AssDialogueBlockDrawing*>(block))
|
|
--blockn;
|
|
else if ((plain = dynamic_cast<AssDialogueBlockPlain*>(block))) {
|
|
// Cursor is in a comment block, so try the previous block instead
|
|
if (plain->GetText().StartsWith("{")) {
|
|
--blockn;
|
|
start = line->Text.rfind('{', start);
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
else {
|
|
ovr = dynamic_cast<AssDialogueBlockOverride*>(block);
|
|
assert(ovr);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// If we didn't hit a suitable block for inserting the override just put
|
|
// it at the beginning of the line
|
|
if (blockn < 0)
|
|
start = 0;
|
|
|
|
wxString insert = tag + value;
|
|
int shift = insert.size();
|
|
if (plain || blockn < 0) {
|
|
line->Text = line->Text.Left(start) + "{" + insert + "}" + line->Text.Mid(start);
|
|
shift += 2;
|
|
line->ParseASSTags();
|
|
}
|
|
else {
|
|
wxString alt;
|
|
if (tag == "\\c") alt = "\\1c";
|
|
// Remove old of same
|
|
bool found = false;
|
|
for (size_t i = 0; i < ovr->Tags.size(); i++) {
|
|
wxString name = ovr->Tags[i]->Name;
|
|
if (tag == name || alt == name) {
|
|
shift -= ((wxString)*ovr->Tags[i]).size();
|
|
if (found) {
|
|
delete ovr->Tags[i];
|
|
ovr->Tags.erase(ovr->Tags.begin() + i);
|
|
i--;
|
|
}
|
|
else {
|
|
ovr->Tags[i]->Params[0]->Set(value);
|
|
ovr->Tags[i]->Params[0]->omitted = false;
|
|
found = true;
|
|
}
|
|
}
|
|
}
|
|
if (!found) {
|
|
ovr->AddTag(insert);
|
|
}
|
|
|
|
line->UpdateText();
|
|
}
|
|
TextEdit->SetTextTo(line->Text);
|
|
if (!atEnd) TextEdit->SetSelectionU(selstart+shift,selend+shift);
|
|
TextEdit->SetFocus();
|
|
}
|
|
void SubsEditBox::OnFlagButton(wxCommandEvent &evt) {
|
|
int id = evt.GetId();
|
|
assert(id < BUTTON_LAST && id >= BUTTON_FIRST);
|
|
|
|
wxString tagname;
|
|
wxString desc;
|
|
bool state = false;
|
|
AssStyle *style = c->ass->GetStyle(line->Style);
|
|
AssStyle defStyle;
|
|
if (!style) style = &defStyle;
|
|
if (id == BUTTON_BOLD) {
|
|
tagname = "\\b";
|
|
desc = _("toggle bold");
|
|
state = style->bold;
|
|
}
|
|
else if (id == BUTTON_ITALICS) {
|
|
tagname = "\\i";
|
|
desc = _("toggle italic");
|
|
state = style->italic;
|
|
}
|
|
else if (id == BUTTON_UNDERLINE) {
|
|
tagname = "\\u";
|
|
desc = _("toggle underline");
|
|
state = style->underline;
|
|
}
|
|
else if (id == BUTTON_STRIKEOUT) {
|
|
tagname = "\\s";
|
|
desc = _("toggle strikeout");
|
|
state = style->strikeout;
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
|
|
line->ParseASSTags();
|
|
int selstart, selend;
|
|
get_selection(TextEdit, selstart, selend);
|
|
int blockn = BlockAtPos(line->Text, selstart);
|
|
|
|
state = get_value(*line, blockn, state, tagname);
|
|
|
|
SetTag(tagname, wxString::Format("%i", !state));
|
|
if (selend != selstart) {
|
|
SetTag(tagname, wxString::Format("%i", state), true);
|
|
}
|
|
|
|
line->ClearBlocks();
|
|
commitId = -1;
|
|
CommitText(desc);
|
|
}
|
|
void SubsEditBox::OnFontButton(wxCommandEvent &) {
|
|
int selstart, selend;
|
|
get_selection(TextEdit, selstart, selend);
|
|
|
|
line->ParseASSTags();
|
|
int blockn = BlockAtPos(line->Text, selstart);
|
|
|
|
wxFont startfont;
|
|
AssStyle *style = c->ass->GetStyle(line->Style);
|
|
AssStyle defStyle;
|
|
if (!style) style = &defStyle;
|
|
|
|
startfont.SetFaceName(get_value(*line, blockn, style->font, "\\fn"));
|
|
startfont.SetPointSize(get_value(*line, blockn, (int)style->fontsize, "\\fs"));
|
|
startfont.SetWeight(get_value(*line, blockn, style->bold, "\\b") ? wxFONTWEIGHT_BOLD : wxFONTWEIGHT_NORMAL);
|
|
startfont.SetStyle(get_value(*line, blockn, style->italic, "\\i") ? wxFONTSTYLE_ITALIC : wxFONTSTYLE_NORMAL);
|
|
startfont.SetUnderlined(get_value(*line, blockn, style->underline, "\\u"));
|
|
|
|
wxFont font = wxGetFontFromUser(this, startfont);
|
|
if (!font.Ok() || font == startfont) {
|
|
line->ClearBlocks();
|
|
return;
|
|
}
|
|
|
|
if (font.GetFaceName() != startfont.GetFaceName()) {
|
|
SetTag("\\fn", font.GetFaceName());
|
|
}
|
|
if (font.GetPointSize() != startfont.GetPointSize()) {
|
|
SetTag("\\fs", wxString::Format("%i", font.GetPointSize()));
|
|
}
|
|
if (font.GetWeight() != startfont.GetWeight()) {
|
|
SetTag("\\b", wxString::Format("%i", font.GetWeight() == wxFONTWEIGHT_BOLD));
|
|
}
|
|
if (font.GetStyle() != startfont.GetStyle()) {
|
|
SetTag("\\i", wxString::Format("%i", font.GetStyle() == wxFONTSTYLE_ITALIC));
|
|
}
|
|
if (font.GetUnderlined() != startfont.GetUnderlined()) {
|
|
SetTag("\\i", wxString::Format("%i", font.GetUnderlined()));
|
|
}
|
|
line->ClearBlocks();
|
|
commitId = -1;
|
|
CommitText(_("set font"));
|
|
}
|
|
void SubsEditBox::OnColorButton(wxCommandEvent &evt) {
|
|
int id = evt.GetId();
|
|
assert(id < BUTTON_LAST && id >= BUTTON_FIRST);
|
|
wxString alt;
|
|
|
|
wxColor color;
|
|
AssStyle *style = c->ass->GetStyle(line->Style);
|
|
AssStyle defStyle;
|
|
if (!style) style = &defStyle;
|
|
if (id == BUTTON_COLOR1) {
|
|
color = style->primary.GetWXColor();
|
|
colorTag = "\\c";
|
|
alt = "\\c1";
|
|
}
|
|
else if (id == BUTTON_COLOR2) {
|
|
color = style->secondary.GetWXColor();
|
|
colorTag = "\\2c";
|
|
}
|
|
else if (id == BUTTON_COLOR3) {
|
|
color = style->outline.GetWXColor();
|
|
colorTag = "\\3c";
|
|
}
|
|
else if (id == BUTTON_COLOR4) {
|
|
color = style->shadow.GetWXColor();
|
|
colorTag = "\\4c";
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
|
|
commitId = -1;
|
|
|
|
line->ParseASSTags();
|
|
int selstart, selend;
|
|
get_selection(TextEdit, selstart, selend);
|
|
int blockn = BlockAtPos(line->Text, selstart);
|
|
|
|
color = get_value(*line, blockn, color, colorTag, alt);
|
|
wxString initialText = line->Text;
|
|
wxColor newColor = GetColorFromUser<SubsEditBox, &SubsEditBox::SetColorCallback>(c->parent, color, this);
|
|
if (newColor == color) {
|
|
TextEdit->SetTextTo(initialText);
|
|
TextEdit->SetSelectionU(selstart, selend);
|
|
}
|
|
|
|
line->ClearBlocks();
|
|
CommitText(_("set color"));
|
|
}
|
|
void SubsEditBox::SetColorCallback(wxColor newColor) {
|
|
if (newColor.Ok()) {
|
|
SetTag(colorTag, AssColor(newColor).GetASSFormatted(false));
|
|
CommitText(_("set color"));
|
|
}
|
|
}
|