You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lnav/src/text_overlay_menu.cc

184 lines
6.8 KiB
C++

/**
* Copyright (c) 2024, Timothy Stack
*
* 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 Timothy Stack 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 REGENTS 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 REGENTS 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.
*/
#include "text_overlay_menu.hh"
#include "command_executor.hh"
#include "config.h"
#include "lnav.hh"
#include "md4cpp.hh"
#include "sysclip.hh"
#include "textview_curses.hh"
using namespace md4cpp::literals;
using namespace lnav::roles::literals;
std::vector<attr_line_t>
text_overlay_menu::list_overlay_menu(const listview_curses& lv, vis_line_t row)
{
static constexpr auto MENU_WIDTH = 25;
const auto* tc = dynamic_cast<const textview_curses*>(&lv);
std::vector<attr_line_t> retval;
if (tc->tc_text_selection_active || !tc->tc_selected_text) {
return retval;
}
const auto* tss = tc->get_sub_source();
const auto& sti = tc->tc_selected_text.value();
const auto supports_filtering
= tss != nullptr && tss->tss_supports_filtering;
if (sti.sti_line != row) {
return retval;
}
auto title = " Actions "_status_title;
auto left = std::max(0, sti.sti_x - 2);
auto dim = lv.get_dimensions();
if (left + MENU_WIDTH >= dim.second) {
left = dim.second - MENU_WIDTH;
}
this->los_menu_items.clear();
auto is_link = !sti.sti_href.empty();
auto menu_line = vis_line_t{1};
if (is_link) {
auto ta = text_attrs{};
ta.ta_attrs |= A_UNDERLINE;
auto href_al
= attr_line_t(" Link: ")
.append(lnav::roles::table_header(sti.sti_href))
.with_attr_for_all(VC_ROLE.value(role_t::VCR_STATUS_INFO))
.with_attr_for_all(VC_STYLE.value(ta));
retval.emplace_back(href_al);
menu_line += 1_vl;
}
retval.emplace_back(attr_line_t().pad_to(left).append(title));
{
attr_line_t al;
int start = left;
if (is_link || supports_filtering) {
if (is_link) {
al.append(":floppy_disk:"_emoji)
.append(" Open in lnav")
.append(" ");
} else {
al.append(" ").append("\u2714 Filter-in"_ok).append(" ");
}
this->los_menu_items.emplace_back(
menu_line,
line_range{start, start + (int) al.length()},
[is_link, sti](const std::string& value) {
const auto cmd = is_link
? ":open $href"
: fmt::format(FMT_STRING(":filter-in {}"),
lnav::pcre2pp::quote(value));
lnav_data.ld_exec_context
.with_provenance(exec_context::mouse_input{})
->execute_with(cmd,
std::make_pair("href", sti.sti_href));
});
start += al.length();
}
if (is_link) {
al.append(" ");
} else {
al.append(":mag_right:"_emoji).append(" Search ");
}
al.with_attr_for_all(VC_ROLE.value(role_t::VCR_STATUS));
if (!is_link) {
this->los_menu_items.emplace_back(
menu_line,
line_range{start, start + (int) al.length()},
[](const std::string& value) {
auto cmd = fmt::format(FMT_STRING("/{}"),
lnav::pcre2pp::quote(value));
lnav_data.ld_exec_context
.with_provenance(exec_context::mouse_input{})
->execute(cmd);
});
}
retval.emplace_back(attr_line_t().pad_to(left).append(al));
}
menu_line += 1_vl;
{
attr_line_t al;
int start = left;
if (is_link || supports_filtering) {
if (is_link) {
al.append(":globe_with_meridians:"_emoji).append(" Open ");
} else {
al.append(" ").append("\u2718 Filter-out"_error).append(" ");
}
this->los_menu_items.emplace_back(
menu_line,
line_range{start, start + (int) al.length()},
[is_link, sti](const std::string& value) {
auto cmd = is_link
? ":xopen $href"
: fmt::format(FMT_STRING(":filter-out {}"),
lnav::pcre2pp::quote(value));
lnav_data.ld_exec_context
.with_provenance(exec_context::mouse_input{})
->execute_with(cmd,
std::make_pair("href", sti.sti_href));
});
start += al.length();
}
al.append(":clipboard:"_emoji)
.append(is_link ? " Copy link " : " Copy ")
.with_attr_for_all(VC_ROLE.value(role_t::VCR_STATUS));
this->los_menu_items.emplace_back(
menu_line,
line_range{start, start + (int) al.length()},
[](const std::string& value) {
auto clip_res = sysclip::open(sysclip::type_t::GENERAL);
if (clip_res.isErr()) {
log_error("unable to open clipboard: %s",
clip_res.unwrapErr().c_str());
return;
}
auto clip_pipe = clip_res.unwrap();
fwrite(value.c_str(), 1, value.length(), clip_pipe.in());
});
retval.emplace_back(attr_line_t().pad_to(left).append(al));
}
return retval;
}