Add double-click positioning to drag visual tool.

Sets the last clicked-on feature to the double-clicked spot and applies
the same relative movement to all other selected lines (including ones
not visible on the current frame).

Updates #513.

Originally committed to SVN as r4350.
This commit is contained in:
Thomas Goyne 2010-05-23 08:53:27 +00:00
parent 0dc6a082ca
commit 1e975eee89
5 changed files with 76 additions and 20 deletions

View file

@ -68,10 +68,12 @@
#include <algorithm> #include <algorithm>
#include <deque> #include <deque>
#include <fstream> #include <fstream>
#include <iterator>
#include <iostream> #include <iostream>
#include <list> #include <list>
#include <map> #include <map>
#include <memory> #include <memory>
#include <set>
#include <string> #include <string>
#include <utility> #include <utility>
#include <vector> #include <vector>

View file

@ -250,7 +250,6 @@ void VisualTool<FeatureType>::OnMouseEvent (wxMouseEvent &event) {
} }
// start hold // start hold
else { else {
ClearSelection();
curDiag = GetActiveDialogueLine(); curDiag = GetActiveDialogueLine();
if (curDiag && InitializeHold()) { if (curDiag && InitializeHold()) {
holding = true; holding = true;

View file

@ -36,6 +36,7 @@
#pragma once #pragma once
#ifndef AGI_PRE #ifndef AGI_PRE
#include <list>
#include <vector> #include <vector>
#include <wx/log.h> #include <wx/log.h>

View file

@ -49,6 +49,9 @@
enum { enum {
BUTTON_TOGGLE_MOVE = VISUAL_SUB_TOOL_START BUTTON_TOGGLE_MOVE = VISUAL_SUB_TOOL_START
}; };
static const DraggableFeatureType DRAG_ORIGIN = DRAG_BIG_TRIANGLE;
static const DraggableFeatureType DRAG_START = DRAG_BIG_SQUARE;
static const DraggableFeatureType DRAG_END = DRAG_BIG_CIRCLE;
/// @brief Constructor /// @brief Constructor
/// @param _parent /// @param _parent
@ -56,6 +59,7 @@ enum {
VisualToolDrag::VisualToolDrag(VideoDisplay *parent, VideoState const& video, wxToolBar * toolBar) VisualToolDrag::VisualToolDrag(VideoDisplay *parent, VideoState const& video, wxToolBar * toolBar)
: VisualTool<VisualToolDragDraggableFeature>(parent, video) : VisualTool<VisualToolDragDraggableFeature>(parent, video)
, toolBar(toolBar) , toolBar(toolBar)
, primary(NULL)
, toggleMoveOnMove(true) , toggleMoveOnMove(true)
{ {
toolBar->AddTool(BUTTON_TOGGLE_MOVE, _("Toggle between \\move and \\pos"), GETIMAGE(visual_move_conv_move_24)); toolBar->AddTool(BUTTON_TOGGLE_MOVE, _("Toggle between \\move and \\pos"), GETIMAGE(visual_move_conv_move_24));
@ -63,8 +67,6 @@ VisualToolDrag::VisualToolDrag(VideoDisplay *parent, VideoState const& video, wx
toolBar->Show(true); toolBar->Show(true);
} }
/// @brief Update toggle buttons
/// @return
void VisualToolDrag::UpdateToggleButtons() { void VisualToolDrag::UpdateToggleButtons() {
// Check which bitmap to use // Check which bitmap to use
bool toMove = true; bool toMove = true;
@ -91,7 +93,6 @@ void VisualToolDrag::UpdateToggleButtons() {
/// @brief Toggle button pressed /// @brief Toggle button pressed
/// @param event /// @param event
/// @return
void VisualToolDrag::OnSubTool(wxCommandEvent &event) { void VisualToolDrag::OnSubTool(wxCommandEvent &event) {
// Get line // Get line
AssDialogue *line = GetActiveDialogueLine(); AssDialogue *line = GetActiveDialogueLine();
@ -117,23 +118,21 @@ void VisualToolDrag::OnSubTool(wxCommandEvent &event) {
} }
} }
/// @brief Refresh
void VisualToolDrag::DoRefresh() { void VisualToolDrag::DoRefresh() {
UpdateToggleButtons(); UpdateToggleButtons();
} }
/// @brief Draw
void VisualToolDrag::Draw() { void VisualToolDrag::Draw() {
DrawAllFeatures(); DrawAllFeatures();
// Draw arrows // Draw arrows
for (std::list<VisualToolDragDraggableFeature>::iterator cur = features.begin(); cur != features.end(); ++cur) { for (std::list<VisualToolDragDraggableFeature>::iterator cur = features.begin(); cur != features.end(); ++cur) {
if (cur->type == DRAG_BIG_SQUARE) continue; if (cur->type == DRAG_START) continue;
VisualDraggableFeature *p2 = &*cur; VisualDraggableFeature *p2 = &*cur;
VisualDraggableFeature *p1 = cur->parent; VisualDraggableFeature *p1 = cur->parent;
// Has arrow? // Has arrow?
bool hasArrow = p2->type == DRAG_BIG_CIRCLE; bool hasArrow = p2->type == DRAG_END;
int arrowLen = hasArrow ? 10 : 0; int arrowLen = hasArrow ? 10 : 0;
// See if the distance between them is enough // See if the distance between them is enough
@ -176,6 +175,7 @@ void VisualToolDrag::Draw() {
/// @brief Populate list /// @brief Populate list
void VisualToolDrag::PopulateFeatureList() { void VisualToolDrag::PopulateFeatureList() {
ClearSelection(); ClearSelection();
primary = NULL;
features.clear(); features.clear();
// Get video data // Get video data
@ -205,7 +205,7 @@ void VisualToolDrag::PopulateFeatureList() {
feat.x = x1; feat.x = x1;
feat.y = y1; feat.y = y1;
feat.layer = 0; feat.layer = 0;
feat.type = DRAG_BIG_SQUARE; feat.type = DRAG_START;
feat.time = t1; feat.time = t1;
feat.line = diag; feat.line = diag;
feat.lineN = i; feat.lineN = i;
@ -217,7 +217,7 @@ void VisualToolDrag::PopulateFeatureList() {
feat.x = x2; feat.x = x2;
feat.y = y2; feat.y = y2;
feat.layer = 1; feat.layer = 1;
feat.type = DRAG_BIG_CIRCLE; feat.type = DRAG_END;
feat.time = t2; feat.time = t2;
feat.line = diag; feat.line = diag;
feat.lineN = i; feat.lineN = i;
@ -229,7 +229,7 @@ void VisualToolDrag::PopulateFeatureList() {
feat.x = torgx; feat.x = torgx;
feat.y = torgy; feat.y = torgy;
feat.layer = -1; feat.layer = -1;
feat.type = DRAG_BIG_TRIANGLE; feat.type = DRAG_ORIGIN;
feat.time = 0; feat.time = 0;
feat.line = diag; feat.line = diag;
feat.lineN = i; feat.lineN = i;
@ -239,6 +239,10 @@ void VisualToolDrag::PopulateFeatureList() {
} }
} }
} }
bool VisualToolDrag::InitializeDrag(VisualToolDragDraggableFeature *feature) {
if (feature->type != DRAG_ORIGIN) primary = feature;
return true;
}
/// @brief Update drag /// @brief Update drag
/// @param feature /// @param feature
@ -251,8 +255,7 @@ void VisualToolDrag::UpdateDrag(VisualToolDragDraggableFeature* feature) {
/// @brief Commit drag /// @brief Commit drag
/// @param feature /// @param feature
void VisualToolDrag::CommitDrag(VisualToolDragDraggableFeature* feature) { void VisualToolDrag::CommitDrag(VisualToolDragDraggableFeature* feature) {
// Origin if (feature->type == DRAG_ORIGIN) {
if (feature->type == DRAG_BIG_TRIANGLE) {
int x = feature->x; int x = feature->x;
int y = feature->y; int y = feature->y;
parent->ToScriptCoords(&x, &y); parent->ToScriptCoords(&x, &y);
@ -261,7 +264,7 @@ void VisualToolDrag::CommitDrag(VisualToolDragDraggableFeature* feature) {
} }
VisualToolDragDraggableFeature *p = feature->parent; VisualToolDragDraggableFeature *p = feature->parent;
if (feature->type == DRAG_BIG_CIRCLE) { if (feature->type == DRAG_END) {
std::swap(feature, p); std::swap(feature, p);
} }
@ -283,3 +286,53 @@ void VisualToolDrag::CommitDrag(VisualToolDragDraggableFeature* feature) {
SetOverride(feature->line, L"\\move", wxString::Format(L"(%i,%i,%i,%i,%i,%i)", x1, y1, x2, y2, feature->time, p->time)); SetOverride(feature->line, L"\\move", wxString::Format(L"(%i,%i,%i,%i,%i,%i)", x1, y1, x2, y2, feature->time, p->time));
} }
} }
void VisualToolDrag::Update() {
if (!leftDClick) return;
int dx, dy;
int vx = video.x;
int vy = video.y;
parent->ToScriptCoords(&vx, &vy);
if (primary) {
dx = primary->x;
dy = primary->y;
}
else {
AssDialogue* line = GetActiveDialogueLine();
if (!line) return;
GetLinePosition(line, dx, dy);
}
parent->ToScriptCoords(&dx, &dy);
dx -= vx;
dy -= vy;
SubtitlesGrid *grid = VideoContext::Get()->grid;
wxArrayInt sel = grid->GetSelection();
for (wxArrayInt::const_iterator cur = sel.begin(); cur != sel.end(); ++cur) {
AssDialogue* line = grid->GetDialogue(*cur);
if (!line) continue;
int x1, y1, x2, y2, t1 = INT_MIN, t2 = INT_MIN;
bool isMove;
GetLineMove(line, isMove, x1, y1, x2, y2, t1, t2);
if (isMove) {
parent->ToScriptCoords(&x1, &y1);
parent->ToScriptCoords(&x2, &y2);
if (t1 > INT_MIN && t2 > INT_MIN)
SetOverride(line, L"\\move", wxString::Format(L"(%i,%i,%i,%i,%i,%i)", x1 - dx, y1 - dy, x2 - dx, y2 - dy, t1, t2));
else
SetOverride(line, L"\\move", wxString::Format(L"(%i,%i,%i,%i)", x1, y1, x2, y2));
}
else {
GetLinePosition(line, x1, y1);
parent->ToScriptCoords(&x1, &y1);
SetOverride(line, L"\\pos", wxString::Format(L"(%i,%i)", x1 - dx, y1 - dy));
}
}
grid->ass->FlagAsModified(_("positioning"));
grid->CommitChanges(false,true);
grid->editBox->Update(false, true, false);
/// @todo: should just move the existing features rather than remaking them all
PopulateFeatureList();
}

View file

@ -63,19 +63,19 @@ public:
/// DOCME /// DOCME
class VisualToolDrag : public VisualTool<VisualToolDragDraggableFeature> { class VisualToolDrag : public VisualTool<VisualToolDragDraggableFeature> {
private: private:
/// DOCME wxToolBar *toolBar; /// The subtoolbar
wxToolBar *toolBar; VisualToolDragDraggableFeature* primary; /// The feature last clicked on
/// DOCME /// When the button is pressed, will it convert the line to a move (vs. from
/// move to pos)? Used to avoid changing the button's icon unnecessarily
bool toggleMoveOnMove; bool toggleMoveOnMove;
/// @brief DOCME
///
bool CanDrag() { return true; }
void PopulateFeatureList(); void PopulateFeatureList();
bool InitializeDrag(VisualToolDragDraggableFeature* feature);
void UpdateDrag(VisualToolDragDraggableFeature* feature); void UpdateDrag(VisualToolDragDraggableFeature* feature);
void CommitDrag(VisualToolDragDraggableFeature* feature); void CommitDrag(VisualToolDragDraggableFeature* feature);
/// Set the pos/move button to the correct icon based on the active line
void UpdateToggleButtons(); void UpdateToggleButtons();
void DoRefresh(); void DoRefresh();
@ -83,5 +83,6 @@ public:
VisualToolDrag(VideoDisplay *parent, VideoState const& video, wxToolBar *toolbar); VisualToolDrag(VideoDisplay *parent, VideoState const& video, wxToolBar *toolbar);
void Draw(); void Draw();
void Update();
void OnSubTool(wxCommandEvent &event); void OnSubTool(wxCommandEvent &event);
}; };