// FbCommands.cc for Fluxbox // Copyright (c) 2003 - 2006 Henrik Kinnunen (fluxgen at fluxbox dot org) // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. // $Id$ #include "FbCommands.hh" #include "fluxbox.hh" #include "Screen.hh" #include "CommandDialog.hh" #include "FocusControl.hh" #include "Workspace.hh" #include "Window.hh" #include "Keys.hh" #include "MenuCreator.hh" #include "FbTk/Theme.hh" #include "FbTk/Menu.hh" #include <sys/types.h> #include <unistd.h> #include <fstream> #include <iostream> #include <set> #ifdef HAVE_CSTDLIB #include <cstdlib> #else #include <stdlib.h> #endif #ifdef HAVE_CSTRING #include <cstring> #else #include <string.h> #endif #ifdef HAVE_CONFIG_H #include "config.h" #endif // HAVE_CONFIG_H #if defined(__EMX__) && defined(HAVE_PROCESS_H) #include <process.h> // for P_NOWAIT #endif // __EMX__ using std::string; using std::pair; using std::set; using std::ofstream; using std::endl; using std::ios; namespace { void showMenu(const BScreen &screen, FbTk::Menu &menu) { // special case for root menu if (&menu == &screen.rootMenu()) { Fluxbox* fb = Fluxbox::instance(); if(fb->menuTimestampsChanged()) { // we dont show the menu here because fluxbox // will bring up the rootmenu after the timed // reread of the menu fb->rereadMenu(true); return; } } Window root_ret; // not used Window window_ret; // not used int rx = 0, ry = 0; int wx, wy; // not used unsigned int mask; // not used XQueryPointer(menu.fbwindow().display(), screen.rootWindow().window(), &root_ret, &window_ret, &rx, &ry, &wx, &wy, &mask); int borderw = menu.fbwindow().borderWidth(); int head = screen.getHead(rx, ry); menu.updateMenu(); pair<int, int> m = screen.clampToHead(head, rx - menu.width() / 2, ry - menu.titleWindow().height() / 2, menu.width() + 2*borderw, menu.height() + 2*borderw); menu.move(m.first, m.second); menu.setScreen(screen.getHeadX(head), screen.getHeadY(head), screen.getHeadWidth(head), screen.getHeadHeight(head)); menu.show(); menu.grabInputFocus(); } } namespace FbCommands { ExecuteCmd::ExecuteCmd(const string &cmd, int screen_num):m_cmd(cmd), m_screen_num(screen_num) { } void ExecuteCmd::execute() { #ifndef __EMX__ run(); #else // __EMX__ spawnlp(P_NOWAIT, "cmd.exe", "cmd.exe", "/c", m_cmd.c_str(), static_cast<void*>(NULL)); #endif // !__EMX__ } int ExecuteCmd::run() { pid_t pid = fork(); if (pid) return pid; string displaystring("DISPLAY="); displaystring += DisplayString(FbTk::App::instance()->display()); char intbuff[64]; int screen_num = m_screen_num; if (screen_num < 0) { if (Fluxbox::instance()->mouseScreen() == 0) screen_num = 0; else screen_num = Fluxbox::instance()->mouseScreen()->screenNumber(); } sprintf(intbuff, "%d", screen_num); // get shell path from the environment // this process exits immediately, so we don't have to worry about memleaks const char *shell = getenv("SHELL"); if (!shell) shell = "/bin/sh"; // remove last number of display and add screen num displaystring.erase(displaystring.size()-1); displaystring += intbuff; setsid(); putenv(const_cast<char *>(displaystring.c_str())); execl(shell, shell, "-c", m_cmd.c_str(), static_cast<void*>(NULL)); exit(0); return pid; // compiler happy -> we are happy ;) } SetModKeyCmd::SetModKeyCmd(const string& modkey) : m_modkey(modkey) { } void SetModKeyCmd::execute() { Fluxbox::instance()->setModKey(m_modkey.c_str()); Fluxbox::instance()->save_rc(); // TODO: we need a better way to do this ... Fluxbox::instance()->reconfigure(); } ExportCmd::ExportCmd(const string& name, const string& value) : m_name(name), m_value(value) { } void ExportCmd::execute() { // the setenv()-routine is not everywhere available and // putenv() doesnt manage the strings in the environment // and hence we have to do that on our own to avoid memleaking static set<char*> stored; char* newenv = new char[m_name.size() + m_value.size() + 2]; if (newenv) { char* oldenv = getenv(m_name.c_str()); // oldenv points to the value .. we have to go back a bit if (oldenv && stored.find(oldenv - (m_name.size() + 1)) != stored.end()) oldenv -= (m_name.size() + 1); else oldenv = NULL; memset(newenv, 0, m_name.size() + m_value.size() + 2); strcat(newenv, m_name.c_str()); strcat(newenv, "="); strcat(newenv, m_value.c_str()); if (putenv(newenv) == 0) { if (oldenv) { stored.erase(oldenv); delete[] oldenv; } stored.insert(newenv); } } } void ExitFluxboxCmd::execute() { Fluxbox::instance()->shutdown(); } void SaveResources::execute() { Fluxbox::instance()->save_rc(); } RestartFluxboxCmd::RestartFluxboxCmd(const string &cmd):m_cmd(cmd){ } void RestartFluxboxCmd::execute() { if (m_cmd.empty()) Fluxbox::instance()->restart(); else Fluxbox::instance()->restart(m_cmd.c_str()); } void ReconfigureFluxboxCmd::execute() { Fluxbox::instance()->reconfigure(); } void ReloadStyleCmd::execute() { SetStyleCmd cmd(Fluxbox::instance()->getStyleFilename()); cmd.execute(); } SetStyleCmd::SetStyleCmd(const string &filename):m_filename(filename) { } void SetStyleCmd::execute() { Fluxbox::instance()->saveStyleFilename(m_filename.c_str()); Fluxbox::instance()->save_rc(); FbTk::ThemeManager::instance().load(m_filename, Fluxbox::instance()->getStyleOverlayFilename()); } KeyModeCmd::KeyModeCmd(const string &arguments):m_keymode(arguments),m_end_args("None Escape") { string::size_type second_pos = m_keymode.find_first_of(" \t", 0); if (second_pos != string::npos) { // ok we have arguments, parsing them here m_end_args = m_keymode.substr(second_pos); m_keymode.erase(second_pos); // remove argument from command } if (m_keymode != "default") Fluxbox::instance()->keys()->addBinding(m_keymode + ": " + m_end_args + " :keymode default"); } void KeyModeCmd::execute() { Fluxbox::instance()->keys()->keyMode(m_keymode); } void HideMenuCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); screen->hideMenus(); if (screen->rootMenu().isVisible()) screen->rootMenu().hide(); if (screen->workspaceMenu().isVisible()) screen->workspaceMenu().hide(); if (FbTk::Menu::shownMenu()) FbTk::Menu::shownMenu()->hide(); } void ShowClientMenuCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; // TODO: ClientMenu only accepts lists of FluxboxWindows for now FocusControl::Focusables *win_list = 0; // if (m_option & FocusControl::CYCLEGROUPS) { win_list = (m_option & FocusControl::CYCLELINEAR) ? &screen->focusControl().creationOrderWinList() : &screen->focusControl().focusedOrderWinList(); /* } else { win_list = (m_option & FocusControl::CYCLELINEAR) ? &screen->focusControl().creationOrderList() : &screen->focusControl().focusedOrderList(); } */ m_list.clear(); FocusControl::Focusables::iterator it = win_list->begin(), it_end = win_list->end(); for (; it != it_end; ++it) { if (typeid(**it) == typeid(FluxboxWindow) && m_pat.match(**it)) m_list.push_back(static_cast<FluxboxWindow *>(*it)); } m_menu = new ClientMenu(*screen, m_list, 0); ::showMenu(*screen, **m_menu); } ShowCustomMenuCmd::ShowCustomMenuCmd(const string &arguments) : custom_menu_file(arguments) {} void ShowCustomMenuCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; m_menu = MenuCreator::createFromFile(custom_menu_file, screen->screenNumber(), true); if (!m_menu.get()) return; ::showMenu(*screen, **m_menu); } void ShowRootMenuCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; ::showMenu(*screen, screen->rootMenu()); } void ShowWorkspaceMenuCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; ::showMenu(*screen, screen->workspaceMenu()); } SetWorkspaceNameCmd::SetWorkspaceNameCmd(const string &name, int spaceid): m_name(name), m_workspace(spaceid) { } void SetWorkspaceNameCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) { screen = Fluxbox::instance()->keyScreen(); if (screen == 0) return; } if (m_workspace < 0) { screen->currentWorkspace()->setName(m_name); } else { Workspace *space = screen->getWorkspace(m_workspace); if (space == 0) return; space->setName(m_name); } screen->updateWorkspaceNamesAtom(); Fluxbox::instance()->save_rc(); } void WorkspaceNameDialogCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; CommandDialog *win = new CommandDialog(*screen, "Set Workspace Name:", "SetWorkspaceName "); win->setText(screen->currentWorkspace()->name()); win->show(); } void CommandDialogCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; FbTk::FbWindow *win = new CommandDialog(*screen, "Fluxbox Command"); win->show(); } SetResourceValueCmd::SetResourceValueCmd(const string &resname, const string &value): m_resname(resname), m_value(value) { } void SetResourceValueCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; screen->resourceManager().setResourceValue(m_resname, m_value); Fluxbox::instance()->save_rc(); } void SetResourceValueDialogCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; FbTk::FbWindow *win = new CommandDialog(*screen, "Type resource name and the value", "SetResourceValue "); win->show(); }; BindKeyCmd::BindKeyCmd(const string &keybind):m_keybind(keybind) { } void BindKeyCmd::execute() { if (Fluxbox::instance()->keys() != 0) { if (Fluxbox::instance()->keys()->addBinding(m_keybind)) { ofstream ofile(Fluxbox::instance()->keys()->filename().c_str(), ios::app); if (!ofile) return; ofile<<m_keybind<<endl; } } } DeiconifyCmd::DeiconifyCmd(Mode mode, Destination dest) : m_mode(mode), m_dest(dest) { } void DeiconifyCmd::execute() { BScreen *screen = Fluxbox::instance()->mouseScreen(); if (screen == 0) return; // we need to make a copy of the list of icons, or else our iterator can // become invalid BScreen::Icons icon_list = screen->iconList(); BScreen::Icons::iterator it = icon_list.begin(); BScreen::Icons::iterator itend= icon_list.end(); unsigned int workspace_num= screen->currentWorkspaceID(); unsigned int old_workspace_num; const bool change_ws= m_dest == ORIGIN; switch(m_mode) { case ALL: case ALLWORKSPACE: for(; it != itend; it++) { old_workspace_num= (*it)->workspaceNumber(); if (m_mode == ALL || old_workspace_num == workspace_num || (*it)->isStuck()) { if (m_dest == ORIGIN || m_dest == ORIGINQUIET) screen->sendToWorkspace(old_workspace_num, (*it), change_ws); else (*it)->deiconify(false); } } break; case LAST: case LASTWORKSPACE: default: for (; it != itend; it++) { old_workspace_num= (*it)->workspaceNumber(); if(m_mode == LAST || old_workspace_num == workspace_num || (*it)->isStuck()) { if ((m_dest == ORIGIN || m_dest == ORIGINQUIET) && m_mode != LASTWORKSPACE) screen->sendToWorkspace(old_workspace_num, (*it), change_ws); else (*it)->deiconify(false); break; } } break; }; } }; // end namespace FbCommands