fluxbox/util/fbrun/FbRun.cc

441 lines
12 KiB
C++
Raw Normal View History

// FbRun.cc
// Copyright (c) 2002-2003 Henrik Kinnunen (fluxgen<at>users.sourceforge.net)
2002-08-20 02:05:17 +00:00
//
// 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: FbRun.cc,v 1.12 2003/06/24 10:22:42 fluxgen Exp $
2002-08-20 02:05:17 +00:00
#include "FbRun.hh"
2002-11-26 17:13:36 +00:00
#include "App.hh"
#include "EventManager.hh"
2002-12-05 00:07:39 +00:00
#include "Color.hh"
2002-08-20 02:05:17 +00:00
#include <X11/Xlib.h>
#include <X11/keysym.h>
#include <X11/Xutil.h>
2003-03-22 11:33:04 +00:00
#include <X11/cursorfont.h>
2002-08-20 02:05:17 +00:00
#include <unistd.h>
#include <iostream>
2002-11-12 19:20:31 +00:00
#include <fstream>
2003-04-27 02:26:21 +00:00
#include <cassert>
2002-08-20 02:05:17 +00:00
using namespace std;
FbRun::FbRun(int x, int y, size_t width):
2002-12-05 00:07:39 +00:00
m_font("fixed"),
m_win((int)0, x, y, //screen num and position
2003-03-22 11:33:04 +00:00
width + m_bevel, m_font.height() + 2, // size
2002-12-05 00:07:39 +00:00
KeyPressMask|ExposureMask), // eventmask
m_display(FbTk::App::instance()->display()),
m_bevel(4),
m_gc(DefaultGC(m_display, DefaultScreen(m_display))),
m_end(false),
2003-03-22 11:33:04 +00:00
m_current_history_item(0),
m_cursor(XCreateFontCursor(FbTk::App::instance()->display(), XC_xterm)),
m_start_pos(0),
m_end_pos(0),
m_cursor_pos(0)
{
m_win.setCursor(m_cursor);
2002-12-05 00:07:39 +00:00
// setting nomaximize in local resize
resize(width, m_font.height());
FbTk::EventManager::instance()->registerEventHandler(*this, m_win.window());
2002-08-20 02:05:17 +00:00
}
2002-08-20 02:05:17 +00:00
FbRun::~FbRun() {
2002-12-05 00:07:39 +00:00
hide();
FbTk::EventManager::instance()->unregisterEventHandler(m_win.window());
2002-08-20 02:05:17 +00:00
}
void FbRun::run(const std::string &command) {
2002-12-05 00:07:39 +00:00
//fork and execute program
if (!fork()) {
setsid();
execl("/bin/sh", "/bin/sh", "-c", command.c_str(), 0);
exit(0); //exit child
}
hide(); // hide gui
2002-12-05 00:07:39 +00:00
// save command history to file
if (m_runtext.size() != 0) { // no need to save empty command
// open file in append mode
ofstream outfile(m_history_file.c_str(), ios::app);
if (outfile)
outfile<<m_runtext<<endl;
else
cerr<<"FbRun Warning: Can't write command history to file: "<<m_history_file<<endl;
}
FbTk::App::instance()->end(); // end application
m_end = true; // mark end of processing
2002-08-20 02:05:17 +00:00
}
2002-11-12 19:20:31 +00:00
bool FbRun::loadHistory(const char *filename) {
2002-12-05 00:07:39 +00:00
if (filename == 0)
return false;
ifstream infile(filename);
if (!infile) {
//even though we fail to load file, we should try save to it
m_history_file = filename;
return false;
}
// clear old history and load new one from file
m_history.clear();
// each line is a command
string line;
while (!infile.eof()) {
getline(infile, line);
if (line.size()) // don't add empty lines
m_history.push_back(line);
}
// set no current histor to display
m_current_history_item = m_history.size();
// set history file
m_history_file = filename;
return true;
2002-11-12 19:20:31 +00:00
}
2002-08-20 02:05:17 +00:00
bool FbRun::loadFont(const string &fontname) {
2002-12-05 00:07:39 +00:00
if (!m_font.load(fontname.c_str()))
return false;
2002-08-20 02:05:17 +00:00
2002-12-05 00:07:39 +00:00
// resize to fit new font height
resize(m_win.width(), m_font.height() + m_bevel);
return true;
2002-08-20 02:05:17 +00:00
}
2002-12-05 00:07:39 +00:00
void FbRun::setForeground(const FbTk::Color &color) {
XSetForeground(m_display, m_gc, color.pixel());
redrawLabel();
2002-08-20 02:05:17 +00:00
}
2002-12-05 00:07:39 +00:00
void FbRun::setBackground(const FbTk::Color &color) {
m_win.setBackgroundColor(color);
redrawLabel();
2002-08-20 02:05:17 +00:00
}
void FbRun::setText(const string &text) {
2002-12-05 00:07:39 +00:00
m_runtext = text;
redrawLabel();
2002-08-20 02:05:17 +00:00
}
void FbRun::setTitle(const string &title) {
2002-12-05 00:07:39 +00:00
m_win.setName(title.c_str());
2002-08-20 02:05:17 +00:00
}
void FbRun::move(int x, int y) {
2002-12-05 00:07:39 +00:00
m_win.move(x, y);
2002-08-20 02:05:17 +00:00
}
void FbRun::resize(size_t width, size_t height) {
m_win.resize(width, height);
2002-12-05 00:07:39 +00:00
setNoMaximize();
2002-08-20 02:05:17 +00:00
}
void FbRun::show() {
2002-12-05 00:07:39 +00:00
m_win.show();
2002-08-20 02:05:17 +00:00
}
void FbRun::hide() {
2002-12-05 00:07:39 +00:00
m_win.hide();
2002-08-20 02:05:17 +00:00
}
void FbRun::redrawLabel() {
2002-12-05 00:07:39 +00:00
m_win.clear();
drawString(m_bevel/2, m_font.ascent() + m_bevel/2,
m_runtext.c_str(), m_runtext.size());
2002-08-20 02:05:17 +00:00
}
void FbRun::drawString(int x, int y,
2002-12-05 00:07:39 +00:00
const char *text, size_t len) {
assert(m_gc);
2002-08-20 02:05:17 +00:00
m_font.drawText(m_win.window(), DefaultScreen(m_display), m_gc, text + m_start_pos, m_end_pos - m_start_pos, x, y - 2);
2003-03-22 11:33:04 +00:00
// draw cursor position
int cursor_pos = m_font.textWidth(text + m_start_pos, m_cursor_pos) + 1;
m_win.drawLine(m_gc, cursor_pos, 0, cursor_pos, m_font.height());
2002-08-20 02:05:17 +00:00
}
void FbRun::keyPressEvent(XKeyEvent &ke) {
2002-12-05 00:07:39 +00:00
KeySym ks;
char keychar[1];
XLookupString(&ke, keychar, 1, &ks, 0);
// a modifier key by itself doesn't do anything
if (IsModifierKey(ks)) return;
if (ke.state) { // a modifier key is down
if (ke.state == ControlMask) {
switch (ks) {
case XK_b:
cursorLeft();
break;
case XK_f:
cursorRight();
break;
case XK_p:
prevHistoryItem();
break;
case XK_n:
nextHistoryItem();
break;
case XK_a:
cursorHome();
break;
case XK_e:
cursorEnd();
break;
case XK_d:
deleteForward();
break;
case XK_k:
killToEnd();
break;
}
} else if (ke.state == (Mod1Mask | ShiftMask)) {
switch (ks) {
case XK_less:
firstHistoryItem();
break;
case XK_greater:
lastHistoryItem();
break;
}
} else if (ke.state == ShiftMask) {
if (isprint(keychar[0]))insertCharacter(ks, keychar);
2002-12-05 00:07:39 +00:00
}
} else { // no modifier key
2002-12-05 00:07:39 +00:00
switch (ks) {
case XK_Escape:
m_end = true;
hide();
FbTk::App::instance()->end(); // end program
break;
case XK_Return:
run(m_runtext);
m_runtext = ""; // clear text
break;
case XK_BackSpace:
backspace();
break;
case XK_Home:
cursorHome();
break;
case XK_End:
cursorEnd();
break;
2002-12-05 00:07:39 +00:00
case XK_Up:
prevHistoryItem();
break;
case XK_Down:
nextHistoryItem();
break;
2003-03-22 11:33:04 +00:00
case XK_Left:
cursorLeft();
break;
case XK_Right:
cursorRight();
break;
case XK_Tab:
tabCompleteHistory();
break;
default:
if (isprint(keychar[0])) insertCharacter(ks, keychar);
2002-12-05 00:07:39 +00:00
}
}
redrawLabel();
2002-08-20 02:05:17 +00:00
}
void FbRun::exposeEvent(XExposeEvent &ev) {
2002-12-05 00:07:39 +00:00
redrawLabel();
}
2002-08-20 02:05:17 +00:00
void FbRun::setNoMaximize() {
2002-12-05 00:07:39 +00:00
// we don't need to maximize this window
XSizeHints sh;
sh.flags = PMaxSize | PMinSize;
sh.max_width = m_win.width();
sh.max_height = m_win.height();
sh.min_width = m_win.width();
sh.min_height = m_win.height();
XSetWMNormalHints(m_display, m_win.window(), &sh);
2002-08-20 02:05:17 +00:00
}
2002-11-12 19:20:31 +00:00
void FbRun::prevHistoryItem() {
if (m_history.size() == 0 || m_current_history_item == 0) {
XBell(m_display, 0);
} else {
2002-12-05 00:07:39 +00:00
m_current_history_item--;
m_runtext = m_history[m_current_history_item];
m_cursor_pos = m_end_pos = m_runtext.size();
adjustStartPos();
}
2002-11-12 19:20:31 +00:00
}
void FbRun::nextHistoryItem() {
if (m_current_history_item == m_history.size()) {
XBell(m_display, 0);
} else {
m_current_history_item++;
if (m_current_history_item == m_history.size()) {
m_current_history_item = m_history.size();
m_runtext = "";
m_start_pos = m_cursor_pos = m_end_pos = 0;
} else {
m_runtext = m_history[m_current_history_item];
m_cursor_pos = m_end_pos = m_runtext.size();
adjustStartPos();
}
}
}
void FbRun::firstHistoryItem() {
if (m_history.size() == 0 || m_current_history_item == 0) {
XBell(m_display, 0);
} else {
m_current_history_item = 0;
2002-12-05 00:07:39 +00:00
m_runtext = m_history[m_current_history_item];
m_cursor_pos = m_end_pos = m_runtext.size();
adjustStartPos();
}
}
2002-11-12 19:20:31 +00:00
void FbRun::lastHistoryItem() {
// actually one past the end
if (m_history.size() == 0) {
XBell(m_display, 0);
} else {
m_current_history_item = m_history.size();
m_runtext = "";
m_start_pos = m_cursor_pos = m_end_pos = 0;
}
2002-11-12 19:20:31 +00:00
}
2003-03-22 11:33:04 +00:00
void FbRun::tabCompleteHistory() {
if (m_current_history_item == 0) {
XBell(m_display, 0);
} else {
int history_item = m_current_history_item - 1;
string prefix = m_runtext.substr(0, m_cursor_pos);
while (history_item > - 1) {
if (m_history[history_item].find(prefix) == 0) {
m_current_history_item = history_item;
m_runtext = m_history[m_current_history_item];
adjustEndPos();
break;
}
history_item--;
}
if (history_item == -1) XBell(m_display, 0);
}
}
2003-03-22 11:33:04 +00:00
void FbRun::cursorLeft() {
if (m_cursor_pos)
2003-03-22 11:33:04 +00:00
m_cursor_pos--;
else if (m_start_pos) {
m_start_pos--;
adjustEndPos();
}
2003-03-22 11:33:04 +00:00
}
void FbRun::cursorRight() {
if (m_start_pos + m_cursor_pos < m_end_pos)
m_cursor_pos++;
else if (m_end_pos < m_runtext.size()) {
m_cursor_pos++;
m_end_pos++;
adjustStartPos();
}
}
void FbRun::cursorHome() {
m_start_pos = m_cursor_pos = 0;
adjustEndPos();
}
void FbRun::cursorEnd() {
m_cursor_pos = m_end_pos = m_runtext.size();
adjustStartPos();
}
void FbRun::backspace() {
if (m_start_pos || m_cursor_pos) {
m_runtext.erase(m_start_pos + m_cursor_pos - 1, 1);
if (m_cursor_pos)
m_cursor_pos--;
else
m_start_pos--;
adjustEndPos();
}
}
void FbRun::deleteForward() {
if (m_start_pos + m_cursor_pos < m_end_pos) {
m_runtext.erase(m_start_pos + m_cursor_pos, 1);
adjustEndPos();
}
}
void FbRun::killToEnd() {
if (m_start_pos + m_cursor_pos < m_end_pos) {
m_runtext.erase(m_start_pos + m_cursor_pos);
adjustEndPos();
}
}
void FbRun::insertCharacter(KeySym ks, char *keychar) {
char in_char[2] = {keychar[0], 0};
m_runtext.insert(m_start_pos + m_cursor_pos, in_char);
2003-03-22 11:33:04 +00:00
m_cursor_pos++;
m_end_pos++;
if (m_start_pos + m_cursor_pos < m_end_pos)
adjustEndPos();
else
adjustStartPos();
}
void FbRun::adjustEndPos() {
m_end_pos = m_runtext.size();
const char *text = m_runtext.c_str();
int text_width = m_font.textWidth(text + m_start_pos, m_end_pos - m_start_pos);
while (text_width > m_win.width()) {
m_end_pos--;
text_width = m_font.textWidth(text + m_start_pos, m_end_pos - m_start_pos);
}
}
void FbRun::adjustStartPos() {
const char *text = m_runtext.c_str();
int text_width = m_font.textWidth(text + m_start_pos, m_end_pos - m_start_pos);
if (text_width < m_win.width()) return;
int start_pos = 0;
text_width = m_font.textWidth(text + start_pos, m_end_pos - start_pos);
while (text_width > m_win.width()) {
start_pos++;
text_width = m_font.textWidth(text + start_pos, m_end_pos - start_pos);
}
// adjust m_cursor_pos according relative to change to m_start_pos
m_cursor_pos -= start_pos - m_start_pos;
m_start_pos = start_pos;
2003-03-22 11:33:04 +00:00
}