// Copyright (c) 2011, Thomas Goyne // // Permission to use, copy, modify, and distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice appear in all copies. // // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR // ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN // ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. // // Aegisub Project http://www.aegisub.org/ /// @file dialog_shift_times.cpp /// @brief Shift Times dialogue box and logic /// @ingroup secondary_ui /// #include "config.h" #include "dialog_shift_times.h" #ifndef AGI_PRE #include #include #include #include #include #include #include #include #include #endif #include #include #include #include #include #include #include "ass_dialogue.h" #include "ass_file.h" #include "ass_time.h" #include "compat.h" #include "include/aegisub/context.h" #include "help_button.h" #include "libresrc/libresrc.h" #include "main.h" #include "standard_paths.h" #include "timeedit_ctrl.h" #include "video_context.h" static wxString get_history_string(json::Object &obj) { wxString filename = lagi_wxString(obj["filename"]); if (filename.empty()) filename = _("unsaved"); wxString shift_amount(lagi_wxString(obj["amount"])); if (!obj["is by time"]) shift_amount = wxString::Format(_("%s frames"), shift_amount); wxString shift_direction = obj["is backward"] ? _("backward") : _("forward"); int64_t time_field = obj["fields"]; wxString fields = time_field == 0 ? _("s+e") : time_field == 1 ? _("s") : _("e") ; json::Array const& sel = obj["selection"]; wxString lines; int64_t sel_mode = obj["mode"]; if (sel_mode == 0) lines = _("all"); else if (sel_mode == 2) lines = wxString::Format(_("from %d onward"), (int)(int64_t)sel.front()["start"]); else { lines += _("sel "); for (json::Array::const_iterator it = sel.begin(); it != sel.end(); ++it) { int beg = (int64_t)(*it)["start"]; int end = (int64_t)(*it)["end"]; if (beg == end) lines += wxString::Format("%d", beg); else lines += wxString::Format("%d-%d", beg, end); if (it + 1 != sel.end()) lines += ";"; } } return wxString::Format("%s, %s %s, %s, %s", filename, shift_amount, shift_direction, fields, lines); } DialogShiftTimes::DialogShiftTimes(agi::Context *context) : wxDialog(context->parent, -1, _("Shift Times")) , context(context) , history_filename(STD_STR(StandardPaths::DecodePath("?user/shift_history.json"))) , history(new json::Array) , timecodes_loaded_slot(context->videoController->AddTimecodesListener(&DialogShiftTimes::OnTimecodesLoaded, this)) , selected_set_changed_slot(context->selectionController->AddSelectionListener(&DialogShiftTimes::OnSelectedSetChanged, this)) { SetIcon(GETICON(shift_times_toolbutton_16)); // Create controls shift_by_time = new wxRadioButton(this, -1, _("&Time: "), wxDefaultPosition, wxDefaultSize, wxRB_GROUP); shift_by_time->SetToolTip(_("Shift by time")); shift_by_time->Bind(wxEVT_COMMAND_RADIOBUTTON_SELECTED, &DialogShiftTimes::OnByTime, this); shift_by_frames = new wxRadioButton(this, -1 , _("&Frames: ")); shift_by_frames->SetToolTip(_("Shift by frames")); shift_by_frames->Bind(wxEVT_COMMAND_RADIOBUTTON_SELECTED, &DialogShiftTimes::OnByFrames, this); shift_time = new TimeEdit(this, -1, context); shift_time->SetToolTip(_("Enter time in h:mm:ss.cs notation")); shift_frames = new wxTextCtrl(this, -1); shift_frames->SetToolTip(_("Enter number of frames to shift by")); shift_forward = new wxRadioButton(this, -1, _("For&ward"), wxDefaultPosition, wxDefaultSize, wxRB_GROUP); shift_forward->SetToolTip(_("Shifts subs forward, making them appear later. Use if they are appearing too soon.")); shift_backward = new wxRadioButton(this, -1, _("&Backward")); shift_backward->SetToolTip(_("Shifts subs backward, making them appear earlier. Use if they are appearing too late.")); wxString selection_mode_vals[] = { _("&All rows"), _("Selected &rows"), _("Selection &onward") }; selection_mode = new wxRadioBox(this, -1, _("Affect"), wxDefaultPosition, wxDefaultSize, 3, selection_mode_vals, 1); wxString time_field_vals[] = { _("Start a&nd End times"), _("&Start times only"), _("&End times only") }; time_fields = new wxRadioBox(this, -1, _("Times"), wxDefaultPosition, wxDefaultSize, 3, time_field_vals, 1); history_box = new wxListBox(this, -1, wxDefaultPosition, wxSize(350, 100), 0, nullptr, wxLB_HSCROLL); wxButton *clear_button = new wxButton(this, -1, _("&Clear")); clear_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::OnClear, this); // Set initial control states OnTimecodesLoaded(context->videoController->FPS()); OnSelectedSetChanged(); LoadHistory(); shift_time->SetTime(OPT_GET("Tool/Shift Times/Time")->GetInt()); *shift_frames << (int)OPT_GET("Tool/Shift Times/Frames")->GetInt(); shift_by_frames->SetValue(!OPT_GET("Tool/Shift Times/ByTime")->GetBool() && shift_by_frames->IsEnabled()); time_fields->SetSelection(OPT_GET("Tool/Shift Times/Type")->GetInt()); selection_mode->SetSelection(OPT_GET("Tool/Shift Times/Affect")->GetInt()); shift_backward->SetValue(OPT_GET("Tool/Shift Times/Direction")->GetBool()); if (shift_by_frames->GetValue()) shift_time->Disable(); else shift_frames->Disable(); // Position controls wxSizer *shift_amount_sizer = new wxFlexGridSizer(2, 2, 5, 5); shift_amount_sizer->Add(shift_by_time, wxSizerFlags(0).Align(wxALIGN_CENTER_VERTICAL)); shift_amount_sizer->Add(shift_time, wxSizerFlags(1)); shift_amount_sizer->Add(shift_by_frames, wxSizerFlags(0).Align(wxALIGN_CENTER_VERTICAL)); shift_amount_sizer->Add(shift_frames, wxSizerFlags(1)); wxSizer *shift_direction_sizer = new wxBoxSizer(wxHORIZONTAL); shift_direction_sizer->Add(shift_forward, wxSizerFlags(1).Expand()); shift_direction_sizer->Add(shift_backward, wxSizerFlags(1).Expand().Border(wxLEFT)); wxSizer *shift_by_sizer = new wxStaticBoxSizer(wxVERTICAL, this, _("Shift by")); shift_by_sizer->Add(shift_amount_sizer, wxSizerFlags().Expand()); shift_by_sizer->Add(shift_direction_sizer, wxSizerFlags().Expand().Border(wxTOP)); wxSizer *left_sizer = new wxBoxSizer(wxVERTICAL); left_sizer->Add(shift_by_sizer, wxSizerFlags().Expand().Border(wxBOTTOM)); left_sizer->Add(selection_mode, wxSizerFlags().Expand().Border(wxBOTTOM)); left_sizer->Add(time_fields, wxSizerFlags().Expand()); wxSizer *history_sizer = new wxStaticBoxSizer(wxVERTICAL, this, _("Load from history")); history_sizer->Add(history_box, wxSizerFlags(1).Expand()); history_sizer->Add(clear_button, wxSizerFlags().Expand().Border(wxTOP)); wxSizer *top_sizer = new wxBoxSizer(wxHORIZONTAL); top_sizer->Add(left_sizer, wxSizerFlags().Border(wxALL & ~wxRIGHT).Expand()); top_sizer->Add(history_sizer, wxSizerFlags().Border().Expand()); wxSizer *main_sizer = new wxBoxSizer(wxVERTICAL); main_sizer->Add(top_sizer, wxSizerFlags().Border(wxALL & ~wxBOTTOM)); main_sizer->Add(CreateButtonSizer(wxOK | wxCANCEL | wxHELP), wxSizerFlags().Right().Border()); SetSizerAndFit(main_sizer); CenterOnParent(); Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::Process, this, wxID_OK); Bind(wxEVT_COMMAND_BUTTON_CLICKED, std::bind(&HelpButton::OpenPage, "Shift Times"), wxID_HELP); shift_time->Bind(wxEVT_COMMAND_TEXT_ENTER, &DialogShiftTimes::Process, this); history_box->Bind(wxEVT_COMMAND_LISTBOX_DOUBLECLICKED, &DialogShiftTimes::OnHistoryClick, this); } DialogShiftTimes::~DialogShiftTimes() { long shift; shift_frames->GetValue().ToLong(&shift); OPT_SET("Tool/Shift Times/Time")->SetInt(shift_time->GetTime()); OPT_SET("Tool/Shift Times/Frames")->SetInt(shift); OPT_SET("Tool/Shift Times/ByTime")->SetBool(shift_by_time->GetValue()); OPT_SET("Tool/Shift Times/Type")->SetInt(time_fields->GetSelection()); OPT_SET("Tool/Shift Times/Affect")->SetInt(selection_mode->GetSelection()); OPT_SET("Tool/Shift Times/Direction")->SetBool(shift_backward->GetValue()); } void DialogShiftTimes::OnTimecodesLoaded(agi::vfr::Framerate const& new_fps) { fps = new_fps; if (fps.IsLoaded()) { shift_by_frames->Enable(); } else { shift_by_time->SetValue(true); shift_by_frames->Disable(); shift_time->Enable(); shift_frames->Disable(); } } void DialogShiftTimes::OnSelectedSetChanged() { if (context->selectionController->GetSelectedSet().empty()) { selection_mode->Enable(1, false); selection_mode->Enable(2, false); selection_mode->SetSelection(0); } else { selection_mode->Enable(1, true); selection_mode->Enable(2, true); } } void DialogShiftTimes::OnClear(wxCommandEvent &) { wxRemoveFile(lagi_wxString(history_filename)); history_box->Clear(); history->clear(); } void DialogShiftTimes::OnByTime(wxCommandEvent &) { shift_time->Enable(true); shift_frames->Enable(false); } void DialogShiftTimes::OnByFrames(wxCommandEvent &) { shift_time->Enable(false); shift_frames->Enable(true); } void DialogShiftTimes::OnHistoryClick(wxCommandEvent &evt) { size_t entry = evt.GetInt(); if (entry >= history->size()) return; json::Object& obj = (*history)[entry]; if (obj["is by time"]) { shift_time->SetTime(AssTime(lagi_wxString(obj["amount"]))); shift_by_time->SetValue(true); OnByTime(evt); } else { shift_frames->SetValue(lagi_wxString(obj["amount"])); if (shift_by_frames->IsEnabled()) { shift_by_frames->SetValue(true); OnByFrames(evt); } } if (obj["is backward"]) shift_backward->SetValue(true); else shift_forward->SetValue(true); selection_mode->SetSelection((int64_t)obj["mode"]); time_fields->SetSelection((int64_t)obj["fields"]); } void DialogShiftTimes::SaveHistory(json::Array const& shifted_blocks) { json::Object new_entry; new_entry["filename"] = STD_STR(wxFileName(context->ass->filename).GetFullName()); new_entry["is by time"] = shift_by_time->GetValue(); new_entry["is backward"] = shift_backward->GetValue(); new_entry["amount"] = STD_STR(shift_by_time->GetValue() ? shift_time->GetValue() : shift_frames->GetValue()); new_entry["fields"] = time_fields->GetSelection(); new_entry["mode"] = selection_mode->GetSelection(); new_entry["selection"] = shifted_blocks; history->push_front(new_entry); try { json::Writer::Write(*history, agi::io::Save(history_filename).Get()); } catch (agi::FileSystemError const& e) { LOG_E("dialog_shift_times/save_history") << "Cannot save shift times history: " << e.GetChainedMessage(); } } void DialogShiftTimes::LoadHistory() { history_box->Clear(); history_box->Freeze(); try { agi::scoped_ptr file(agi::io::Open(history_filename)); json::UnknownElement root; json::Reader::Read(root, *file); *history = root; for (auto& history_entry : *history) history_box->Append(get_history_string(history_entry)); } catch (agi::FileSystemError const& e) { LOG_D("dialog_shift_times/load_history") << "Cannot load shift times history: " << e.GetChainedMessage(); } catch (...) { history_box->Thaw(); throw; } history_box->Thaw(); } void DialogShiftTimes::Process(wxCommandEvent &) { int mode = selection_mode->GetSelection(); int type = time_fields->GetSelection(); bool reverse = shift_backward->GetValue(); bool by_time = shift_by_time->GetValue(); bool start = type != 2; bool end = type != 1; SubtitleSelection sel = context->selectionController->GetSelectedSet(); long shift; if (by_time) { shift = shift_time->GetTime(); if (shift == 0) { Close(); return; } } else shift_frames->GetValue().ToLong(&shift); if (reverse) shift = -shift; // Track which rows were shifted for the log int row_number = 0; int block_start = 0; json::Array shifted_blocks; for (auto line : context->ass->Line | agi::of_type()) { ++row_number; if (!sel.count(line)) { if (block_start) { json::Object block; block["start"] = block_start; block["end"] = row_number - 1; shifted_blocks.push_back(block); block_start = 0; } if (mode == 1) continue; if (mode == 2 && shifted_blocks.empty()) continue; } else if (!block_start) block_start = row_number; if (start) line->Start = Shift(line->Start, shift, by_time, agi::vfr::START); if (end) line->End = Shift(line->End, shift, by_time, agi::vfr::END); } context->ass->Commit(_("shifting"), AssFile::COMMIT_DIAG_TIME); if (block_start) { json::Object block; block["start"] = block_start; block["end"] = row_number - 1; shifted_blocks.push_back(block); } SaveHistory(shifted_blocks); Close(); } int DialogShiftTimes::Shift(int initial_time, int shift, bool by_time, agi::vfr::Time type) { if (by_time) return initial_time + shift; else return fps.TimeAtFrame(shift + fps.FrameAtTime(initial_time, type), type); }