fltk/fluid/shell_command.cxx
Matthias Melcher fd791a068e
Separate FLUID user documentation, screen shot automation (#936)
* CMake integration, no autotiools
* alignment panel is now correctly renamed to setting panel
* source view is now correctly renamed to code view
* Merge FLTK FLUID docs into FLUID user manual.
* Add two simple entry tutorials
* Remove FLUID chapter form FLTK docs.
* GitHub action to generate HTML and PDF docs and
  make the available as artefacts
2024-04-17 17:51:32 +02:00

1000 lines
30 KiB
C++

//
// FLUID main entry for the Fast Light Tool Kit (FLTK).
//
// Copyright 1998-2023 by Bill Spitzak and others.
//
// This library is free software. Distribution and use rights are outlined in
// the file "COPYING" which should have been included with this file. If this
// file is missing or damaged, see the license at:
//
// https://www.fltk.org/COPYING.php
//
// Please see the following page on how to report bugs and issues:
//
// https://www.fltk.org/bugs.php
//
// in progress:
// FLUID comes with example shell commands to build the current project file
// and run the project. This is accomplished by calling `fltk-config` on the
// files generated by FLUID, and by calling the executable directly.
//
// If the user wants more complex commands, he can add or modify them in the
// "Shell" settings panel. Modified shell commands are saved with the .fl
// file.
// The Shell panel has a list of shell commands in the upper half. Under the
// list are buttons to add, duplicate, and delete shell commands. A popup
// menu offers import and export functionality and a list of sample scripts.
// We may want to add up and down buttons, so the user can change the
// order of commands.
// Selecting any shell command in the list fills in and activates a list of
// options in the lower half of the panel. Those settings are:
// - Name: the name of the shell command in the list
// - Label: the label in the pulldown menu (could be the same as name?)
// - Shortcut: shortcut key to launch the command
// - Storage: where to store this shell command
// - Condition: pulldown menu to make the entry conditional for various
// target platforms, for example, a "Windows only" entry would only be added
// to the Shell menu on a Windows machine. Other options could be:
// - Linux only, macOS only, never (to make a list header!?), inactive?
// - Command: a multiline input for the actual shell command
// - Variables: a pulldown menu that insert variable names like $<sourcefile>
// - options to save project, code, and strings before running
// - test-run button
// TODO: add @APPDIR@?
// TODO: get a macro to find `fltk-config` @FLTK_CONFIG@
// TODO: add an input field so the user can insert their preferred file and path for fltk-config (user setting)
// `fltk-config` is actually tricky to find
// for live builds, we could check the program launch directory
// if we know where build/Xcode/bin/Debug/fluid is, we
// may or may not find ./build/Xcode/fltk-config
// on macOS with homebrew, we find /opt/homebrew/bin/fltk-config but the user
// can set their own install path.
// We can query the shell path, but that requires knowing the users shell (echo $SHELL).
// We can run the shell as a login shell with `-l`, so the user $PTH is set: /bin/bash -l -c 'fltk-config'
// The shell should output the path of the fltk-config that it found and why it is using that one.
// This can also output the fltk-config version.
// TODO: add a bunch of sensible sample shell commands
// TODO: when this new feature is used for the very first time, import two or three samples as initial user setting
// TODO: make the settings dialog resizable
// TODO: make g_shell_config static, not a pointer, but don't load anything in batch mode
// FEATURE: Fd_Tool_Store icons are currently redundant with @file and @save and could be improved
// FEATURE: hostname, username, getenv support?
// FEATURE: ad the files ./fluid.prefs and ./fluid.user.prefs as tool locations
/*
Some ideas:
default shell is in $SHELL on linux and macOS
On macOS, we can write Apple Scripts:
#!/usr/bin/env osascript
say "@BASENAME@"
osascript <<EOD
say "spark"
EOD
osascript <<EOD
tell application "Xcode"
build workspace document 1
end tell
EOD
powershell -c "$wshell = New-Object -ComObject wscript.shell; $wshell.SendKeys('^{ESCAPE}')
*/
#include "shell_command.h"
#include "fluid.h"
#include "file.h"
#include "settings_panel.h"
#include <FL/Fl_Double_Window.H>
#include <FL/Fl_Menu_Bar.H>
#include <FL/fl_message.H>
#include <FL/fl_string_functions.h>
#include <errno.h>
static Fl_String fltk_config_cmd;
static Fl_Process s_proc;
/**
See if shell command is running (public)
*/
bool shell_command_running() {
return s_proc.desc() ? true : false;
}
/**
Reads an entry from the group. A default value must be
supplied. The return value indicates if the value was available
(non-zero) or the default was used (0).
\param[in] prefs preference group
\param[in] key name of entry
\param[out] value returned from preferences or default value if none was set
\param[in] defaultValue default value to be used if no preference was set
\return 0 if the default value was used
*/
char preferences_get(Fl_Preferences &prefs, const char *key, Fl_String &value, const Fl_String &defaultValue) {
char *v = NULL;
char ret = prefs.get(key, v, defaultValue.c_str());
value = v;
::free(v);
return ret;
}
/**
Sets an entry (name/value pair). The return value indicates if there
was a problem storing the data in memory. However it does not
reflect if the value was actually stored in the preference file.
\param[in] prefs preference group
\param[in] entry name of entry
\param[in] value set this entry to value (stops at the first nul character).
\return 0 if setting the value failed
*/
char preferences_set(Fl_Preferences &prefs, const char *key, const Fl_String &value) {
return prefs.set(key, value.c_str());
}
/** \class Fl_Process
Launch an external shell command.
*/
/**
Create a process manager
*/
Fl_Process::Fl_Process() {
_fpt= NULL;
}
/**
Destroy the project manager.
*/
Fl_Process::~Fl_Process() {
// TODO: check what we need to do if a task is still running
if (_fpt) close();
}
/**
Open a process.
\param[in] cmd the shell command that we want to run
\param[in] mode "r" or "w" for creating a stream that can read or write
\return a stream that is redirected from the shell command stdout
*/
FILE * Fl_Process::popen(const char *cmd, const char *mode) {
#if defined(_WIN32) && !defined(__CYGWIN__)
// PRECONDITIONS
if (!mode || !*mode || (*mode!='r' && *mode!='w') ) return NULL;
if (_fpt) close(); // close first before reuse
ptmode = *mode;
pin[0] = pin[1] = pout[0] = pout[1] = perr[0] = perr[1] = INVALID_HANDLE_VALUE;
// stderr to stdout wanted ?
int fusion = (strstr(cmd,"2>&1") !=NULL);
// Create windows pipes
if (!createPipe(pin) || !createPipe(pout) || (!fusion && !createPipe(perr) ) )
return freeHandles(); // error
// Initialize Startup Info
ZeroMemory(&si, sizeof(STARTUPINFO));
si.cb = sizeof(STARTUPINFO);
si.dwFlags = STARTF_USESTDHANDLES;
si.hStdInput = pin[0];
si.hStdOutput = pout[1];
si.hStdError = fusion ? pout[1] : perr [1];
if ( CreateProcess(NULL, (LPTSTR) cmd,NULL,NULL,TRUE,
DETACHED_PROCESS,NULL,NULL, &si, &pi)) {
// don't need theses handles inherited by child process:
clean_close(pin[0]); clean_close(pout[1]); clean_close(perr[1]);
HANDLE & h = *mode == 'r' ? pout[0] : pin[1];
_fpt = _fdopen(_open_osfhandle((fl_intptr_t) h,_O_BINARY),mode);
h= INVALID_HANDLE_VALUE; // reset the handle pointer that is shared
// with _fpt so we don't free it twice
}
if (!_fpt) freeHandles();
return _fpt;
#else
_fpt=::popen(cmd,mode);
return _fpt;
#endif
}
/**
Close the current process.
*/
int Fl_Process::close() {
#if defined(_WIN32) && !defined(__CYGWIN__)
if (_fpt) {
fclose(_fpt);
clean_close(perr[0]);
clean_close(pin[1]);
clean_close(pout[0]);
_fpt = NULL;
return 0;
}
return -1;
#else
int ret = ::pclose(_fpt);
_fpt=NULL;
return ret;
#endif
}
/**
non-null if file is open.
\return the current file descriptor of the process' stdout
*/
FILE *Fl_Process::desc() const {
return _fpt;
}
/**
Receive a single line from the current process.
\param[out] line buffer to receive the line
\param[in] s size of the provided buffer
\return NULL if an error occurred, otherwise a pointer to the string
*/
char *Fl_Process::get_line(char * line, size_t s) const {
return _fpt ? fgets(line, (int)s, _fpt) : NULL;
}
// returns fileno(FILE*):
// (file must be open, i.e. _fpt must be non-null)
// *FIXME* we should find a better solution for the 'fileno' issue
// non null if file is open
int Fl_Process::get_fileno() const {
#ifdef _MSC_VER
return _fileno(_fpt); // suppress MSVC warning
#else
return fileno(_fpt);
#endif
}
#if defined(_WIN32) && !defined(__CYGWIN__)
bool Fl_Process::createPipe(HANDLE * h, BOOL bInheritHnd) {
SECURITY_ATTRIBUTES sa;
sa.nLength = sizeof(sa);
sa.lpSecurityDescriptor = NULL;
sa.bInheritHandle = bInheritHnd;
return CreatePipe (&h[0],&h[1],&sa,0) ? true : false;
}
FILE *Fl_Process::freeHandles() {
clean_close(pin[0]); clean_close(pin[1]);
clean_close(pout[0]); clean_close(pout[1]);
clean_close(perr[0]); clean_close(perr[1]);
return NULL; // convenient for error management
}
void Fl_Process::clean_close(HANDLE& h) {
if (h!= INVALID_HANDLE_VALUE) CloseHandle(h);
h = INVALID_HANDLE_VALUE;
}
#endif
/**
Prepare FLUID for running a shell command according to the command flags.
\param[in] flags set various flags to save the project, code, and string before running the command
\return false if the previous command is still running
*/
static bool prepare_shell_command(int flags) {
// settings_window->hide();
if (s_proc.desc()) {
fl_alert("Previous shell command still running!");
return false;
}
if (flags & Fd_Shell_Command::SAVE_PROJECT) {
save_cb(0, 0);
}
if (flags & Fd_Shell_Command::SAVE_SOURCECODE) {
write_code_files(true);
}
if (flags & Fd_Shell_Command::SAVE_STRINGS) {
write_strings_cb(0, 0);
}
return true;
}
/**
Called by the file handler when the command is finished.
*/
void shell_proc_done() {
shell_run_terminal->append("... END SHELL COMMAND ...\n");
shell_run_button->activate();
shell_run_window->label("FLUID Shell");
fl_beep();
}
void shell_timer_cb(void*) {
if (!s_proc.desc()) {
shell_proc_done();
} else {
Fl::add_timeout(0.25, shell_timer_cb);
}
}
// Support the full piped shell command...
void shell_pipe_cb(FL_SOCKET, void*) {
char line[1024]=""; // Line from command output...
if (s_proc.get_line(line, sizeof(line)) != NULL) {
// Add the line to the output list...
shell_run_terminal->append(line);
} else {
// End of file; tell the parent...
Fl::remove_timeout(shell_timer_cb);
Fl::remove_fd(s_proc.get_fileno());
s_proc.close();
shell_proc_done();
}
}
/** Find the script `fltk-config` that most closely relates to this version of FLUID.
This is not implemented yet.
*/
//static void find_fltk_config() {
//
//}
static void expand_macro(Fl_String &cmd, const Fl_String &macro, const Fl_String &content) {
for (int i=0;;) {
i = cmd.find(macro, i);
if (i==Fl_String::npos) break;
cmd.replace(i, macro.size(), content);
}
}
static void expand_macros(Fl_String &cmd) {
expand_macro(cmd, "@BASENAME@", g_project.basename());
expand_macro(cmd, "@PROJECTFILE_PATH@", g_project.projectfile_path());
expand_macro(cmd, "@PROJECTFILE_NAME@", g_project.projectfile_name());
expand_macro(cmd, "@CODEFILE_PATH@", g_project.codefile_path());
expand_macro(cmd, "@CODEFILE_NAME@", g_project.codefile_name());
expand_macro(cmd, "@HEADERFILE_PATH@", g_project.headerfile_path());
expand_macro(cmd, "@HEADERFILE_NAME@", g_project.headerfile_name());
expand_macro(cmd, "@TEXTFILE_PATH@", g_project.stringsfile_path());
expand_macro(cmd, "@TEXTFILE_NAME@", g_project.stringsfile_name());
// TODO: implement finding the script `fltk-config` for all platforms
// if (cmd.find("@FLTK_CONFIG@") != Fl_String::npos) {
// find_fltk_config();
// expand_macro(cmd, "@FLTK_CONFIG@", fltk_config_cmd.c_str());
// }
if (cmd.find("@TMPDIR@") != Fl_String::npos)
expand_macro(cmd, "@TMPDIR@", get_tmpdir());
}
/**
Show the terminal window where it was last positioned.
*/
void show_terminal_window() {
Fl_Preferences pos(fluid_prefs, "shell_run_Window_pos");
int x, y, w, h;
pos.get("x", x, -1);
pos.get("y", y, 0);
pos.get("w", w, 640);
pos.get("h", h, 480);
if (x!=-1) {
shell_run_window->resize(x, y, w, h);
}
shell_run_window->show();
}
/**
Prepare for and run a shell command.
\param[in] cmd the command that is sent to `/bin/sh -c ...` or `cmd.exe` on Windows machines
\param[in] flags various flags in preparation of the command
*/
void run_shell_command(const Fl_String &cmd, int flags) {
if (cmd.empty()) {
fl_alert("No shell command entered!");
return;
}
if (!prepare_shell_command(flags)) return;
Fl_String expanded_cmd = cmd;
expand_macros(expanded_cmd);
if ( ((flags & Fd_Shell_Command::DONT_SHOW_TERMINAL) == 0)
&& (!shell_run_window->visible()))
{
show_terminal_window();
}
// Show the output window and clear things...
if (flags & Fd_Shell_Command::CLEAR_TERMINAL)
shell_run_terminal->printf("\033[2J\033[H");
if (flags & Fd_Shell_Command::CLEAR_HISTORY)
shell_run_terminal->printf("\033[3J");
shell_run_terminal->scrollbar->value(0);
shell_run_terminal->printf("\033[0;32m%s\033[0m\n", expanded_cmd.c_str());
shell_run_window->label(expanded_cmd.c_str());
if (s_proc.popen((char *)expanded_cmd.c_str()) == NULL) {
shell_run_terminal->printf("\033[1;31mUnable to run shell command: %s\033[0m\n",
strerror(errno));
shell_run_window->label("FLUID Shell");
return;
}
shell_run_button->deactivate();
// if the function below does not for some reason, we will check periodically
// to see if the command is done
Fl::add_timeout(0.25, shell_timer_cb);
// this will tell us when the shell command is done
Fl::add_fd(s_proc.get_fileno(), shell_pipe_cb);
}
/**
Create an empty shell command structure.
*/
Fd_Shell_Command::Fd_Shell_Command()
: shortcut(0),
storage(FD_STORE_USER),
condition(0),
flags(0),
shell_menu_item_(NULL)
{
}
/**
Copy the aspects of a shell command dataset into a new shell command.
\param[in] rhs copy from this prototype
*/
Fd_Shell_Command::Fd_Shell_Command(const Fd_Shell_Command *rhs)
: name(rhs->name),
label(rhs->label),
shortcut(rhs->shortcut),
storage(rhs->storage),
condition(rhs->condition),
condition_data(rhs->condition_data),
command(rhs->command),
flags(rhs->flags),
shell_menu_item_(NULL)
{
}
/**
Create a default storage for a shell command and how it is accessible in FLUID.
\param[in] name is used as a stand-in for the command name and label
*/
Fd_Shell_Command::Fd_Shell_Command(const Fl_String &in_name)
: name(in_name),
label(in_name),
shortcut(0),
storage(FD_STORE_USER),
condition(Fd_Shell_Command::ALWAYS),
command("echo \"Hello, FLUID!\""),
flags(Fd_Shell_Command::SAVE_PROJECT|Fd_Shell_Command::SAVE_SOURCECODE),
shell_menu_item_(NULL)
{
}
/**
Create a storage for a shell command and how it is accessible in FLUID.
\param[in] in_name name of this command in the command list in the settings panel
\param[in] in_label label text in the main pulldown menu
\param[in] in_shortcut a keyboard shortcut that will also appear in the main menu
\param[in] in_storage storage location for this command
\param[in] in_condition commands can be hidden for certain platforms by setting a condition
\param[in] in_condition_data more details for future conditions, i.e. per user, per host, etc.
\param[in] in_command the shell command that we want to run
\param[in] in_flags some flags to tell FLUID to save the project, code, or strings before running the command
*/
Fd_Shell_Command::Fd_Shell_Command(const Fl_String &in_name,
const Fl_String &in_label,
Fl_Shortcut in_shortcut,
Fd_Tool_Store in_storage,
int in_condition,
const Fl_String &in_condition_data,
const Fl_String &in_command,
int in_flags)
: name(in_name),
label(in_label),
shortcut(in_shortcut),
storage(in_storage),
condition(in_condition),
condition_data(in_condition_data),
command(in_command),
flags(in_flags),
shell_menu_item_(NULL)
{
}
/**
Run this command now.
Will open the Shell Panel and execute the command if no other command is
currently running.
*/
void Fd_Shell_Command::run() {
if (!command.empty())
run_shell_command(command, flags);
}
/**
Update the shell submenu in main menu with the shortcut and a copy of the label.
*/
void Fd_Shell_Command::update_shell_menu() {
if (shell_menu_item_) {
const char *old_label = shell_menu_item_->label(); // can be NULL
const char *new_label = label.c_str(); // never NULL
if (!old_label || (old_label && strcmp(old_label, new_label))) {
if (old_label) ::free((void*)old_label);
shell_menu_item_->label(fl_strdup(new_label));
}
shell_menu_item_->shortcut(shortcut);
}
}
/**
Check if the set condition is met.
\return true if this command appears in the main menu
*/
bool Fd_Shell_Command::is_active() {
switch (condition) {
case ALWAYS: return true;
case NEVER: return false;
#ifdef _WIN32
case MAC_ONLY: return false;
case UX_ONLY: return false;
case WIN_ONLY: return true;
case MAC_AND_UX_ONLY: return false;
#elif defined(__APPLE__)
case MAC_ONLY: return true;
case UX_ONLY: return false;
case WIN_ONLY: return false;
case MAC_AND_UX_ONLY: return true;
#else
case MAC_ONLY: return false;
case UX_ONLY: return true;
case WIN_ONLY: return false;
case MAC_AND_UX_ONLY: return true;
#endif
case USER_ONLY: return false; // TODO: get user name
case HOST_ONLY: return false; // TODO: get host name
case ENV_ONLY: {
const char *value = fl_getenv(condition_data.c_str());
if (value && *value) return true;
return false;
}
}
return false;
}
void Fd_Shell_Command::read(Fl_Preferences &prefs) {
int tmp;
preferences_get(prefs, "name", name, "<unnamed>");
preferences_get(prefs, "label", label, "<no label>");
prefs.get("shortcut", tmp, 0);
shortcut = (Fl_Shortcut)tmp;
prefs.get("storage", tmp, -1);
if (tmp != -1) storage = (Fd_Tool_Store)tmp;
prefs.get("condition", condition, ALWAYS);
preferences_get(prefs, "condition_data", condition_data, "");
preferences_get(prefs, "command", command, "");
prefs.get("flags", flags, 0);
}
void Fd_Shell_Command::write(Fl_Preferences &prefs, bool save_location) {
preferences_set(prefs, "name", name);
preferences_set(prefs, "label", label);
if (shortcut != 0) prefs.set("shortcut", (int)shortcut);
if (save_location) prefs.set("storage", (int)storage);
if (condition != ALWAYS) prefs.set("condition", condition);
if (!condition_data.empty()) preferences_set(prefs, "condition_data", condition_data);
if (!command.empty()) preferences_set(prefs, "command", command);
if (flags != 0) prefs.set("flags", flags);
}
void Fd_Shell_Command::read(class Fd_Project_Reader *in) {
const char *c = in->read_word(1);
if (strcmp(c, "{")!=0) return; // expecting start of group
storage = FD_STORE_PROJECT;
for (;;) {
c = in->read_word(1);
if (strcmp(c, "}")==0) break; // end of command list
else if (strcmp(c, "name")==0)
name = in->read_word();
else if (strcmp(c, "label")==0)
label = in->read_word();
else if (strcmp(c, "shortcut")==0)
shortcut = in->read_int();
else if (strcmp(c, "condition")==0)
condition = in->read_int();
else if (strcmp(c, "condition_data")==0)
condition_data = in->read_word();
else if (strcmp(c, "command")==0)
command = in->read_word();
else if (strcmp(c, "flags")==0)
flags = in->read_int();
else
in->read_word(); // skip an unknown word
}
}
void Fd_Shell_Command::write(class Fd_Project_Writer *out) {
out->write_string("\n command {");
out->write_string("\n name "); out->write_word(name.c_str());
out->write_string("\n label "); out->write_word(label.c_str());
if (shortcut) out->write_string("\n shortcut %d", shortcut);
if (condition) out->write_string("\n condition %d", condition);
if (!condition_data.empty()) {
out->write_string("\n condition_data "); out->write_word(condition_data.c_str());
}
if (!command.empty()) {
out->write_string("\n command "); out->write_word(command.c_str());
}
if (flags) out->write_string("\n flags %d", flags);
out->write_string("\n }");
}
/**
Manage a list of shell commands and their parameters.
*/
Fd_Shell_Command_List::Fd_Shell_Command_List()
: list(NULL),
list_size(0),
list_capacity(0),
shell_menu_(NULL)
{
}
/**
Release all shell commands and destroy this class.
*/
Fd_Shell_Command_List::~Fd_Shell_Command_List() {
clear();
}
/**
Return the shell command at the given index.
\param[in] index must be between 0 and list_size-1
\return a pointer to the shell command data
*/
Fd_Shell_Command *Fd_Shell_Command_List::at(int index) const {
return list[index];
}
/**
Clear all shell commands.
*/
void Fd_Shell_Command_List::clear() {
if (list) {
for (int i=0; i<list_size; i++) {
delete list[i];
}
::free(list);
list_size = 0;
list_capacity = 0;
list = 0;
}
}
/**
remove all shell commands of the given storage location from the list.
*/
void Fd_Shell_Command_List::clear(Fd_Tool_Store storage) {
for (int i=list_size-1; i>=0; i--) {
if (list[i]->storage == storage) {
remove(i);
}
}
}
/**
Read shell configuration from a preferences group.
*/
void Fd_Shell_Command_List::read(Fl_Preferences &prefs, Fd_Tool_Store storage) {
// import the old shell commands from previous user settings
if (&fluid_prefs == &prefs) {
int version;
prefs.get("shell_commands_version", version, 0);
if (version == 0) {
int save_fl, save_code, save_strings;
Fd_Shell_Command *cmd = new Fd_Shell_Command();
cmd->storage = FD_STORE_USER;
cmd->name = "Sample Shell Command";
cmd->label = "Sample Shell Command";
cmd->shortcut = FL_ALT+'g';
preferences_get(fluid_prefs, "shell_command", cmd->command, "echo \"Sample Shell Command\"");
fluid_prefs.get("shell_savefl", save_fl, 1);
fluid_prefs.get("shell_writecode", save_code, 1);
fluid_prefs.get("shell_writemsgs", save_strings, 0);
if (save_fl) cmd->flags |= Fd_Shell_Command::SAVE_PROJECT;
if (save_code) cmd->flags |= Fd_Shell_Command::SAVE_SOURCECODE;
if (save_strings) cmd->flags |= Fd_Shell_Command::SAVE_STRINGS;
add(cmd);
}
version = 1;
prefs.set("shell_commands_version", version);
}
Fl_Preferences shell_commands(prefs, "shell_commands");
int n = shell_commands.groups();
for (int i=0; i<n; i++) {
Fl_Preferences cmd_prefs(shell_commands, Fl_Preferences::Name(i));
Fd_Shell_Command *cmd = new Fd_Shell_Command();
cmd->storage = FD_STORE_USER;
cmd->read(cmd_prefs);
add(cmd);
}
}
/**
Write shell configuration to a preferences group.
*/
void Fd_Shell_Command_List::write(Fl_Preferences &prefs, Fd_Tool_Store storage) {
Fl_Preferences shell_commands(prefs, "shell_commands");
shell_commands.delete_all_groups();
int index = 0;
for (int i=0; i<list_size; i++) {
if (list[i]->storage == FD_STORE_USER) {
Fl_Preferences cmd(shell_commands, Fl_Preferences::Name(index++));
list[i]->write(cmd);
}
}
}
/**
Read shell configuration from a project file.
*/
void Fd_Shell_Command_List::read(Fd_Project_Reader *in) {
const char *c = in->read_word(1);
if (strcmp(c, "{")!=0) return; // expecting start of group
clear(FD_STORE_PROJECT);
for (;;) {
c = in->read_word(1);
if (strcmp(c, "}")==0) break; // end of command list
else if (strcmp(c, "command")==0) {
Fd_Shell_Command *cmd = new Fd_Shell_Command();
add(cmd);
cmd->read(in);
} else {
in->read_word(); // skip an unknown group
}
}
}
/**
Write shell configuration to a project file.
*/
void Fd_Shell_Command_List::write(Fd_Project_Writer *out) {
int n_in_project_file = 0;
for (int i=0; i<list_size; i++) {
if (list[i]->storage == FD_STORE_PROJECT)
n_in_project_file++;
}
if (n_in_project_file > 0) {
out->write_string("\nshell_commands {");
for (int i=0; i<list_size; i++) {
if (list[i]->storage == FD_STORE_PROJECT)
list[i]->write(out);
}
out->write_string("\n}");
}
}
/**
Add a previously created shell command to the end of the list.
\param[in] cmd a pointer to the command that we want to add
*/
void Fd_Shell_Command_List::add(Fd_Shell_Command *cmd) {
if (list_size == list_capacity) {
list_capacity += 16;
list = (Fd_Shell_Command**)::realloc(list, list_capacity * sizeof(Fd_Shell_Command*));
}
list[list_size++] = cmd;
}
/**
Insert a newly created shell command at the given position in the list.
\param[in] index must be between 0 and list_size-1
\param[in] cmd a pointer to the command that we want to add
*/
void Fd_Shell_Command_List::insert(int index, Fd_Shell_Command *cmd) {
if (list_size == list_capacity) {
list_capacity += 16;
list = (Fd_Shell_Command**)::realloc(list, list_capacity * sizeof(Fd_Shell_Command*));
}
::memmove(list+index+1, list+index, (list_size-index)*sizeof(Fd_Shell_Command**));
list_size++;
list[index] = cmd;
}
/**
Remove and delete the command at the given index.
\param[in] index must be between 0 and list_size-1
*/
void Fd_Shell_Command_List::remove(int index) {
delete list[index];
list_size--;
::memmove(list+index, list+index+1, (list_size-index)*sizeof(Fd_Shell_Command**));
}
/**
This is called whenever the user clicks a shell command menu in the main menu.
\param[in] u cast tp long to get the index of the shell command
*/
void menu_shell_cmd_cb(Fl_Widget*, void *u) {
long index = (long)(fl_intptr_t)u;
g_shell_config->list[index]->run();
}
/**
This is called when the user selects the menu to edit the shell commands.
It pops up the setting panel at the shell settings tab.
*/
void menu_shell_customize_cb(Fl_Widget*, void*) {
settings_window->show();
w_settings_tabs->value(w_settings_shell_tab);
}
/**
Rebuild the entire shell submenu from scratch and replace the old menu.
*/
void Fd_Shell_Command_List::rebuild_shell_menu() {
static Fl_Menu_Item *shell_submenu = NULL;
if (!shell_submenu)
shell_submenu = (Fl_Menu_Item*)main_menubar->find_item(menu_marker);
int i, j, num_active_items = 0;
// count the active commands
for (i=0; i<list_size; i++) {
if (list[i]->is_active()) num_active_items++;
}
// allocate a menu item array
Fl_Menu_Item *mi = (Fl_Menu_Item*)::calloc(num_active_items+2, sizeof(Fl_Menu_Item));
// set the menu item pointer for all active commands
for (i=j=0; i<list_size; i++) {
Fd_Shell_Command *cmd = list[i];
if (cmd->is_active()) {
cmd->shell_menu_item_ = mi + j;
mi[j].callback(menu_shell_cmd_cb);
mi[j].argument(i);
cmd->update_shell_menu();
j++;
}
}
if (j>0) mi[j-1].flags |= FL_MENU_DIVIDER;
mi[j].label(fl_strdup("Customize..."));
mi[j].shortcut(FL_ALT+'x');
mi[j].callback(menu_shell_customize_cb);
// replace the old menu array with the new one
Fl_Menu_Item *mi_old = shell_menu_;
shell_menu_ = mi;
shell_submenu->user_data(shell_menu_);
// free all resources from the old menu
if (mi_old && (mi_old != default_menu)) {
for (i=0; ; i++) {
const char *label = mi_old[i].label();
if (!label) break;
::free((void*)label);
}
::free(mi_old);
}
}
/**
Tell the settings dialog to query this list and update its GUI elements.
*/
void Fd_Shell_Command_List::update_settings_dialog() {
if (w_settings_shell_tab)
w_settings_shell_tab->do_callback(w_settings_shell_tab, LOAD);
}
/**
The default shell submenu in batch mode.
*/
Fl_Menu_Item Fd_Shell_Command_List::default_menu[] = {
{ "Customize...", FL_ALT+'x', menu_shell_customize_cb },
{ NULL }
};
/**
Used to find the shell submenu within the main menu tree.
*/
void Fd_Shell_Command_List::menu_marker(Fl_Widget*, void*) {
// intentionally left empty
}
/**
Export all selected shell commands to an external file.
Verify that g_shell_config and w_settings_shell_list are not NULL. Open a
file chooser and export all items that are selected in w_settings_shell_list
into an external file.
*/
void Fd_Shell_Command_List::export_selected() {
if (!g_shell_config || (g_shell_config->list_size == 0)) return;
if (!w_settings_shell_list) return;
Fl_Native_File_Chooser dialog;
dialog.title("Export selected shell commands:");
dialog.type(Fl_Native_File_Chooser::BROWSE_SAVE_FILE);
dialog.filter("FLUID Files\t*.flcmd\n");
dialog.directory(g_project.projectfile_path().c_str());
dialog.preset_file((g_project.basename() + ".flcmd").c_str());
if (dialog.show() != 0) return;
Fl_Preferences file(dialog.filename(), "flcmd.fluid.fltk.org", NULL, (Fl_Preferences::Root)(Fl_Preferences::C_LOCALE|Fl_Preferences::CLEAR));
Fl_Preferences shell_commands(file, "shell_commands");
int i, index = 0, n = w_settings_shell_list->size();
for (i = 0; i < n; i++) {
if (w_settings_shell_list->selected(i+1)) {
Fl_Preferences cmd(shell_commands, Fl_Preferences::Name(index++));
g_shell_config->list[i]->write(cmd, true);
}
}
}
/**
Import shell commands from an external file and add them to the list.
Verify that g_shell_config and w_settings_shell_list are not NULL. Open a
file chooser and import all items.
*/
void Fd_Shell_Command_List::import_from_file() {
if (!g_shell_config || (g_shell_config->list_size == 0)) return;
if (!w_settings_shell_list) return;
Fl_Native_File_Chooser dialog;
dialog.title("Import shell commands:");
dialog.type(Fl_Native_File_Chooser::BROWSE_FILE);
dialog.filter("FLUID Files\t*.flcmd\n");
dialog.directory(g_project.projectfile_path().c_str());
dialog.preset_file((g_project.basename() + ".flcmd").c_str());
if (dialog.show() != 0) return;
Fl_Preferences file(dialog.filename(), "flcmd.fluid.fltk.org", NULL, Fl_Preferences::C_LOCALE);
Fl_Preferences shell_commands(file, "shell_commands");
int i, n = shell_commands.groups();
for (i = 0; i < n; i++) {
Fl_Preferences cmd_prefs(shell_commands, Fl_Preferences::Name(i));
Fd_Shell_Command *cmd = new Fd_Shell_Command();
cmd->storage = FD_STORE_USER;
cmd->read(cmd_prefs);
g_shell_config->add(cmd);
}
w_settings_shell_list->do_callback(w_settings_shell_list, LOAD);
w_settings_shell_cmd->do_callback(w_settings_shell_cmd, LOAD);
w_settings_shell_toolbox->do_callback(w_settings_shell_toolbox, LOAD);
g_shell_config->rebuild_shell_menu();
}
/**
A pointer to the list of shell commands if we are not in batch mode.
*/
Fd_Shell_Command_List *g_shell_config = NULL;