// 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/ // // $Id$ /// @file visual_tool.cpp /// @brief Base class for visual typesetting functions /// @ingroup visual_ts #include "config.h" #ifndef AGI_PRE #include #endif #include "visual_tool.h" #include "ass_dialogue.h" #include "ass_file.h" #include "ass_override.h" #include "ass_style.h" #include "ass_time.h" #include "include/aegisub/context.h" #include "main.h" #include "utils.h" #include "video_context.h" #include "video_display.h" #include "video_provider_manager.h" #include "visual_feature.h" #include "visual_tool_clip.h" #include "visual_tool_drag.h" #include "visual_tool_vector_clip.h" template static void for_each(C &range, F func) { std::for_each(range.begin(), range.end(), func); } using std::tr1::placeholders::_1; const wxColour VisualToolBase::colour[4] = {wxColour(106,32,19), wxColour(255,169,40), wxColour(255,253,185), wxColour(187,0,0)}; VisualToolBase::VisualToolBase(VideoDisplay *parent, agi::Context *context) : c(context) , parent(parent) , holding(false) , active_line(0) , dragging(false) , frame_number(c->videoController->GetFrameN()) , left_click(false) , left_double(false) , shift_down(false) , ctrl_down(false) , alt_down(false) , file_changed_connection(c->ass->AddCommitListener(&VisualToolBase::OnCommit, this)) , commit_id(-1) { int script_w, script_h; c->ass->GetResolution(script_w, script_h); script_res = Vector2D(script_w, script_h); active_line = GetActiveDialogueLine(); c->selectionController->AddSelectionListener(this); connections.push_back(c->videoController->AddSeekListener(&VisualToolBase::OnSeek, this)); parent->Bind(wxEVT_MOUSE_CAPTURE_LOST, &VisualToolBase::OnMouseCaptureLost, this); } VisualToolBase::~VisualToolBase() { c->selectionController->RemoveSelectionListener(this); } void VisualToolBase::OnCommit(int type) { holding = false; dragging = false; if (type == AssFile::COMMIT_NEW || type & AssFile::COMMIT_SCRIPTINFO) { int script_w, script_h; c->ass->GetResolution(script_w, script_h); script_res = Vector2D(script_w, script_h); OnCoordinateSystemsChanged(); } if (type & AssFile::COMMIT_DIAG_FULL || type & AssFile::COMMIT_DIAG_ADDREM) { active_line = GetActiveDialogueLine(); OnFileChanged(); } } void VisualToolBase::OnSeek(int new_frame) { if (frame_number == new_frame) return; frame_number = new_frame; dragging = false; OnFrameChanged(); AssDialogue *new_line = GetActiveDialogueLine(); if (new_line != active_line) { active_line = new_line; OnLineChanged(); } } void VisualToolBase::OnMouseCaptureLost(wxMouseCaptureLostEvent &) { holding = false; dragging = false; left_click = false; } void VisualToolBase::OnActiveLineChanged(AssDialogue *new_line) { if (!IsDisplayed(new_line)) new_line = 0; holding = false; dragging = false; if (new_line != active_line) { active_line = new_line; OnLineChanged(); } } bool VisualToolBase::IsDisplayed(AssDialogue *line) const { int frame = c->videoController->GetFrameN(); return line && c->videoController->FrameAtTime(line->Start.GetMS(), agi::vfr::START) <= frame && c->videoController->FrameAtTime(line->End.GetMS(), agi::vfr::END) >= frame; } void VisualToolBase::Commit(wxString message) { file_changed_connection.Block(); if (message.empty()) message = _("visual typesetting"); commit_id = c->ass->Commit(message, AssFile::COMMIT_DIAG_TEXT, commit_id); file_changed_connection.Unblock(); } AssDialogue* VisualToolBase::GetActiveDialogueLine() { AssDialogue *diag = c->selectionController->GetActiveLine(); if (IsDisplayed(diag)) return diag; return 0; } void VisualToolBase::SetDisplayArea(int x, int y, int w, int h) { video_pos = Vector2D(x, y); video_res = Vector2D(w, h); holding = false; dragging = false; OnCoordinateSystemsChanged(); } Vector2D VisualToolBase::ToScriptCoords(Vector2D point) const { return (point - video_pos) * script_res / video_res; } Vector2D VisualToolBase::FromScriptCoords(Vector2D point) const { return (point * video_res / script_res) + video_pos; } template VisualTool::VisualTool(VideoDisplay *parent, agi::Context *context) : VisualToolBase(parent, context) , sel_changed(false) { active_feature = features.begin(); } template void VisualTool::OnMouseEvent(wxMouseEvent &event) { left_click = event.LeftDown(); left_double = event.LeftDClick(); shift_down = event.ShiftDown(); ctrl_down = event.CmdDown(); alt_down = event.AltDown(); mouse_pos = event.GetPosition(); bool need_render = false; if (event.Leaving()) { mouse_pos = Vector2D(); parent->Render(); return; } if (event.Entering() && !OPT_GET("Tool/Visual/Always Show")->GetBool()) need_render = true; if (!dragging) { feature_iterator prev_feature = active_feature; int max_layer = INT_MIN; active_feature = features.end(); for (feature_iterator cur = features.begin(); cur != features.end(); ++cur) { if (cur->IsMouseOver(mouse_pos) && cur->layer >= max_layer) { active_feature = cur; max_layer = cur->layer; } } need_render |= active_feature != prev_feature; } if (dragging) { // continue drag if (event.LeftIsDown()) { for_each(sel_features, bind(&FeatureType::UpdateDrag, _1, mouse_pos - drag_start, shift_down)); for_each(sel_features, bind(&VisualTool::UpdateDrag, this, _1)); Commit(); need_render = true; } // end drag else { dragging = false; // mouse didn't move, fiddle with selection if (active_feature != features.end() && !active_feature->HasMoved()) { // Don't deselect stuff that was selected in this click's mousedown event if (!sel_changed) { if (ctrl_down) RemoveSelection(active_feature); else SetSelection(active_feature, true); } } active_feature = features.end(); parent->ReleaseMouse(); parent->SetFocus(); } } else if (holding) { if (event.LeftIsDown()) { UpdateHold(); need_render = true; } // end hold else { holding = false; parent->ReleaseMouse(); parent->SetFocus(); } Commit(); } else if (left_click) { drag_start = mouse_pos; // start drag if (active_feature != features.end()) { if (!sel_features.count(active_feature)) { sel_changed = true; SetSelection(active_feature, !ctrl_down); } else sel_changed = false; if (active_feature->line) c->selectionController->SetActiveLine(active_feature->line); if (InitializeDrag(active_feature)) { for_each(sel_features, bind(&VisualDraggableFeature::StartDrag, _1)); dragging = true; parent->CaptureMouse(); } } // start hold else { if (!alt_down) { sel_features.clear(); Selection sel; sel.insert(c->selectionController->GetActiveLine()); c->selectionController->SetSelectedSet(sel); need_render = true; } if (active_line && InitializeHold()) { holding = true; parent->CaptureMouse(); } } } if (active_line && left_double) OnDoubleClick(); //if (need_render) parent->Render(); // Only coalesce the changes made in a single drag if (!event.LeftIsDown()) commit_id = -1; } template void VisualTool::DrawAllFeatures() { gl.SetLineColour(colour[0], 1.0f, 2); for (feature_iterator cur = features.begin(); cur != features.end(); ++cur) { int fill = 1; if (cur == active_feature) fill = 2; else if (sel_features.count(cur)) fill = 3; gl.SetFillColour(colour[fill], 0.6f); cur->Draw(gl); } } template void VisualTool::SetSelection(feature_iterator feat, bool clear) { if (clear) sel_features.clear(); if (sel_features.insert(feat).second && feat->line) { Selection sel; if (!clear) sel = c->selectionController->GetSelectedSet(); if (sel.insert(feat->line).second) c->selectionController->SetSelectedSet(sel); } } template void VisualTool::RemoveSelection(feature_iterator feat) { if (!sel_features.erase(feat) || !feat->line) return; for (selection_iterator it = sel_features.begin(); it != sel_features.end(); ++it) { if ((*it)->line == feat->line) return; } Selection sel = c->selectionController->GetSelectedSet(); // Don't deselect the only selected line if (sel.size() <= 1) return; sel.erase(feat->line); // Set the active line to an arbitrary selected line if we just // deselected the active line if (feat->line == c->selectionController->GetActiveLine()) { c->selectionController->SetActiveLine(*sel.begin()); } c->selectionController->SetSelectedSet(sel); } //////// PARSERS typedef const std::vector * param_vec; // Parse line on creation and unparse at the end of scope struct scoped_tag_parse { AssDialogue *diag; scoped_tag_parse(AssDialogue *diag) : diag(diag) { diag->ParseASSTags(); } ~scoped_tag_parse() { diag->ClearBlocks(); } }; // Find a tag's parameters in a line or return NULL if it's not found static param_vec find_tag(const AssDialogue *line, wxString tag_name) { for (size_t i = 0; i < line->Blocks.size(); ++i) { const AssDialogueBlockOverride *ovr = dynamic_cast(line->Blocks[i]); if (!ovr) continue; for (size_t j = 0; j < ovr->Tags.size(); ++j) { if (ovr->Tags[j]->Name == tag_name) return &ovr->Tags[j]->Params; } } return 0; } // Get a Vector2D from the given tag parameters, or Vector2D::Bad() if they are not valid static Vector2D vec_or_bad(param_vec tag, size_t x_idx, size_t y_idx) { if (!tag || tag->size() <= x_idx || tag->size() <= y_idx || (*tag)[x_idx]->omitted || (*tag)[y_idx]->omitted || (*tag)[x_idx]->GetType() == VARDATA_NONE || (*tag)[y_idx]->GetType() == VARDATA_NONE) { return Vector2D(); } return Vector2D((*tag)[x_idx]->Get(), (*tag)[y_idx]->Get()); } Vector2D VisualToolBase::GetLinePosition(AssDialogue *diag) { scoped_tag_parse parse(diag); if (Vector2D ret = vec_or_bad(find_tag(diag, "\\pos"), 0, 1)) return ret; if (Vector2D ret = vec_or_bad(find_tag(diag, "\\move"), 0, 1)) return ret; // Get default position int margin[4]; std::copy(diag->Margin, diag->Margin + 4, margin); int align = 2; if (AssStyle *style = c->ass->GetStyle(diag->Style)) { align = style->alignment; for (int i = 0; i < 4; i++) { if (margin[i] == 0) margin[i] = style->Margin[i]; } } param_vec align_tag; if ((align_tag = find_tag(diag, "\\an")) && !(*align_tag)[0]->omitted) align = (*align_tag)[0]->Get(); else if ((align_tag = find_tag(diag, "\\a"))) { align = (*align_tag)[0]->Get(2); // \a -> \an values mapping static int align_mapping[] = { 2, 1, 2, 3, 7, 7, 8, 9, 7, 4, 5, 6 }; if (static_cast(align) < sizeof(align_mapping) / sizeof(int)) align = align_mapping[align]; else align = 2; } // Alignment type int hor = (align - 1) % 3; int vert = (align - 1) / 3; // Calculate positions int x, y; if (hor == 0) x = margin[0]; else if (hor == 1) x = (script_res.X() + margin[0] - margin[1]) / 2; else x = margin[1]; if (vert == 0) y = script_res.Y() - margin[2]; else if (vert == 1) y = script_res.Y() / 2; else y = margin[2]; return Vector2D(x, y); } Vector2D VisualToolBase::GetLineOrigin(AssDialogue *diag) { scoped_tag_parse parse(diag); return vec_or_bad(find_tag(diag, "\\org"), 0, 1); } bool VisualToolBase::GetLineMove(AssDialogue *diag, Vector2D &p1, Vector2D &p2, int &t1, int &t2) { scoped_tag_parse parse(diag); param_vec tag = find_tag(diag, "\\move"); if (!tag) return false; p1 = vec_or_bad(tag, 0, 1); p2 = vec_or_bad(tag, 2, 3); // VSFilter actually defaults to -1, but it uses <= 0 to check for default and 0 seems less bug-prone t1 = (*tag)[4]->Get(0); t2 = (*tag)[5]->Get(0); return p1 && p2; } void VisualToolBase::GetLineRotation(AssDialogue *diag, float &rx, float &ry, float &rz) { rx = ry = rz = 0.f; if (AssStyle *style = c->ass->GetStyle(diag->Style)) rz = style->angle; scoped_tag_parse parse(diag); if (param_vec tag = find_tag(diag, "\\frx")) rx = tag->front()->Get(rx); if (param_vec tag = find_tag(diag, "\\fry")) ry = tag->front()->Get(ry); if (param_vec tag = find_tag(diag, "\\frz")) rz = tag->front()->Get(rz); else if (param_vec tag = find_tag(diag, "\\fr")) rz = tag->front()->Get(rz); } void VisualToolBase::GetLineScale(AssDialogue *diag, Vector2D &scale) { float x = 100.f, y = 100.f; if (AssStyle *style = c->ass->GetStyle(diag->Style)) { x = style->scalex; y = style->scaley; } scoped_tag_parse parse(diag); if (param_vec tag = find_tag(diag, "\\fscx")) x = tag->front()->Get(x); if (param_vec tag = find_tag(diag, "\\fscy")) y = tag->front()->Get(y); scale = Vector2D(x, y); } void VisualToolBase::GetLineClip(AssDialogue *diag, Vector2D &p1, Vector2D &p2, bool &inverse) { inverse = false; scoped_tag_parse parse(diag); param_vec tag = find_tag(diag, "\\iclip"); if (tag) inverse = true; else tag = find_tag(diag, "\\clip"); if (tag && tag->size() == 4) { p1 = vec_or_bad(tag, 0, 1); p2 = vec_or_bad(tag, 2, 3); } else { p1 = Vector2D(0, 0); p2 = script_res - 1; } } wxString VisualToolBase::GetLineVectorClip(AssDialogue *diag, int &scale, bool &inverse) { scoped_tag_parse parse(diag); scale = 1; inverse = false; param_vec tag = find_tag(diag, "\\iclip"); if (tag) inverse = true; else tag = find_tag(diag, "\\clip"); if (tag && tag->size() == 4) { return wxString::Format("m %d %d l %d %d %d %d %d %d", (*tag)[0]->Get(), (*tag)[1]->Get(), (*tag)[2]->Get(), (*tag)[1]->Get(), (*tag)[2]->Get(), (*tag)[3]->Get(), (*tag)[0]->Get(), (*tag)[3]->Get()); } if (tag) { scale = (*tag)[0]->Get(scale); return (*tag)[1]->Get(""); } return ""; } void VisualToolBase::SetSelectedOverride(wxString const& tag, wxString const& value) { for_each(c->selectionController->GetSelectedSet(), bind(&VisualToolBase::SetOverride, this, _1, tag, value)); } void VisualToolBase::SetOverride(AssDialogue* line, wxString const& tag, wxString const& value) { if (!line) return; wxString removeTag; if (tag == "\\1c") removeTag = "\\c"; else if (tag == "\\frz") removeTag = "\\fr"; else if (tag == "\\pos") removeTag = "\\move"; else if (tag == "\\move") removeTag = "\\pos"; else if (tag == "\\clip") removeTag = "\\iclip"; else if (tag == "\\iclip") removeTag = "\\clip"; wxString insert = tag + value; // Get block at start line->ParseASSTags(); AssDialogueBlock *block = line->Blocks.front(); // Get current block as plain or override assert(dynamic_cast(block) == NULL); if (dynamic_cast(block)) line->Text = "{" + insert + "}" + line->Text; else if (AssDialogueBlockOverride *ovr = dynamic_cast(block)) { // Remove old of same for (size_t i = 0; i < ovr->Tags.size(); i++) { wxString name = ovr->Tags[i]->Name; if (tag == name || removeTag == name) { delete ovr->Tags[i]; ovr->Tags.erase(ovr->Tags.begin() + i); i--; } } ovr->AddTag(insert); line->UpdateText(); } } // If only export worked template class VisualTool; template class VisualTool; template class VisualTool; template class VisualTool;