Aegisub/src/command/time.cpp

406 lines
13 KiB
C++

// Copyright (c) 2005-2010, Niels Martin Hansen
// Copyright (c) 2005-2010, Rodrigo Braz Monteiro
// Copyright (c) 2010, Amar Takhar
// 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/
#include "command.h"
#include "../ass_dialogue.h"
#include "../ass_file.h"
#include "../audio_controller.h"
#include "../audio_timing.h"
#include "../dialog_manager.h"
#include "../dialog_shift_times.h"
#include "../include/aegisub/context.h"
#include "../libresrc/libresrc.h"
#include "../options.h"
#include "../selection_controller.h"
#include "../video_context.h"
#include <libaegisub/util.h>
#include <algorithm>
namespace {
using cmd::Command;
struct validate_video_loaded : public Command {
CMD_TYPE(COMMAND_VALIDATE)
bool Validate(const agi::Context *c) override {
return c->videoController->IsLoaded();
}
};
struct validate_adjoinable : public Command {
CMD_TYPE(COMMAND_VALIDATE)
bool Validate(const agi::Context *c) override {
auto const& sel = c->selectionController->GetSelectedSet();
if (sel.size() < 2) return !sel.empty();
size_t found = 0;
for (auto& diag : c->ass->Events) {
if (sel.count(&diag)) {
if (++found == sel.size())
return true;
}
else if (found)
return false;
}
return true;
}
};
static void adjoin_lines(agi::Context *c, bool set_start) {
auto const& sel = c->selectionController->GetSelectedSet();
AssDialogue *prev = nullptr;
size_t seen = 0;
bool prev_sel = false;
for (auto& diag : c->ass->Events) {
bool cur_sel = !!sel.count(&diag);
if (prev) {
// One row selections act as if the previous or next line was selected
if (set_start && cur_sel && (sel.size() == 1 || prev_sel))
diag.Start = prev->End;
else if (!set_start && prev_sel && (cur_sel || sel.size() == 1))
prev->End = diag.Start;
}
if (seen == sel.size())
break;
if (cur_sel)
++seen;
prev = &diag;
prev_sel = cur_sel;
}
c->ass->Commit(_("adjoin"), AssFile::COMMIT_DIAG_TIME);
}
struct time_continuous_end final : public validate_adjoinable {
CMD_NAME("time/continuous/end")
STR_MENU("Change &End")
STR_DISP("Change End")
STR_HELP("Change end times of lines to the next line's start time")
void operator()(agi::Context *c) override {
adjoin_lines(c, false);
}
};
struct time_continuous_start final : public validate_adjoinable {
CMD_NAME("time/continuous/start")
STR_MENU("Change &Start")
STR_DISP("Change Start")
STR_HELP("Change start times of lines to the previous line's end time")
void operator()(agi::Context *c) override {
adjoin_lines(c, true);
}
};
struct time_frame_current final : public validate_video_loaded {
CMD_NAME("time/frame/current")
CMD_ICON(shift_to_frame)
STR_MENU("Shift to &Current Frame")
STR_DISP("Shift to Current Frame")
STR_HELP("Shift selection so that the active line starts at current frame")
void operator()(agi::Context *c) override {
if (!c->videoController->IsLoaded()) return;
auto const& sel = c->selectionController->GetSelectedSet();
const auto active_line = c->selectionController->GetActiveLine();
if (sel.empty() || !active_line) return;
int target_start = std::max(0, c->videoController->TimeAtFrame(c->videoController->GetFrameN(), agi::vfr::START));
int shift_by = target_start - active_line->Start;
for (auto line : sel) {
line->Start = line->Start + shift_by;
line->End = line->End + shift_by;
}
c->ass->Commit(_("shift to frame"), AssFile::COMMIT_DIAG_TIME);
}
};
struct time_shift final : public Command {
CMD_NAME("time/shift")
CMD_ICON(shift_times_toolbutton)
STR_MENU("S&hift Times...")
STR_DISP("Shift Times")
STR_HELP("Shift subtitles by time or frames")
void operator()(agi::Context *c) override {
c->dialog->Show<DialogShiftTimes>(c);
}
};
static void snap_subs_video(agi::Context *c, bool set_start) {
auto const& sel = c->selectionController->GetSelectedSet();
if (!c->videoController->IsLoaded() || sel.empty()) return;
int start = c->videoController->TimeAtFrame(c->videoController->GetFrameN(), agi::vfr::START);
int end = c->videoController->TimeAtFrame(c->videoController->GetFrameN(), agi::vfr::END);
for (auto line : sel) {
if (set_start || line->Start > start)
line->Start = start;
if (!set_start || line->End < end)
line->End = end;
}
c->ass->Commit(_("timing"), AssFile::COMMIT_DIAG_TIME);
}
struct time_snap_end_video final : public validate_video_loaded {
CMD_NAME("time/snap/end_video")
CMD_ICON(subend_to_video)
STR_MENU("Snap &End to Video")
STR_DISP("Snap End to Video")
STR_HELP("Set end of selected subtitles to current video frame")
void operator()(agi::Context *c) override {
snap_subs_video(c, false);
}
};
struct time_snap_scene final : public validate_video_loaded {
CMD_NAME("time/snap/scene")
CMD_ICON(snap_subs_to_scene)
STR_MENU("Snap to S&cene")
STR_DISP("Snap to Scene")
STR_HELP("Set start and end of subtitles to the keyframes around current video frame")
void operator()(agi::Context *c) override {
VideoContext *con = c->videoController.get();
if (!con->IsLoaded() || !con->KeyFramesLoaded()) return;
int curFrame = con->GetFrameN();
int prev = 0;
int next = 0;
auto const& keyframes = con->GetKeyFrames();
if (curFrame < keyframes.front())
next = keyframes.front();
else if (curFrame >= keyframes.back()) {
prev = keyframes.back();
next = con->GetLength();
}
else {
auto kf = std::lower_bound(keyframes.begin(), keyframes.end(), curFrame);
if (*kf == curFrame) {
prev = *kf;
next = *(kf + 1);
}
else {
prev = *(kf - 1);
next = *kf;
}
}
int start_ms = con->TimeAtFrame(prev,agi::vfr::START);
int end_ms = con->TimeAtFrame(next-1,agi::vfr::END);
for (auto line : c->selectionController->GetSelectedSet()) {
line->Start = start_ms;
line->End = end_ms;
}
c->ass->Commit(_("snap to scene"), AssFile::COMMIT_DIAG_TIME);
}
};
struct time_add_lead_both final : public Command {
CMD_NAME("time/lead/both")
STR_MENU("Add lead in and out")
STR_DISP("Add lead in and out")
STR_HELP("Add both lead in and out to the selected lines")
void operator()(agi::Context *c) override {
if (AudioTimingController *tc = c->audioController->GetTimingController()) {
tc->AddLeadIn();
tc->AddLeadOut();
}
}
};
struct time_add_lead_in final : public Command {
CMD_NAME("time/lead/in")
CMD_ICON(button_leadin)
STR_MENU("Add lead in")
STR_DISP("Add lead in")
STR_HELP("Add the lead in time to the selected lines")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->AddLeadIn();
}
};
struct time_add_lead_out final : public Command {
CMD_NAME("time/lead/out")
CMD_ICON(button_leadout)
STR_MENU("Add lead out")
STR_DISP("Add lead out")
STR_HELP("Add the lead out time to the selected lines")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->AddLeadOut();
}
};
struct time_length_increase final : public Command {
CMD_NAME("time/length/increase")
STR_MENU("Increase length")
STR_DISP("Increase length")
STR_HELP("Increase the length of the current timing unit")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->ModifyLength(1, false);
}
};
struct time_length_increase_shift final : public Command {
CMD_NAME("time/length/increase/shift")
STR_MENU("Increase length and shift")
STR_DISP("Increase length and shift")
STR_HELP("Increase the length of the current timing unit and shift the following items")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->ModifyLength(1, true);
}
};
struct time_length_decrease final : public Command {
CMD_NAME("time/length/decrease")
STR_MENU("Decrease length")
STR_DISP("Decrease length")
STR_HELP("Decrease the length of the current timing unit")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->ModifyLength(-1, false);
}
};
struct time_length_decrease_shift final : public Command {
CMD_NAME("time/length/decrease/shift")
STR_MENU("Decrease length and shift")
STR_DISP("Decrease length and shift")
STR_HELP("Decrease the length of the current timing unit and shift the following items")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->ModifyLength(-1, true);
}
};
struct time_start_increase final : public Command {
CMD_NAME("time/start/increase")
STR_MENU("Shift start time forward")
STR_DISP("Shift start time forward")
STR_HELP("Shift the start time of the current timing unit forward")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->ModifyStart(1);
}
};
struct time_start_decrease final : public Command {
CMD_NAME("time/start/decrease")
STR_MENU("Shift start time backward")
STR_DISP("Shift start time backward")
STR_HELP("Shift the start time of the current timing unit backward")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->ModifyStart(-1);
}
};
struct time_snap_start_video final : public validate_video_loaded {
CMD_NAME("time/snap/start_video")
CMD_ICON(substart_to_video)
STR_MENU("Snap &Start to Video")
STR_DISP("Snap Start to Video")
STR_HELP("Set start of selected subtitles to current video frame")
void operator()(agi::Context *c) override {
snap_subs_video(c, true);
}
};
struct time_next final : public Command {
CMD_NAME("time/next")
CMD_ICON(button_next)
STR_MENU("Next Line")
STR_DISP("Next Line")
STR_HELP("Next line or syllable")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->Next(AudioTimingController::TIMING_UNIT);
}
};
struct time_prev final : public Command {
CMD_NAME("time/prev")
CMD_ICON(button_prev)
STR_MENU("Previous Line")
STR_DISP("Previous Line")
STR_HELP("Previous line or syllable")
void operator()(agi::Context *c) override {
if (c->audioController->GetTimingController())
c->audioController->GetTimingController()->Prev();
}
};
}
namespace cmd {
void init_time() {
reg(agi::util::make_unique<time_add_lead_both>());
reg(agi::util::make_unique<time_add_lead_in>());
reg(agi::util::make_unique<time_add_lead_out>());
reg(agi::util::make_unique<time_continuous_end>());
reg(agi::util::make_unique<time_continuous_start>());
reg(agi::util::make_unique<time_frame_current>());
reg(agi::util::make_unique<time_length_decrease>());
reg(agi::util::make_unique<time_length_decrease_shift>());
reg(agi::util::make_unique<time_length_increase>());
reg(agi::util::make_unique<time_length_increase_shift>());
reg(agi::util::make_unique<time_next>());
reg(agi::util::make_unique<time_prev>());
reg(agi::util::make_unique<time_shift>());
reg(agi::util::make_unique<time_snap_end_video>());
reg(agi::util::make_unique<time_snap_scene>());
reg(agi::util::make_unique<time_snap_start_video>());
reg(agi::util::make_unique<time_start_decrease>());
reg(agi::util::make_unique<time_start_increase>());
}
}