mirror of https://github.com/odrling/Aegisub
Make modifying multiple lines at once less brittle
Only count characters outside of override blocks for the relative positions to do the edits on each line, which handles the case where the lines have different lengths of stuff in override blocks but the same text.
This commit is contained in:
parent
7ed3fbade0
commit
780c93ed4d
|
@ -157,8 +157,15 @@ AssDialogue *paste_over(wxWindow *parent, std::vector<bool>& pasteOverOptions, A
|
||||||
return old_line;
|
return old_line;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename T>
|
struct parsed_line {
|
||||||
T get_value(std::vector<std::unique_ptr<AssDialogueBlock>> const& blocks, int blockn, T initial, std::string const& tag_name, std::string alt = "") {
|
AssDialogue *line;
|
||||||
|
std::vector<std::unique_ptr<AssDialogueBlock>> blocks;
|
||||||
|
|
||||||
|
parsed_line(AssDialogue *line) : line(line), blocks(line->ParseTags()) { }
|
||||||
|
parsed_line(parsed_line&& r) : line(r.line), blocks(std::move(r.blocks)) { }
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
T get_value(int blockn, T initial, std::string const& tag_name, std::string alt = "") const {
|
||||||
for (auto ovr : blocks | sliced(0, blockn + 1) | reversed | agi::of_type<AssDialogueBlockOverride>()) {
|
for (auto ovr : blocks | sliced(0, blockn + 1) | reversed | agi::of_type<AssDialogueBlockOverride>()) {
|
||||||
for (auto const& tag : ovr->Tags | reversed) {
|
for (auto const& tag : ovr->Tags | reversed) {
|
||||||
if (tag.Name == tag_name || tag.Name == alt)
|
if (tag.Name == tag_name || tag.Name == alt)
|
||||||
|
@ -166,58 +173,54 @@ T get_value(std::vector<std::unique_ptr<AssDialogueBlock>> const& blocks, int bl
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return initial;
|
return initial;
|
||||||
}
|
}
|
||||||
|
|
||||||
int block_at_pos(std::string const& text, int pos) {
|
int block_at_pos(int pos) const {
|
||||||
|
auto const& text = line->Text.get();
|
||||||
int n = 0;
|
int n = 0;
|
||||||
int max = text.size() - 1;
|
int max = text.size() - 1;
|
||||||
bool in_block = false;
|
bool in_block = false;
|
||||||
|
|
||||||
for (int i = 0; i <= pos && i <= max; ++i) {
|
for (int i = 0; i <= max; ++i) {
|
||||||
if (text[i] == '{') {
|
if (text[i] == '{') {
|
||||||
if (!in_block && i > 0)
|
if (!in_block && i > 0 && pos >= 0)
|
||||||
++n;
|
++n;
|
||||||
in_block = true;
|
in_block = true;
|
||||||
}
|
}
|
||||||
else if (text[i] == '}' && in_block) {
|
else if (text[i] == '}' && in_block) {
|
||||||
in_block = false;
|
in_block = false;
|
||||||
if (i != max && i != pos && i != pos -1 && (i+1 == max || text[i+1] != '{'))
|
if (pos > 0 && (i + 1 == max || text[i + 1] != '{'))
|
||||||
n++;
|
n++;
|
||||||
}
|
}
|
||||||
}
|
else if (!in_block) {
|
||||||
|
if (--pos == 0)
|
||||||
if (in_block) {
|
return n + (i < max && text[i + 1] == '{');
|
||||||
for (int i = pos + 1; i <= max; ++i) {
|
|
||||||
if (text[i] == '}') {
|
|
||||||
in_block = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return n - in_block;
|
return n - in_block;
|
||||||
}
|
}
|
||||||
|
|
||||||
int set_tag(AssDialogue *line, std::vector<std::unique_ptr<AssDialogueBlock>> &blocks, std::string const& tag, std::string const& value, int sel_start, int sel_end, bool at_end = false) {
|
int set_tag(std::string const& tag, std::string const& value, int norm_pos, int orig_pos) {
|
||||||
int start = at_end ? sel_end : sel_start;
|
int blockn = block_at_pos(norm_pos);
|
||||||
int blockn = block_at_pos(line->Text, start);
|
|
||||||
|
|
||||||
AssDialogueBlockPlain *plain = nullptr;
|
AssDialogueBlockPlain *plain = nullptr;
|
||||||
AssDialogueBlockOverride *ovr = nullptr;
|
AssDialogueBlockOverride *ovr = nullptr;
|
||||||
while (blockn >= 0) {
|
while (blockn >= 0 && !plain && !ovr) {
|
||||||
AssDialogueBlock *block = blocks[blockn].get();
|
AssDialogueBlock *block = blocks[blockn].get();
|
||||||
if (dynamic_cast<AssDialogueBlockDrawing*>(block))
|
switch (block->GetType()) {
|
||||||
--blockn;
|
case AssBlockType::PLAIN:
|
||||||
else if (dynamic_cast<AssDialogueBlockComment*>(block)) {
|
plain = static_cast<AssDialogueBlockPlain *>(block);
|
||||||
// Cursor is in a comment block, so try the previous block instead
|
|
||||||
--blockn;
|
|
||||||
start = line->Text.get().rfind('{', start);
|
|
||||||
}
|
|
||||||
else if ((plain = dynamic_cast<AssDialogueBlockPlain*>(block)))
|
|
||||||
break;
|
break;
|
||||||
else {
|
case AssBlockType::DRAWING:
|
||||||
ovr = dynamic_cast<AssDialogueBlockOverride*>(block);
|
--blockn;
|
||||||
assert(ovr);
|
break;
|
||||||
|
case AssBlockType::COMMENT:
|
||||||
|
--blockn;
|
||||||
|
orig_pos = line->Text.get().rfind('{', orig_pos);
|
||||||
|
break;
|
||||||
|
case AssBlockType::OVERRIDE:
|
||||||
|
ovr = static_cast<AssDialogueBlockOverride*>(block);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -225,12 +228,12 @@ int set_tag(AssDialogue *line, std::vector<std::unique_ptr<AssDialogueBlock>> &b
|
||||||
// If we didn't hit a suitable block for inserting the override just put
|
// If we didn't hit a suitable block for inserting the override just put
|
||||||
// it at the beginning of the line
|
// it at the beginning of the line
|
||||||
if (blockn < 0)
|
if (blockn < 0)
|
||||||
start = 0;
|
orig_pos = 0;
|
||||||
|
|
||||||
std::string insert(tag + value);
|
std::string insert(tag + value);
|
||||||
int shift = insert.size();
|
int shift = insert.size();
|
||||||
if (plain || blockn < 0) {
|
if (plain || blockn < 0) {
|
||||||
line->Text = line->Text.get().substr(0, start) + "{" + insert + "}" + line->Text.get().substr(start);
|
line->Text = line->Text.get().substr(0, orig_pos) + "{" + insert + "}" + line->Text.get().substr(orig_pos);
|
||||||
shift += 2;
|
shift += 2;
|
||||||
blocks = line->ParseTags();
|
blocks = line->ParseTags();
|
||||||
}
|
}
|
||||||
|
@ -262,17 +265,36 @@ int set_tag(AssDialogue *line, std::vector<std::unique_ptr<AssDialogueBlock>> &b
|
||||||
assert(false);
|
assert(false);
|
||||||
|
|
||||||
return shift;
|
return shift;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
int normalize_pos(std::string const& text, int pos) {
|
||||||
|
int plain_len = 0;
|
||||||
|
bool in_block = false;
|
||||||
|
|
||||||
|
for (int i = 0, max = text.size() - 1; i < pos && i <= max; ++i) {
|
||||||
|
if (text[i] == '{')
|
||||||
|
in_block = true;
|
||||||
|
if (!in_block)
|
||||||
|
++plain_len;
|
||||||
|
if (text[i] == '}' && in_block)
|
||||||
|
in_block = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return plain_len;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Func>
|
template<typename Func>
|
||||||
void update_lines(const agi::Context *c, wxString const& undo_msg, Func&& f) {
|
void update_lines(const agi::Context *c, wxString const& undo_msg, Func&& f) {
|
||||||
|
const auto active_line = c->selectionController->GetActiveLine();
|
||||||
const int sel_start = c->textSelectionController->GetSelectionStart();
|
const int sel_start = c->textSelectionController->GetSelectionStart();
|
||||||
const int sel_end = c->textSelectionController->GetSelectionEnd();
|
const int sel_end = c->textSelectionController->GetSelectionEnd();
|
||||||
const auto active_line = c->selectionController->GetActiveLine();
|
const int norm_sel_start = normalize_pos(active_line->Text, sel_start);
|
||||||
|
const int norm_sel_end = normalize_pos(active_line->Text, sel_end);
|
||||||
int active_sel_shift = 0;
|
int active_sel_shift = 0;
|
||||||
|
|
||||||
for (const auto line : c->selectionController->GetSelectedSet()) {
|
for (const auto line : c->selectionController->GetSelectedSet()) {
|
||||||
int shift = f(line, sel_start, sel_end);
|
int shift = f(line, sel_start, sel_end, norm_sel_start, norm_sel_end);
|
||||||
if (line == active_line)
|
if (line == active_line)
|
||||||
active_sel_shift = shift;
|
active_sel_shift = shift;
|
||||||
}
|
}
|
||||||
|
@ -284,78 +306,66 @@ void update_lines(const agi::Context *c, wxString const& undo_msg, Func&& f) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void toggle_override_tag(const agi::Context *c, bool (AssStyle::*field), const char *tag, wxString const& undo_msg) {
|
void toggle_override_tag(const agi::Context *c, bool (AssStyle::*field), const char *tag, wxString const& undo_msg) {
|
||||||
update_lines(c, undo_msg, [&](AssDialogue *line, int sel_start, int sel_end) {
|
update_lines(c, undo_msg, [&](AssDialogue *line, int sel_start, int sel_end, int norm_sel_start, int norm_sel_end) {
|
||||||
AssStyle const* const style = c->ass->GetStyle(line->Style);
|
AssStyle const* const style = c->ass->GetStyle(line->Style);
|
||||||
bool state = style ? style->*field : AssStyle().*field;
|
bool state = style ? style->*field : AssStyle().*field;
|
||||||
|
|
||||||
auto blocks = line->ParseTags();
|
parsed_line parsed(line);
|
||||||
int blockn = block_at_pos(line->Text, sel_start);
|
int blockn = parsed.block_at_pos(norm_sel_start);
|
||||||
|
|
||||||
state = get_value(blocks, blockn, state, tag);
|
state = parsed.get_value(blockn, state, tag);
|
||||||
|
|
||||||
int shift = set_tag(line, blocks, tag, state ? "0" : "1", sel_start, sel_end);
|
int shift = parsed.set_tag(tag, state ? "0" : "1", norm_sel_start, sel_start);
|
||||||
if (sel_start != sel_end)
|
if (sel_start != sel_end)
|
||||||
set_tag(line, blocks, tag, state ? "1" : "0", sel_start + shift, sel_end + shift, true);
|
parsed.set_tag(tag, state ? "1" : "0", norm_sel_end, sel_end + shift);
|
||||||
return shift;
|
return shift;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void show_color_picker(const agi::Context *c, agi::Color (AssStyle::*field), const char *tag, const char *alt, const char *alpha) {
|
void show_color_picker(const agi::Context *c, agi::Color (AssStyle::*field), const char *tag, const char *alt, const char *alpha) {
|
||||||
agi::Color initial_color;
|
agi::Color initial_color;
|
||||||
const int sel_start = c->textSelectionController->GetSelectionStart();
|
|
||||||
const int sel_end = c->textSelectionController->GetSelectionEnd();
|
|
||||||
const auto active_line = c->selectionController->GetActiveLine();
|
const auto active_line = c->selectionController->GetActiveLine();
|
||||||
int active_sel_start = sel_start, active_sel_end = sel_end;
|
const int sel_start = c->textSelectionController->GetSelectionStart();
|
||||||
|
const int sel_end = c->textSelectionController->GetSelectionStart();
|
||||||
struct line_info {
|
const int norm_sel_start = normalize_pos(active_line->Text, sel_start);
|
||||||
AssDialogue *line;
|
|
||||||
agi::Color color;
|
|
||||||
std::vector<std::unique_ptr<AssDialogueBlock>> blocks;
|
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
line_info(AssDialogue *line, agi::Color color, std::vector<std::unique_ptr<AssDialogueBlock>> blocks)
|
|
||||||
: line(line), color(color), blocks(std::move(blocks)) { }
|
|
||||||
line_info(line_info &&r) : line(r.line), color(r.color), blocks(std::move(r.blocks)) { }
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
auto const& sel = c->selectionController->GetSelectedSet();
|
auto const& sel = c->selectionController->GetSelectedSet();
|
||||||
|
using line_info = std::pair<agi::Color, parsed_line>;
|
||||||
std::vector<line_info> lines;
|
std::vector<line_info> lines;
|
||||||
for (auto line : sel) {
|
for (auto line : sel) {
|
||||||
AssStyle const* const style = c->ass->GetStyle(line->Style);
|
AssStyle const* const style = c->ass->GetStyle(line->Style);
|
||||||
agi::Color color = (style ? style->*field : AssStyle().*field);
|
agi::Color color = (style ? style->*field : AssStyle().*field);
|
||||||
|
|
||||||
auto blocks = line->ParseTags();
|
parsed_line parsed(line);
|
||||||
int blockn = block_at_pos(line->Text, sel_start);
|
int blockn = parsed.block_at_pos(norm_sel_start);
|
||||||
|
|
||||||
int a = get_value(blocks, blockn, (int)color.a, alpha, "\\alpha");
|
int a = parsed.get_value(blockn, (int)color.a, alpha, "\\alpha");
|
||||||
color = get_value(blocks, blockn, color, tag, alt);
|
color = parsed.get_value(blockn, color, tag, alt);
|
||||||
color.a = a;
|
color.a = a;
|
||||||
|
|
||||||
if (line == active_line)
|
if (line == active_line)
|
||||||
initial_color = color;
|
initial_color = color;
|
||||||
|
|
||||||
lines.push_back(line_info{line, color, std::move(blocks)});
|
lines.emplace_back(color, std::move(parsed));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int active_shift = 0;
|
||||||
int commit_id = -1;
|
int commit_id = -1;
|
||||||
bool ok = GetColorFromUser(c->parent, initial_color, true, [&](agi::Color new_color) {
|
bool ok = GetColorFromUser(c->parent, initial_color, true, [&](agi::Color new_color) {
|
||||||
for (auto& line : lines) {
|
for (auto& line : lines) {
|
||||||
int shift = set_tag(line.line, line.blocks, tag, new_color.GetAssOverrideFormatted(), sel_start, sel_end);
|
int shift = line.second.set_tag(tag, new_color.GetAssOverrideFormatted(), norm_sel_start, sel_start);
|
||||||
if (new_color.a != line.color.a) {
|
if (new_color.a != line.first.a) {
|
||||||
shift += set_tag(line.line, line.blocks, alpha, str(boost::format("&H%02X&") % (int)new_color.a), sel_start + shift, sel_end + shift);
|
shift += line.second.set_tag(alpha, str(boost::format("&H%02X&") % (int)new_color.a), norm_sel_start, sel_start + shift);
|
||||||
line.color.a = new_color.a;
|
line.first.a = new_color.a;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (line.line == active_line) {
|
if (line.second.line == active_line)
|
||||||
active_sel_start = sel_start + shift;
|
active_shift = shift;
|
||||||
active_sel_end = sel_end + shift;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
commit_id = c->ass->Commit(_("set color"), AssFile::COMMIT_DIAG_TEXT, commit_id, sel.size() == 1 ? *sel.begin() : nullptr);
|
commit_id = c->ass->Commit(_("set color"), AssFile::COMMIT_DIAG_TEXT, commit_id, sel.size() == 1 ? *sel.begin() : nullptr);
|
||||||
if (active_sel_start >= 0 && active_sel_end >= 0)
|
if (active_shift)
|
||||||
c->textSelectionController->SetSelection(active_sel_start, active_sel_end);
|
c->textSelectionController->SetSelection(sel_start + active_shift, sel_start + active_shift);
|
||||||
});
|
});
|
||||||
c->ass->Commit(_("set color"), AssFile::COMMIT_DIAG_TEXT, commit_id, sel.size() == 1 ? *sel.begin() : nullptr);
|
c->ass->Commit(_("set color"), AssFile::COMMIT_DIAG_TEXT, commit_id, sel.size() == 1 ? *sel.begin() : nullptr);
|
||||||
|
|
||||||
|
@ -469,35 +479,36 @@ struct edit_font final : public Command {
|
||||||
STR_HELP("Select a font face and size")
|
STR_HELP("Select a font face and size")
|
||||||
|
|
||||||
void operator()(agi::Context *c) override {
|
void operator()(agi::Context *c) override {
|
||||||
const int insertion_point = c->textSelectionController->GetInsertionPoint();
|
const parsed_line active(c->selectionController->GetActiveLine());
|
||||||
std::vector<std::unique_ptr<AssDialogueBlock>> blocks;
|
const int insertion_point = normalize_pos(active.line->Text, c->textSelectionController->GetInsertionPoint());
|
||||||
auto font_for_line = [&](AssDialogue *const line) -> wxFont {
|
|
||||||
blocks = line->ParseTags();
|
|
||||||
const int blockn = block_at_pos(line->Text, insertion_point);
|
|
||||||
|
|
||||||
const AssStyle *style = c->ass->GetStyle(line->Style);
|
auto font_for_line = [&](parsed_line const& line) -> wxFont {
|
||||||
|
const int blockn = line.block_at_pos(insertion_point);
|
||||||
|
|
||||||
|
const AssStyle *style = c->ass->GetStyle(line.line->Style);
|
||||||
const AssStyle default_style;
|
const AssStyle default_style;
|
||||||
if (!style)
|
if (!style)
|
||||||
style = &default_style;
|
style = &default_style;
|
||||||
|
|
||||||
return wxFont(
|
return wxFont(
|
||||||
get_value(blocks, blockn, (int)style->fontsize, "\\fs"),
|
line.get_value(blockn, (int)style->fontsize, "\\fs"),
|
||||||
wxFONTFAMILY_DEFAULT,
|
wxFONTFAMILY_DEFAULT,
|
||||||
get_value(blocks, blockn, style->italic, "\\i") ? wxFONTSTYLE_ITALIC : wxFONTSTYLE_NORMAL,
|
line.get_value(blockn, style->italic, "\\i") ? wxFONTSTYLE_ITALIC : wxFONTSTYLE_NORMAL,
|
||||||
get_value(blocks, blockn, style->bold, "\\b") ? wxFONTWEIGHT_BOLD : wxFONTWEIGHT_NORMAL,
|
line.get_value(blockn, style->bold, "\\b") ? wxFONTWEIGHT_BOLD : wxFONTWEIGHT_NORMAL,
|
||||||
get_value(blocks, blockn, style->underline, "\\u"),
|
line.get_value(blockn, style->underline, "\\u"),
|
||||||
to_wx(get_value(blocks, blockn, style->font, "\\fn")));
|
to_wx(line.get_value(blockn, style->font, "\\fn")));
|
||||||
};
|
};
|
||||||
|
|
||||||
const wxFont initial = font_for_line(c->selectionController->GetActiveLine());
|
const wxFont initial = font_for_line(active);
|
||||||
const wxFont font = wxGetFontFromUser(c->parent, initial);
|
const wxFont font = wxGetFontFromUser(c->parent, initial);
|
||||||
if (!font.Ok() || font == initial) return;
|
if (!font.Ok() || font == initial) return;
|
||||||
|
|
||||||
update_lines(c, _("set font"), [&](AssDialogue *line, int sel_start, int sel_end) {
|
update_lines(c, _("set font"), [&](AssDialogue *line, int sel_start, int sel_end, int norm_sel_start, int norm_sel_end) {
|
||||||
const wxFont startfont = font_for_line(line);
|
parsed_line parsed(line);
|
||||||
|
const wxFont startfont = font_for_line(parsed);
|
||||||
int shift = 0;
|
int shift = 0;
|
||||||
auto do_set_tag = [&](const char *tag_name, std::string const& value) {
|
auto do_set_tag = [&](const char *tag_name, std::string const& value) {
|
||||||
shift += set_tag(line, blocks, tag_name, value, sel_start + shift, sel_end + shift);
|
shift += parsed.set_tag(tag_name, value, norm_sel_start, sel_start + shift);
|
||||||
};
|
};
|
||||||
|
|
||||||
if (font.GetFaceName() != startfont.GetFaceName())
|
if (font.GetFaceName() != startfont.GetFaceName())
|
||||||
|
|
Loading…
Reference in New Issue