Load Shift Times settings from history on double-click

Redesign how shift times history is saved. Previously it stored the
localized strings in the history file, which are not particularly
parsable as the format may differ between locales. Rather than doing
this, store the raw settings in a json file, and generate the history
strings on display. In addition to making it much easier to load old
settings, this makes it so that the history is always displayed using
the current locale, rather than the locale in which the shifting was
done.

Closes #1427.

Originally committed to SVN as r6363.
This commit is contained in:
Thomas Goyne 2012-01-25 23:09:45 +00:00
parent 2094814077
commit 214079af58
2 changed files with 121 additions and 49 deletions

View file

@ -40,7 +40,10 @@
#include <libaegisub/io.h> #include <libaegisub/io.h>
#include <libaegisub/log.h> #include <libaegisub/log.h>
#include <libaegisub/scoped_ptr.h>
#include <libaegisub/cajun/elements.h>
#include <libaegisub/cajun/reader.h>
#include <libaegisub/cajun/writer.h>
#include "ass_dialogue.h" #include "ass_dialogue.h"
#include "ass_file.h" #include "ass_file.h"
@ -55,10 +58,53 @@
#include "utils.h" #include "utils.h"
#include "video_context.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) DialogShiftTimes::DialogShiftTimes(agi::Context *context)
: wxDialog(context->parent, -1, _("Shift Times")) : wxDialog(context->parent, -1, _("Shift Times"))
, context(context) , context(context)
, history_filename(STD_STR(StandardPaths::DecodePath("?user/shift_history.txt"))) , history_filename(STD_STR(StandardPaths::DecodePath("?user/shift_history.json")))
, history(new json::Array)
, timecodes_loaded_slot(context->videoController->AddTimecodesListener(&DialogShiftTimes::OnTimecodesLoaded, this)) , timecodes_loaded_slot(context->videoController->AddTimecodesListener(&DialogShiftTimes::OnTimecodesLoaded, this))
{ {
SetIcon(BitmapToIcon(GETIMAGE(shift_times_toolbutton_24))); SetIcon(BitmapToIcon(GETIMAGE(shift_times_toolbutton_24)));
@ -90,7 +136,7 @@ DialogShiftTimes::DialogShiftTimes(agi::Context *context)
wxString time_field_vals[] = { _("Start a&nd End times"), _("&Start times only"), _("&End times only") }; 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); time_fields = new wxRadioBox(this, -1, _("Times"), wxDefaultPosition, wxDefaultSize, 3, time_field_vals, 1);
history = new wxListBox(this, -1, wxDefaultPosition, wxSize(350, 100), 0, NULL, wxLB_HSCROLL); history_box = new wxListBox(this, -1, wxDefaultPosition, wxSize(350, 100), 0, NULL, wxLB_HSCROLL);
wxButton *clear_button = new wxButton(this, -1, _("&Clear")); wxButton *clear_button = new wxButton(this, -1, _("&Clear"));
clear_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::OnClear, this); clear_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::OnClear, this);
@ -134,7 +180,7 @@ DialogShiftTimes::DialogShiftTimes(agi::Context *context)
left_sizer->Add(time_fields, wxSizerFlags().Expand()); left_sizer->Add(time_fields, wxSizerFlags().Expand());
wxSizer *history_sizer = new wxStaticBoxSizer(wxVERTICAL, this, _("History")); wxSizer *history_sizer = new wxStaticBoxSizer(wxVERTICAL, this, _("History"));
history_sizer->Add(history, wxSizerFlags(1).Expand()); history_sizer->Add(history_box, wxSizerFlags(1).Expand());
history_sizer->Add(clear_button, wxSizerFlags().Expand().Border(wxTOP)); history_sizer->Add(clear_button, wxSizerFlags().Expand().Border(wxTOP));
wxSizer *top_sizer = new wxBoxSizer(wxHORIZONTAL); wxSizer *top_sizer = new wxBoxSizer(wxHORIZONTAL);
@ -150,6 +196,7 @@ DialogShiftTimes::DialogShiftTimes(agi::Context *context)
Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::Process, this, wxID_OK); Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::Process, this, wxID_OK);
Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::OnClose, this, wxID_CANCEL); Bind(wxEVT_COMMAND_BUTTON_CLICKED, &DialogShiftTimes::OnClose, this, wxID_CANCEL);
Bind(wxEVT_COMMAND_BUTTON_CLICKED, std::tr1::bind(&HelpButton::OpenPage, "Shift Times"), wxID_HELP); Bind(wxEVT_COMMAND_BUTTON_CLICKED, std::tr1::bind(&HelpButton::OpenPage, "Shift Times"), wxID_HELP);
history_box->Bind(wxEVT_COMMAND_LISTBOX_DOUBLECLICKED, &DialogShiftTimes::OnHistoryClick, this);
context->selectionController->AddSelectionListener(this); context->selectionController->AddSelectionListener(this);
} }
@ -185,7 +232,8 @@ void DialogShiftTimes::OnSelectedSetChanged(Selection const&, Selection const&)
void DialogShiftTimes::OnClear(wxCommandEvent &) { void DialogShiftTimes::OnClear(wxCommandEvent &) {
wxRemoveFile(lagi_wxString(history_filename)); wxRemoveFile(lagi_wxString(history_filename));
history->Clear(); history_box->Clear();
history->clear();
} }
void DialogShiftTimes::OnClose(wxCommandEvent &) { void DialogShiftTimes::OnClose(wxCommandEvent &) {
@ -212,39 +260,47 @@ void DialogShiftTimes::OnByFrames(wxCommandEvent &) {
shift_frames->Enable(true); shift_frames->Enable(true);
} }
void DialogShiftTimes::SaveHistory(std::vector<std::pair<int, int> > const& shifted_blocks) { void DialogShiftTimes::OnHistoryClick(wxCommandEvent &evt) {
wxString filename = wxFileName(context->ass->filename).GetFullName(); size_t entry = evt.GetInt();
int fields = time_fields->GetSelection(); if (entry >= history->size()) return;
wxString new_line = wxString::Format("%s, %s %s, %s, ", json::Object& obj = (*history)[entry];
filename.empty() ? _("unsaved") : filename, if (obj["is by time"]) {
shift_by_time->GetValue() ? shift_time->GetValue() : shift_frames->GetValue() + _(" frames"), shift_time->SetValue(lagi_wxString(obj["amount"]));
shift_backward->GetValue() ? _("backward") : _("forward"), shift_by_time->SetValue(true);
fields == 0 ? _("s+e") : fields == 1 ? _("s") : _("e")); OnByTime(evt);
}
int sel_mode = selection_mode->GetSelection();
if (sel_mode == 0)
new_line += _("all");
else if (sel_mode == 2)
new_line += wxString::Format(_("from %d onward"), shifted_blocks.front().first);
else { else {
new_line += _("sel "); shift_frames->SetValue(lagi_wxString(obj["amount"]));
for (size_t i = 0; i < shifted_blocks.size(); ++i) { if (shift_by_frames->IsEnabled()) {
std::pair<int, int> const& b = shifted_blocks[i]; shift_by_frames->SetValue(true);
wxString term = i == shifted_blocks.size() - 1 ? "" : ";"; OnByFrames(evt);
if (b.first == b.second)
new_line += wxString::Format("%d%s", b.first, term);
else
new_line += wxString::Format("%d-%d%s", b.first, b.second, term);
} }
} }
try { if (obj["is backward"])
agi::io::Save file(history_filename); shift_backward->SetValue(true);
else
shift_forward->SetValue(true);
for (size_t i = 0; i < history->GetCount(); ++i) selection_mode->SetSelection((int64_t)obj["mode"]);
file.Get() << history->GetString(i).utf8_str() << std::endl; time_fields->SetSelection((int64_t)obj["fields"]);
file.Get() << new_line.utf8_str() << std::endl; }
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) { catch (agi::FileSystemError const& e) {
LOG_E("dialog_shift_times/save_history") << "Cannot save shift times history: " << e.GetChainedMessage(); LOG_E("dialog_shift_times/save_history") << "Cannot save shift times history: " << e.GetChainedMessage();
@ -252,27 +308,27 @@ void DialogShiftTimes::SaveHistory(std::vector<std::pair<int, int> > const& shif
} }
void DialogShiftTimes::LoadHistory() { void DialogShiftTimes::LoadHistory() {
history->Clear(); history_box->Clear();
history->Freeze(); history_box->Freeze();
try { try {
agi::scoped_ptr<std::istream> file(agi::io::Open(history_filename)); agi::scoped_ptr<std::istream> file(agi::io::Open(history_filename));
std::string buffer; json::UnknownElement root;
while(!file->eof()) { json::Reader::Read(root, *file);
getline(*file, buffer); *history = root;
if (buffer.size())
history->Insert(lagi_wxString(buffer), 0); for (json::Array::iterator it = history->begin(); it != history->end(); ++it)
} history_box->Append(get_history_string(*it));
} }
catch (agi::FileSystemError const& e) { catch (agi::FileSystemError const& e) {
LOG_E("dialog_shift_times/save_history") << "Cannot load shift times history: " << e.GetChainedMessage(); LOG_D("dialog_shift_times/load_history") << "Cannot load shift times history: " << e.GetChainedMessage();
} }
catch (...) { catch (...) {
history->Thaw(); history_box->Thaw();
throw; throw;
} }
history->Thaw(); history_box->Thaw();
} }
void DialogShiftTimes::Process(wxCommandEvent &) { void DialogShiftTimes::Process(wxCommandEvent &) {
@ -303,7 +359,7 @@ void DialogShiftTimes::Process(wxCommandEvent &) {
// Track which rows were shifted for the log // Track which rows were shifted for the log
int row_number = 0; int row_number = 0;
int block_start = 0; int block_start = 0;
std::vector<std::pair<int, int> > shifted_blocks; json::Array shifted_blocks;
for (entryIter it = context->ass->Line.begin(); it != context->ass->Line.end(); ++it) { for (entryIter it = context->ass->Line.begin(); it != context->ass->Line.end(); ++it) {
AssDialogue *line = dynamic_cast<AssDialogue*>(*it); AssDialogue *line = dynamic_cast<AssDialogue*>(*it);
@ -312,7 +368,10 @@ void DialogShiftTimes::Process(wxCommandEvent &) {
if (!sel.count(line)) { if (!sel.count(line)) {
if (block_start) { if (block_start) {
shifted_blocks.push_back(std::make_pair(block_start, row_number - 1)); json::Object block;
block["start"] = block_start;
block["end"] = row_number - 1;
shifted_blocks.push_back(block);
block_start = 0; block_start = 0;
} }
if (mode == 1) continue; if (mode == 1) continue;
@ -329,8 +388,12 @@ void DialogShiftTimes::Process(wxCommandEvent &) {
context->ass->Commit(_("shifting"), AssFile::COMMIT_DIAG_TIME); context->ass->Commit(_("shifting"), AssFile::COMMIT_DIAG_TIME);
if (block_start) if (block_start) {
shifted_blocks.push_back(std::make_pair(block_start, row_number - 1)); json::Object block;
block["start"] = block_start;
block["end"] = row_number - 1;
shifted_blocks.push_back(block);
}
SaveHistory(shifted_blocks); SaveHistory(shifted_blocks);
Close(); Close();

View file

@ -22,9 +22,12 @@
/// ///
#ifndef AGI_PRE #ifndef AGI_PRE
#include <deque>
#include <wx/dialog.h> #include <wx/dialog.h>
#endif #endif
#include <libaegisub/scoped_ptr.h>
#include <libaegisub/signal.h> #include <libaegisub/signal.h>
#include <libaegisub/vfr.h> #include <libaegisub/vfr.h>
@ -37,6 +40,10 @@ class wxRadioBox;
class wxRadioButton; class wxRadioButton;
class wxTextCtrl; class wxTextCtrl;
namespace agi { struct Context; } namespace agi { struct Context; }
namespace json {
class UnknownElement;
typedef std::deque<UnknownElement> Array;
}
/// DOCME /// DOCME
/// @class DialogShiftTimes /// @class DialogShiftTimes
@ -47,6 +54,7 @@ class DialogShiftTimes : public wxDialog, private SelectionListener<AssDialogue>
agi::Context *context; agi::Context *context;
std::string history_filename; std::string history_filename;
agi::scoped_ptr<json::Array> history;
agi::vfr::Framerate fps; agi::vfr::Framerate fps;
agi::signal::Connection timecodes_loaded_slot; agi::signal::Connection timecodes_loaded_slot;
@ -58,9 +66,9 @@ class DialogShiftTimes : public wxDialog, private SelectionListener<AssDialogue>
wxRadioButton *shift_backward; wxRadioButton *shift_backward;
wxRadioBox *selection_mode; wxRadioBox *selection_mode;
wxRadioBox *time_fields; wxRadioBox *time_fields;
wxListBox *history; wxListBox *history_box;
void SaveHistory(std::vector<std::pair<int, int> > const& shifted_blocks); void SaveHistory(json::Array const& shifted_blocks);
void LoadHistory(); void LoadHistory();
void Process(wxCommandEvent&); void Process(wxCommandEvent&);
int Shift(int initial_time, int shift, bool by_time, agi::vfr::Time type); int Shift(int initial_time, int shift, bool by_time, agi::vfr::Time type);
@ -69,6 +77,7 @@ class DialogShiftTimes : public wxDialog, private SelectionListener<AssDialogue>
void OnClose(wxCommandEvent&); void OnClose(wxCommandEvent&);
void OnByTime(wxCommandEvent&); void OnByTime(wxCommandEvent&);
void OnByFrames(wxCommandEvent&); void OnByFrames(wxCommandEvent&);
void OnHistoryClick(wxCommandEvent&);
void OnActiveLineChanged(AssDialogue*) { } void OnActiveLineChanged(AssDialogue*) { }
void OnSelectedSetChanged(Selection const&, Selection const&); void OnSelectedSetChanged(Selection const&, Selection const&);