/* * "Software pw3270, desenvolvido com base nos códigos fontes do WC3270 e X3270 * (Paul Mattes Paul.Mattes@usa.net), de emulação de terminal 3270 para acesso a * aplicativos mainframe. Registro no INPI sob o nome G3270. * * Copyright (C) <2008> * * Este programa é software livre. Você pode redistribuí-lo e/ou modificá-lo sob * os termos da GPL v.2 - Licença Pública Geral GNU, conforme publicado pela * Free Software Foundation. * * Este programa é distribuído na expectativa de ser útil, mas SEM QUALQUER * GARANTIA; sem mesmo a garantia implícita de COMERCIALIZAÇÃO ou de ADEQUAÇÃO * A QUALQUER PROPÓSITO EM PARTICULAR. Consulte a Licença Pública Geral GNU para * obter mais detalhes. * * Você deve ter recebido uma cópia da Licença Pública Geral GNU junto com este * programa; se não, escreva para a Free Software Foundation, Inc., 59 Temple * Place, Suite 330, Boston, MA, 02111-1307, USA * * Este programa está nomeado como pluginmain.c e possui - linhas de código. * * Contatos: * * perry.werneck@gmail.com (Alexandre Perry de Souza Werneck) * erico.mendonca@gmail.com (Erico Mascarenhas Mendonça) * * Referencias: * * * http://www.oorexx.org/docs/rexxpg/x14097.htm * * http://www.oorexx.org/docs/rexxpg/c2539.htm * */ #define ENABLE_NLS #define GETTEXT_PACKAGE PACKAGE_NAME #include "rx3270.h" #include #include #include #include #include #include #include #include #include /*--[ Globals ]--------------------------------------------------------------------------------------*/ #if GTK_CHECK_VERSION(2,32,0) static GMutex mutex; #else static GStaticMutex mutex = G_STATIC_MUTEX_INIT; #endif // GTK_CHECK_VERSION /*--[ Rexx application data block ]--------------------------------------------------------------------------*/ struct rexx_application_data { GtkAction * action; GtkWidget * widget; const gchar * filename; }; /*--[ Running rexx scripts ]---------------------------------------------------------------------------------*/ static int REXXENTRY Rexx_IO_exit(RexxExitContext *context, int exitnumber, int subfunction, PEXIT parmBlock) { trace("%s call with ExitNumber: %d Subfunction: %d",__FUNCTION__,(int) exitnumber, (int) subfunction); if(subfunction == RXSIOSAY) { GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(((struct rexx_application_data * )context->GetApplicationData())->widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_INFO, GTK_BUTTONS_OK_CANCEL, "%s", (((RXSIOSAY_PARM *) parmBlock)->rxsio_string).strptr ); gtk_window_set_title(GTK_WINDOW(dialog), _( "Script message" ) ); if(gtk_dialog_run(GTK_DIALOG (dialog)) == GTK_RESPONSE_CANCEL) context->RaiseException0(Rexx_Error_Program_interrupted); gtk_widget_destroy(dialog); return RXEXIT_HANDLED; } return RXEXIT_NOT_HANDLED; } static void call_rexx_script(GtkAction *action, GtkWidget *widget, const gchar *filename) { const gchar * args = (const gchar *) g_object_get_data(G_OBJECT(action),"args"); struct rexx_application_data appdata = { action, widget, filename }; RexxInstance * instance; RexxThreadContext * threadContext; RexxOption options[25]; RexxContextExit exits[2]; memset(options,0,sizeof(options)); memset(exits,0,sizeof(exits)); exits[0].sysexit_code = RXSIO; exits[0].handler = Rexx_IO_exit; // http://www.oorexx.org/docs/rexxpg/c2539.htm options[0].optionName = DIRECT_EXITS; options[0].option = (void *) exits; options[1].optionName = APPLICATION_DATA; options[1].option = (void *) &appdata; // options[0].optionName = LOAD_REQUIRED_LIBRARY; // options[0].option = "rx3270"; rx3270_set_package_option(&options[2]); options[3].optionName = EXTERNAL_CALL_PATH; options[3].option = pw3270_get_datadir(NULL); trace("Rexxdir: \"%s\"",(gchar *) ((void *) options[3].option)); if(!RexxCreateInterpreter(&instance, &threadContext, options)) { GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CANCEL, _( "Can't start %s script" ), "rexx" ); gtk_window_set_title(GTK_WINDOW(dialog),_( "Rexx error" )); gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog),_( "Can't create %s interpreter instance" ), "rexx"); gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); } else { RexxArrayObject rxArgs; trace("%s start",__FUNCTION__); if(args) { gchar **arg = g_strsplit(args,",",-1); size_t sz = g_strv_length(arg); rxArgs = threadContext->NewArray(sz); for(unsigned int i = 0; iArrayPut(rxArgs, threadContext->String(arg[i]), i + 1); g_strfreev(arg); } else { rxArgs = threadContext->NewArray(1); threadContext->ArrayPut(rxArgs, threadContext->String(""),1); } v3270_set_script(widget,'R',TRUE); RexxObjectPtr result = threadContext->CallProgram(filename, rxArgs); v3270_set_script(widget,'R',FALSE); if (threadContext->CheckCondition()) { RexxCondition condition; // retrieve the error information and get it into a decoded form RexxDirectoryObject cond = threadContext->GetConditionInfo(); threadContext->DecodeConditionInfo(cond, &condition); // display the errors GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CANCEL, _( "%s script failed" ), "Rexx" ); gtk_window_set_title(GTK_WINDOW(dialog),_( "Rexx error" )); gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG(dialog), _( "%s error %d: %s\n%s" ), "Rexx", (int) condition.code, threadContext->CString(condition.errortext), threadContext->CString(condition.message) ); gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); } else if (result != NULLOBJECT) { CSTRING resultString = threadContext->CString(result); lib3270_write_log(NULL,"REXX","%s exits with rc=%s",filename,resultString); } instance->Terminate(); trace("%s ends",__FUNCTION__); } g_free(options[3].option); } extern "C" { LIB3270_EXPORT void pw3270_action_rexx_activated(GtkAction *action, GtkWidget *widget) { gchar *filename = (gchar *) g_object_get_data(G_OBJECT(action),"src"); lib3270_trace_event(v3270_get_session(widget),"Action %s activated on widget %p",gtk_action_get_name(action),widget); #if GTK_CHECK_VERSION(2,32,0) if(!g_mutex_trylock(&mutex)) #else if(!g_static_mutex_trylock(&mutex)) #endif // GTK_CHECK_VERSION { GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CANCEL, "%s", _( "Can't start script" )); gtk_window_set_title(GTK_WINDOW(dialog),_( "System busy" )); gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog),"%s",_( "Please, try again in a few moments" )); gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); return; } gtk_action_set_sensitive(action,FALSE); if(filename) { // Has filename, call it directly call_rexx_script(action,widget,filename); } else { // No filename, ask user static const struct _list { const gchar *name; const gchar *pattern; } list[] = { { N_( "Rexx script file" ), "*.rex" }, { N_( "Rexx class file" ), "*.cls" } }; GtkFileFilter * filter[G_N_ELEMENTS(list)+1]; unsigned int f; memset(filter,0,sizeof(filter)); for(f=0;fhSession = hSession; } plugin::~plugin() { trace("%s",__FUNCTION__); } char * plugin::get_version(void) { return strdup(lib3270_get_version()); } LIB3270_CSTATE plugin::get_cstate(void) { return lib3270_get_connection_state(hSession); } int plugin::disconnect(void) { lib3270_disconnect(hSession); return 0; } int plugin::connect(const char *uri, bool wait) { return lib3270_connect(hSession,uri,wait); } bool plugin::is_connected(void) { return lib3270_is_connected(hSession) != 0; } int plugin::iterate(bool wait) { if(!lib3270_is_connected(hSession)) return ENOTCONN; lib3270_main_iterate(hSession,wait); return 0; } int plugin::wait(int seconds) { return lib3270_wait(hSession,seconds); } int plugin::enter(void) { return lib3270_enter(hSession); } int plugin::pfkey(int key) { return lib3270_pfkey(hSession,key); } int plugin::pakey(int key) { return lib3270_pakey(hSession,key); } int plugin::wait_for_ready(int seconds) { return lib3270_wait_for_ready(hSession,seconds); } char * plugin::get_text_at(int row, int col, size_t sz) { return lib3270_get_text_at(hSession,row,col,(int) sz); } int plugin::cmp_text_at(int row, int col, const char *text) { return lib3270_cmp_text_at(hSession,row,col,text); } int plugin::set_text_at(int row, int col, const char *str) { return lib3270_set_text_at(hSession,row,col,(const unsigned char *) str); } bool plugin::is_ready(void) { return lib3270_is_ready(hSession) != 0; } int plugin::set_cursor_position(int row, int col) { return lib3270_set_cursor_position(hSession,row,col); } int plugin::set_toggle(LIB3270_TOGGLE ix, bool value) { return lib3270_set_toggle(hSession,ix,(int) value); } void plugin::logva(const char *fmt, va_list args) { lib3270_write_va_log(hSession,"REXX",fmt,args); } char * plugin::get_text(int baddr, size_t len) { return lib3270_get_text(hSession,baddr,len); } int plugin::get_field_start(int baddr) { return lib3270_get_field_start(hSession,baddr); } int plugin::get_field_len(int baddr) { return lib3270_get_field_len(hSession,baddr); } int plugin::set_copy(const char *text) { v3270_set_copy(GTK_WIDGET(lib3270_get_widget(hSession)),text); return 0; } char * plugin::get_copy(void) { return v3270_get_copy(GTK_WIDGET(lib3270_get_widget(hSession))); } char * plugin::get_clipboard(void) { trace("%s toplevel=%p",__FUNCTION__,pw3270_get_toplevel()); return gtk_clipboard_wait_for_text(gtk_widget_get_clipboard(pw3270_get_toplevel(),GDK_SELECTION_CLIPBOARD)); } int plugin::set_clipboard(const char *text) { gtk_clipboard_set_text(gtk_widget_get_clipboard(pw3270_get_toplevel(),GDK_SELECTION_CLIPBOARD),(gchar *) text, -1); return 0; } void plugin::free(void *ptr) { g_free(ptr); } int plugin::set_cursor_addr(int addr) { return lib3270_set_cursor_address(hSession,addr); } int plugin::get_cursor_addr(void) { return lib3270_get_cursor_address(hSession); } int plugin::emulate_input(const char *str) { return lib3270_emulate_input(hSession, str, -1, 1); } int plugin::get_next_unprotected(int baddr) { return lib3270_get_next_unprotected(hSession,baddr); } int plugin::popup_dialog(LIB3270_NOTIFY id , const char *title, const char *message, const char *fmt, ...) { va_list args; va_start(args, fmt); lib3270_popup_va(hSession, id, title, message, fmt, args); va_end(args); return 0; }