2021-03-26 21:55:40 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
|
2021-07-07 02:36:23 +03:00
|
|
|
* Copyright (c) 2021, Aziz Berkay Yesilyurt <abyesilyurt@gmail.com>
|
2022-01-25 04:22:03 +03:00
|
|
|
* Copyright (c) 2022, Alex Major
|
2021-03-26 21:55:40 +03:00
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2021-03-26 21:55:40 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <AK/Format.h>
|
2021-07-07 02:36:23 +03:00
|
|
|
#include <AK/Optional.h>
|
2021-08-01 06:22:27 +03:00
|
|
|
#include <AK/URL.h>
|
2021-03-26 21:55:40 +03:00
|
|
|
#include <LibCore/ArgsParser.h>
|
|
|
|
#include <LibCore/DateTime.h>
|
2022-10-07 22:56:37 +03:00
|
|
|
#include <LibCore/Process.h>
|
2023-05-20 11:14:22 +03:00
|
|
|
#include <LibFileSystem/FileSystem.h>
|
2021-03-26 21:55:40 +03:00
|
|
|
#include <LibGUI/Application.h>
|
2021-03-26 23:30:32 +03:00
|
|
|
#include <LibGUI/Clipboard.h>
|
2022-02-25 13:39:33 +03:00
|
|
|
#include <LibGUI/ConnectionToWindowServer.h>
|
2021-07-07 02:36:23 +03:00
|
|
|
#include <LibGUI/Painter.h>
|
|
|
|
#include <LibGUI/Widget.h>
|
|
|
|
#include <LibGUI/Window.h>
|
2023-03-21 21:58:06 +03:00
|
|
|
#include <LibGfx/ImageFormats/PNGWriter.h>
|
2021-07-07 02:36:23 +03:00
|
|
|
#include <LibGfx/Palette.h>
|
2022-01-25 04:22:03 +03:00
|
|
|
#include <LibMain/Main.h>
|
2021-03-27 12:58:47 +03:00
|
|
|
#include <unistd.h>
|
2021-03-26 21:55:40 +03:00
|
|
|
|
2021-07-07 02:36:23 +03:00
|
|
|
class SelectableLayover final : public GUI::Widget {
|
|
|
|
C_OBJECT(SelectableLayover)
|
|
|
|
public:
|
|
|
|
virtual ~SelectableLayover() override {};
|
|
|
|
|
|
|
|
Gfx::IntRect region() const
|
|
|
|
{
|
|
|
|
return m_region;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2021-11-01 01:38:04 +03:00
|
|
|
SelectableLayover(GUI::Window* window)
|
|
|
|
: m_window(window)
|
|
|
|
, m_background_color(palette().threed_highlight().with_alpha(128))
|
|
|
|
{
|
|
|
|
set_override_cursor(Gfx::StandardCursor::Crosshair);
|
|
|
|
}
|
|
|
|
|
2021-07-07 02:36:23 +03:00
|
|
|
virtual void mousedown_event(GUI::MouseEvent& event) override
|
|
|
|
{
|
2021-10-27 14:20:27 +03:00
|
|
|
if (event.button() == GUI::MouseButton::Primary)
|
2021-07-07 02:36:23 +03:00
|
|
|
m_anchor_point = event.position();
|
|
|
|
};
|
|
|
|
|
|
|
|
virtual void mousemove_event(GUI::MouseEvent& event) override
|
|
|
|
{
|
|
|
|
if (m_anchor_point.has_value()) {
|
|
|
|
m_region = Gfx::IntRect::from_two_points(*m_anchor_point, event.position());
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
virtual void mouseup_event(GUI::MouseEvent& event) override
|
|
|
|
{
|
2021-10-27 14:20:27 +03:00
|
|
|
if (event.button() == GUI::MouseButton::Primary)
|
2021-07-07 02:36:23 +03:00
|
|
|
m_window->close();
|
|
|
|
};
|
|
|
|
|
|
|
|
virtual void paint_event(GUI::PaintEvent&) override
|
|
|
|
{
|
2021-07-09 12:42:03 +03:00
|
|
|
GUI::Painter painter(*this);
|
|
|
|
painter.clear_rect(m_window->rect(), Gfx::Color::Transparent);
|
|
|
|
painter.fill_rect(m_window->rect(), m_background_color);
|
|
|
|
|
|
|
|
if (m_region.is_empty())
|
2021-07-07 02:36:23 +03:00
|
|
|
return;
|
|
|
|
|
2021-07-09 12:42:03 +03:00
|
|
|
painter.clear_rect(m_region, Gfx::Color::Transparent);
|
2021-07-07 02:36:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void keydown_event(GUI::KeyEvent& event) override
|
|
|
|
{
|
|
|
|
if (event.key() == Key_Escape) {
|
|
|
|
m_region = Gfx::IntRect();
|
|
|
|
m_window->close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Optional<Gfx::IntPoint> m_anchor_point;
|
|
|
|
Gfx::IntRect m_region;
|
|
|
|
GUI::Window* m_window = nullptr;
|
|
|
|
Gfx::Color const m_background_color;
|
|
|
|
};
|
|
|
|
|
2022-01-25 04:22:03 +03:00
|
|
|
ErrorOr<int> serenity_main(Main::Arguments arguments)
|
2021-03-26 21:55:40 +03:00
|
|
|
{
|
|
|
|
Core::ArgsParser args_parser;
|
|
|
|
|
2022-12-04 21:02:33 +03:00
|
|
|
DeprecatedString output_path;
|
2021-03-26 23:30:32 +03:00
|
|
|
bool output_to_clipboard = false;
|
2021-07-06 19:31:59 +03:00
|
|
|
unsigned delay = 0;
|
2021-07-07 02:36:23 +03:00
|
|
|
bool select_region = false;
|
2022-10-07 22:56:37 +03:00
|
|
|
bool edit_image = false;
|
2021-06-19 06:42:25 +03:00
|
|
|
int screen = -1;
|
2021-03-26 23:30:32 +03:00
|
|
|
|
2021-03-26 21:55:40 +03:00
|
|
|
args_parser.add_positional_argument(output_path, "Output filename", "output", Core::ArgsParser::Required::No);
|
2021-03-26 23:30:32 +03:00
|
|
|
args_parser.add_option(output_to_clipboard, "Output to clipboard", "clipboard", 'c');
|
2021-03-27 12:58:47 +03:00
|
|
|
args_parser.add_option(delay, "Seconds to wait before taking a screenshot", "delay", 'd', "seconds");
|
2021-06-19 06:42:25 +03:00
|
|
|
args_parser.add_option(screen, "The index of the screen (default: -1 for all screens)", "screen", 's', "index");
|
2021-07-07 02:36:23 +03:00
|
|
|
args_parser.add_option(select_region, "Select a region to capture", "region", 'r');
|
2022-10-07 22:56:37 +03:00
|
|
|
args_parser.add_option(edit_image, "Open in PixelPaint", "edit", 'e');
|
2021-03-26 21:55:40 +03:00
|
|
|
|
2022-01-25 04:22:03 +03:00
|
|
|
args_parser.parse(arguments);
|
2021-03-26 21:55:40 +03:00
|
|
|
|
|
|
|
if (output_path.is_empty()) {
|
2022-12-06 04:12:49 +03:00
|
|
|
output_path = Core::DateTime::now().to_deprecated_string("screenshot-%Y-%m-%d-%H-%M-%S.png"sv);
|
2021-03-26 21:55:40 +03:00
|
|
|
}
|
|
|
|
|
2023-05-05 07:24:53 +03:00
|
|
|
auto app = TRY(GUI::Application::create(arguments));
|
2021-11-03 23:39:45 +03:00
|
|
|
Optional<Gfx::IntRect> crop_region;
|
2021-07-07 02:36:23 +03:00
|
|
|
if (select_region) {
|
|
|
|
auto window = GUI::Window::construct();
|
2023-01-06 19:48:37 +03:00
|
|
|
auto container = TRY(window->set_main_widget<SelectableLayover>(window));
|
2021-07-07 02:36:23 +03:00
|
|
|
|
|
|
|
window->set_title("shot");
|
2021-07-09 12:42:03 +03:00
|
|
|
window->set_has_alpha_channel(true);
|
2021-07-07 02:36:23 +03:00
|
|
|
window->set_fullscreen(true);
|
|
|
|
window->show();
|
|
|
|
app->exec();
|
|
|
|
|
2023-01-06 19:48:37 +03:00
|
|
|
crop_region = container->region();
|
2021-11-03 23:39:45 +03:00
|
|
|
if (crop_region.value().is_empty()) {
|
2021-07-07 02:36:23 +03:00
|
|
|
dbgln("cancelled...");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-27 12:58:47 +03:00
|
|
|
sleep(delay);
|
2021-06-19 06:42:25 +03:00
|
|
|
Optional<u32> screen_index;
|
|
|
|
if (screen >= 0)
|
|
|
|
screen_index = (u32)screen;
|
|
|
|
dbgln("getting screenshot...");
|
2022-02-25 13:39:33 +03:00
|
|
|
auto shared_bitmap = GUI::ConnectionToWindowServer::the().get_screen_bitmap(crop_region, screen_index);
|
2021-06-19 06:42:25 +03:00
|
|
|
dbgln("got screenshot");
|
2021-03-26 21:55:40 +03:00
|
|
|
|
2021-07-07 02:36:23 +03:00
|
|
|
RefPtr<Gfx::Bitmap> bitmap = shared_bitmap.bitmap();
|
2021-03-26 21:55:40 +03:00
|
|
|
if (!bitmap) {
|
|
|
|
warnln("Failed to grab screenshot");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2021-03-26 23:30:32 +03:00
|
|
|
if (output_to_clipboard) {
|
|
|
|
GUI::Clipboard::the().set_bitmap(*bitmap);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-12-05 22:34:27 +03:00
|
|
|
auto encoded_bitmap_or_error = Gfx::PNGWriter::encode(*bitmap);
|
|
|
|
if (encoded_bitmap_or_error.is_error()) {
|
2021-03-26 21:55:40 +03:00
|
|
|
warnln("Failed to encode PNG");
|
|
|
|
return 1;
|
|
|
|
}
|
2022-12-05 22:34:27 +03:00
|
|
|
auto encoded_bitmap = encoded_bitmap_or_error.release_value();
|
|
|
|
|
2022-10-07 22:56:37 +03:00
|
|
|
if (edit_image)
|
2022-12-06 04:12:49 +03:00
|
|
|
output_path = Core::DateTime::now().to_deprecated_string("/tmp/screenshot-%Y-%m-%d-%H-%M-%S.png"sv);
|
2021-03-26 21:55:40 +03:00
|
|
|
|
2023-02-09 05:02:46 +03:00
|
|
|
auto file_or_error = Core::File::open(output_path, Core::File::OpenMode::ReadWrite);
|
2021-03-26 21:55:40 +03:00
|
|
|
if (file_or_error.is_error()) {
|
|
|
|
warnln("Could not open '{}' for writing: {}", output_path, file_or_error.error());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& file = *file_or_error.value();
|
2023-03-01 19:24:50 +03:00
|
|
|
TRY(file.write_until_depleted(encoded_bitmap.bytes()));
|
2021-03-26 21:55:40 +03:00
|
|
|
|
2022-10-07 22:56:37 +03:00
|
|
|
if (edit_image)
|
|
|
|
TRY(Core::Process::spawn("/bin/PixelPaint"sv, Array { output_path }));
|
|
|
|
|
2021-08-01 06:22:27 +03:00
|
|
|
bool printed_hyperlink = false;
|
|
|
|
if (isatty(STDOUT_FILENO)) {
|
2023-05-20 11:14:22 +03:00
|
|
|
auto full_path_or_error = FileSystem::real_path(output_path);
|
|
|
|
if (!full_path_or_error.is_error()) {
|
2021-08-01 06:22:27 +03:00
|
|
|
char hostname[HOST_NAME_MAX];
|
|
|
|
VERIFY(gethostname(hostname, sizeof(hostname)) == 0);
|
|
|
|
|
2023-05-20 11:14:22 +03:00
|
|
|
auto url = URL::create_with_file_scheme(full_path_or_error.value().to_deprecated_string(), {}, hostname);
|
2021-08-01 06:22:27 +03:00
|
|
|
out("\033]8;;{}\033\\", url.serialize());
|
|
|
|
printed_hyperlink = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out("{}", output_path);
|
|
|
|
|
|
|
|
if (printed_hyperlink) {
|
|
|
|
out("\033]8;;\033\\");
|
|
|
|
}
|
|
|
|
|
|
|
|
outln("");
|
2021-03-26 21:55:40 +03:00
|
|
|
return 0;
|
|
|
|
}
|