/**************************************************************************** KHotKeys Copyright (C) 1999-2001 Lubos Lunak <l.lunak@kde.org> Distributed under the terms of the GNU General Public License version 2. ****************************************************************************/ #define _ACTIONS_CPP_ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include "actions.h" #include <krun.h> #include <kconfig.h> #include <kdebug.h> #include <kurifilter.h> #include <kglobal.h> #include <kstandarddirs.h> #include <kapplication.h> #include <dcopclient.h> #include <kdesktopfile.h> #include <klocale.h> #include <kaccel.h> #include <kservice.h> #include <kprocess.h> #include "windows.h" #include "action_data.h" #include <X11/X.h> namespace KHotKeys { // Action Action* Action::create_cfg_read( KConfig& cfg_P, Action_data* data_P ) { TQString type = cfg_P.readEntry( "Type" ); if( type == "COMMAND_URL" ) return new Command_url_action( cfg_P, data_P ); if( type == "MENUENTRY" ) return new Menuentry_action( cfg_P, data_P ); if( type == "DCOP" ) return new Dcop_action( cfg_P, data_P ); if( type == "KEYBOARD_INPUT" ) return new Keyboard_input_action( cfg_P, data_P ); if( type == "ACTIVATE_WINDOW" ) return new Activate_window_action( cfg_P, data_P ); kdWarning( 1217 ) << "Unknown Action type read from cfg file\n"; return NULL; } void Action::cfg_write( KConfig& cfg_P ) const { cfg_P.writeEntry( "Type", "ERROR" ); // derived classes should call with their type } // Action_list Action_list::Action_list( KConfig& cfg_P, Action_data* data_P ) : TQPtrList< Action >() { setAutoDelete( true ); TQString save_cfg_group = cfg_P.group(); int cnt = cfg_P.readNumEntry( "ActionsCount", 0 ); for( int i = 0; i < cnt; ++i ) { cfg_P.setGroup( save_cfg_group + TQString::number( i )); Action* action = Action::create_cfg_read( cfg_P, data_P ); if( action ) append( action ); } cfg_P.setGroup( save_cfg_group ); } void Action_list::cfg_write( KConfig& cfg_P ) const { TQString save_cfg_group = cfg_P.group(); int i = 0; for( Iterator it( *this ); it; ++it, ++i ) { cfg_P.setGroup( save_cfg_group + TQString::number( i )); it.current()->cfg_write( cfg_P ); } cfg_P.setGroup( save_cfg_group ); cfg_P.writeEntry( "ActionsCount", i ); } // Command_url_action Command_url_action::Command_url_action( KConfig& cfg_P, Action_data* data_P ) : Action( cfg_P, data_P ) { _command_url = cfg_P.readEntry( "CommandURL" ); } void Command_url_action::cfg_write( KConfig& cfg_P ) const { base::cfg_write( cfg_P ); cfg_P.writeEntry( "CommandURL", command_url()); cfg_P.writeEntry( "Type", "COMMAND_URL" ); // overwrites value set in base::cfg_write() } void Command_url_action::execute() { if( command_url().isEmpty()) return; KURIFilterData uri; TQString cmd = command_url(); static bool sm_ready = false; if( !sm_ready ) { kapp->propagateSessionManager(); sm_ready = true; } // int space_pos = command_url().find( ' ' ); // if( command_url()[ 0 ] != '\'' && command_url()[ 0 ] != '"' && space_pos > -1 // && command_url()[ space_pos - 1 ] != '\\' ) // cmd = command_url().left( space_pos ); // get first 'word' uri.setData( cmd ); KURIFilter::self()->filterURI( uri ); if( uri.uri().isLocalFile() && !uri.uri().hasRef() ) cmd = uri.uri().path(); else cmd = uri.uri().url(); switch( uri.uriType()) { case KURIFilterData::LOCAL_FILE: case KURIFilterData::LOCAL_DIR: case KURIFilterData::NET_PROTOCOL: case KURIFilterData::HELP: { ( void ) new KRun( uri.uri()); break; } case KURIFilterData::EXECUTABLE: { if (!kapp->authorize("shell_access")) return; if( !uri.hasArgsAndOptions()) { KService::Ptr service = KService::serviceByDesktopName( cmd ); if( service != NULL ) { KRun::run( *service, KURL::List()); break; } } // fall though } case KURIFilterData::SHELL: { if (!kapp->authorize("shell_access")) return; if( !KRun::runCommand( cmd + ( uri.hasArgsAndOptions() ? uri.argsAndOptions() : "" ), cmd, uri.iconName())) { // CHECKME ? } break; } default: // error return; } timeout.start( 1000, true ); // 1sec timeout } TQString Command_url_action::description() const { return i18n( "Command/URL : " ) + command_url(); } Action* Command_url_action::copy( Action_data* data_P ) const { return new Command_url_action( data_P, command_url()); } // Menuentry_action void Menuentry_action::cfg_write( KConfig& cfg_P ) const { base::cfg_write( cfg_P ); cfg_P.writeEntry( "Type", "MENUENTRY" ); // overwrites value set in base::cfg_write() } KService::Ptr Menuentry_action::service() const { if (!_service) { const_cast<Menuentry_action *>(this)->_service = KService::serviceByStorageId(command_url()); } return _service; } void Menuentry_action::execute() { (void) service(); if (!_service) return; KRun::run( *_service, KURL::List()); timeout.start( 1000, true ); // 1sec timeout } TQString Menuentry_action::description() const { (void) service(); return i18n( "Menuentry : " ) + (_service ? _service->name() : TQString::null); } Action* Menuentry_action::copy( Action_data* data_P ) const { return new Menuentry_action( data_P, command_url()); } // Dcop_action Dcop_action::Dcop_action( KConfig& cfg_P, Action_data* data_P ) : Action( cfg_P, data_P ) { app = cfg_P.readEntry( "RemoteApp" ); obj = cfg_P.readEntry( "RemoteObj" ); call = cfg_P.readEntry( "Call" ); args = cfg_P.readEntry( "Arguments" ); } void Dcop_action::cfg_write( KConfig& cfg_P ) const { base::cfg_write( cfg_P ); cfg_P.writeEntry( "Type", "DCOP" ); // overwrites value set in base::cfg_write() cfg_P.writeEntry( "RemoteApp", app ); cfg_P.writeEntry( "RemoteObj", obj ); cfg_P.writeEntry( "Call", call ); cfg_P.writeEntry( "Arguments", args ); } void Dcop_action::execute() { if( app.isEmpty() || obj.isEmpty() || call.isEmpty()) return; TQStringList args_list; TQString args_str = args; while( !args_str.isEmpty()) { unsigned int pos = 0; while( args_str[ pos ] == ' ' ) ++pos; if( args_str[ pos ] == '\"' || args_str[ pos ] == '\'' ) { TQString val = ""; TQChar sep = args_str[ pos ]; bool skip = false; ++pos; for(; pos < args_str.length(); ++pos ) { if( args_str[ pos ] == '\\' ) { skip = true; continue; } if( !skip && args_str[ pos ] == sep ) break; skip = false; val += args_str[ pos ]; } if( pos >= args_str.length()) return; ++pos; args_str = args_str.mid( pos ); args_list.append( val ); } else { // one word if( pos != 0 ) args_str = args_str.mid( pos ); int nxt_pos = args_str.find( ' ' ); args_list.append( args_str.left( nxt_pos )); // should be ok if nxt_pos is -1 args_str = nxt_pos >= 0 ? args_str.mid( nxt_pos ) : ""; } } kdDebug( 1217 ) << "DCOP call:" << app << ":" << obj << ":" << call << ":" << args_list << endl; KProcess proc; proc << "dcop" << app << obj << call << args_list; proc.start( KProcess::DontCare ); } TQString Dcop_action::description() const { return i18n( "DCOP : " ) + remote_application() + "::" + remote_object() + "::" + called_function(); } Action* Dcop_action::copy( Action_data* data_P ) const { return new Dcop_action( data_P, remote_application(), remote_object(), called_function(), arguments()); } // Keyboard_input_action Keyboard_input_action::Keyboard_input_action( KConfig& cfg_P, Action_data* data_P ) : Action( cfg_P, data_P ) { _input = cfg_P.readEntry( "Input" ); if( cfg_P.readBoolEntry( "IsDestinationWindow" )) { TQString save_cfg_group = cfg_P.group(); cfg_P.setGroup( save_cfg_group + "DestinationWindow" ); _dest_window = new Windowdef_list( cfg_P ); _active_window = false; // ignored with _dest_window set anyway cfg_P.setGroup( save_cfg_group ); } else { _dest_window = NULL; _active_window = cfg_P.readBoolEntry( "ActiveWindow" ); } } Keyboard_input_action::~Keyboard_input_action() { delete _dest_window; } void Keyboard_input_action::cfg_write( KConfig& cfg_P ) const { base::cfg_write( cfg_P ); cfg_P.writeEntry( "Type", "KEYBOARD_INPUT" ); // overwrites value set in base::cfg_write() cfg_P.writeEntry( "Input", input()); if( dest_window() != NULL ) { cfg_P.writeEntry( "IsDestinationWindow", true ); TQString save_cfg_group = cfg_P.group(); cfg_P.setGroup( save_cfg_group + "DestinationWindow" ); dest_window()->cfg_write( cfg_P ); cfg_P.setGroup( save_cfg_group ); } else cfg_P.writeEntry( "IsDestinationWindow", false ); cfg_P.writeEntry( "ActiveWindow", _active_window ); } void Keyboard_input_action::execute() { if( input().isEmpty()) return; Window w = InputFocus; if( dest_window() != NULL ) { w = windows_handler->find_window( dest_window()); if( w == None ) w = InputFocus; } else { if( !_active_window ) w = windows_handler->action_window(); if( w == None ) w = InputFocus; } int last_index = -1, start = 0; while(( last_index = input().find( ':', last_index + 1 )) != -1 ) // find next ';' { TQString key = input().mid( start, last_index - start ).stripWhiteSpace(); if( key == "Enter" && KKey( key ).keyCodeQt() == 0 ) key = "Return"; // CHECKE hack keyboard_handler->send_macro_key( KKey( key ), w ); start = last_index + 1; } // and the last one TQString key = input().mid( start, input().length()).stripWhiteSpace(); if( key == "Enter" && KKey( key ).keyCodeQt() == 0 ) key = "Return"; keyboard_handler->send_macro_key( KKey( key ), w ); // the rest XFlush( qt_xdisplay()); } TQString Keyboard_input_action::description() const { TQString tmp = input(); tmp.replace( '\n', ' ' ); tmp.truncate( 30 ); return i18n( "Keyboard input : " ) + tmp; } Action* Keyboard_input_action::copy( Action_data* data_P ) const { return new Keyboard_input_action( data_P, input(), dest_window() ? dest_window()->copy() : NULL, _active_window ); } // Activate_window_action Activate_window_action::Activate_window_action( KConfig& cfg_P, Action_data* data_P ) : Action( cfg_P, data_P ) { TQString save_cfg_group = cfg_P.group(); cfg_P.setGroup( save_cfg_group + "Window" ); _window = new Windowdef_list( cfg_P ); cfg_P.setGroup( save_cfg_group ); } Activate_window_action::~Activate_window_action() { delete _window; } void Activate_window_action::cfg_write( KConfig& cfg_P ) const { base::cfg_write( cfg_P ); cfg_P.writeEntry( "Type", "ACTIVATE_WINDOW" ); // overwrites value set in base::cfg_write() TQString save_cfg_group = cfg_P.group(); cfg_P.setGroup( save_cfg_group + "Window" ); window()->cfg_write( cfg_P ); cfg_P.setGroup( save_cfg_group ); } void Activate_window_action::execute() { if( window()->match( windows_handler->active_window())) return; // is already active WId win_id = windows_handler->find_window( window()); if( win_id != None ) windows_handler->activate_window( win_id ); } TQString Activate_window_action::description() const { return i18n( "Activate window : " ) + window()->comment(); } Action* Activate_window_action::copy( Action_data* data_P ) const { return new Activate_window_action( data_P, window()->copy()); } } // namespace KHotKeys