/* * Copyright (c) 2020, the SerenityOS developers. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. 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. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT HOLDER 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 "HelpWindow.h" #include "Spreadsheet.h" #include "SpreadsheetWidget.h" #include #include #include #include #include #include #include #include #include #include #include #include int main(int argc, char* argv[]) { if (pledge("stdio shared_buffer accept rpath unix cpath wpath fattr thread", nullptr) < 0) { perror("pledge"); return 1; } auto app = GUI::Application::construct(argc, argv); if (pledge("stdio thread rpath accept cpath wpath shared_buffer unix", nullptr) < 0) { perror("pledge"); return 1; } const char* filename = nullptr; Core::ArgsParser args_parser; args_parser.add_positional_argument(filename, "File to read from", "file", Core::ArgsParser::Required::No); args_parser.parse(argc, argv); if (filename) { if (!Core::File::exists(filename) || Core::File::is_directory(filename)) { warnln("File does not exist or is a directory: {}", filename); return 1; } } if (unveil("/tmp/portal/webcontent", "rw") < 0) { perror("unveil"); return 1; } if (unveil("/etc", "r") < 0) { perror("unveil"); return 1; } if (unveil(Core::StandardPaths::home_directory().characters(), "rwc") < 0) { perror("unveil"); return 1; } if (unveil("/res", "r") < 0) { perror("unveil"); return 1; } if (unveil(nullptr, nullptr) < 0) { perror("unveil"); return 1; } auto app_icon = GUI::Icon::default_icon("app-spreadsheet"); auto window = GUI::Window::construct(); window->set_title("Spreadsheet"); window->resize(640, 480); window->set_icon(app_icon.bitmap_for_size(16)); auto& spreadsheet_widget = window->set_main_widget(NonnullRefPtrVector {}, filename == nullptr); if (filename) spreadsheet_widget.load(filename); auto menubar = GUI::MenuBar::construct(); auto& app_menu = menubar->add_menu("Spreadsheet"); app_menu.add_action(GUI::Action::create("Add New Sheet", Gfx::Bitmap::load_from_file("/res/icons/16x16/new-tab.png"), [&](auto&) { spreadsheet_widget.add_sheet(); })); app_menu.add_separator(); app_menu.add_action(GUI::CommonActions::make_quit_action([&](auto&) { if (!spreadsheet_widget.request_close()) return; app->quit(0); })); window->on_close_request = [&]() -> GUI::Window::CloseRequestDecision { if (spreadsheet_widget.request_close()) return GUI::Window::CloseRequestDecision::Close; return GUI::Window::CloseRequestDecision::StayOpen; }; auto& file_menu = menubar->add_menu("File"); file_menu.add_action(GUI::CommonActions::make_open_action([&](auto&) { Optional load_path = GUI::FilePicker::get_open_filepath(window); if (!load_path.has_value()) return; spreadsheet_widget.load(load_path.value()); })); file_menu.add_action(GUI::CommonActions::make_save_action([&](auto&) { if (spreadsheet_widget.current_filename().is_empty()) { String name = "workbook"; Optional save_path = GUI::FilePicker::get_save_filepath(window, name, "sheets"); if (!save_path.has_value()) return; spreadsheet_widget.save(save_path.value()); } else { spreadsheet_widget.save(spreadsheet_widget.current_filename()); } })); file_menu.add_action(GUI::CommonActions::make_save_as_action([&](auto&) { auto current_filename = spreadsheet_widget.current_filename(); String name = "workbook"; Optional save_path = GUI::FilePicker::get_save_filepath(window, name, "sheets"); if (!save_path.has_value()) return; spreadsheet_widget.save(save_path.value()); if (!current_filename.is_empty()) spreadsheet_widget.set_filename(current_filename); })); auto& edit_menu = menubar->add_menu("Edit"); edit_menu.add_action(GUI::CommonActions::make_copy_action([&](auto&) { /// text/x-spreadsheet-data: /// - currently selected cell /// - selected cell+ auto& cells = spreadsheet_widget.current_worksheet().selected_cells(); ASSERT(!cells.is_empty()); StringBuilder text_builder, url_builder; bool first = true; auto cursor = spreadsheet_widget.current_selection_cursor(); if (cursor) { Spreadsheet::Position position { spreadsheet_widget.current_worksheet().column(cursor->column()), (size_t)cursor->row() }; url_builder.append(position.to_url().to_string()); url_builder.append('\n'); } for (auto& cell : cells) { if (first && !cursor) { url_builder.append(cell.to_url().to_string()); url_builder.append('\n'); } url_builder.append(cell.to_url().to_string()); url_builder.append('\n'); auto cell_data = spreadsheet_widget.current_worksheet().at(cell); if (!first) text_builder.append('\t'); if (cell_data) text_builder.append(cell_data->data()); first = false; } HashMap metadata; metadata.set("text/x-spreadsheet-data", url_builder.to_string()); GUI::Clipboard::the().set_data(text_builder.string_view().bytes(), "text/plain", move(metadata)); }, window)); edit_menu.add_action(GUI::CommonActions::make_paste_action([&](auto&) { ScopeGuard update_after_paste { [&] { spreadsheet_widget.update(); } }; auto& cells = spreadsheet_widget.current_worksheet().selected_cells(); ASSERT(!cells.is_empty()); const auto& data = GUI::Clipboard::the().data_and_type(); if (auto spreadsheet_data = data.metadata.get("text/x-spreadsheet-data"); spreadsheet_data.has_value()) { Vector source_positions, target_positions; auto& sheet = spreadsheet_widget.current_worksheet(); for (auto& line : spreadsheet_data.value().split_view('\n')) { dbg() << "Paste line '" << line << "'"; auto position = sheet.position_from_url(line); if (position.has_value()) source_positions.append(position.release_value()); } for (auto& position : spreadsheet_widget.current_worksheet().selected_cells()) target_positions.append(position); if (source_positions.is_empty()) return; auto first_position = source_positions.take_first(); sheet.copy_cells(move(source_positions), move(target_positions), first_position); } else { for (auto& cell : spreadsheet_widget.current_worksheet().selected_cells()) spreadsheet_widget.current_worksheet().ensure(cell).set_data(StringView { data.data.data(), data.data.size() }); spreadsheet_widget.update(); } }, window)); auto& help_menu = menubar->add_menu("Help"); help_menu.add_action(GUI::Action::create( "Functions Help", [&](auto&) { auto docs = spreadsheet_widget.current_worksheet().gather_documentation(); auto help_window = Spreadsheet::HelpWindow::the(); help_window->set_docs(move(docs)); help_window->show(); }, window)); app_menu.add_separator(); help_menu.add_action(GUI::Action::create("About", [&](auto&) { GUI::AboutDialog::show("Spreadsheet", app_icon.bitmap_for_size(32), window); })); app->set_menubar(move(menubar)); window->show(); return app->exec(); }