#
# A GTK based console program
#
-# Copyright (c) 2001 Dirk Koopman G1TLH
+# Copyright (c) 2001-6 Dirk Koopman G1TLH
#
# $Id$
#
+use strict;
+
+our $root;
+
# search local then perl directories
BEGIN {
# root of directory tree for this system
unshift @INC, "$root/local";
}
-use strict;
+use Glib;
+use Gtk2 qw(-init);
+use Gtk2::Helper;
+use Gtk2::SimpleList;
-use Gtk qw(-init);
+use Text::Wrap;
use vars qw(@modules $font);
@modules = (); # is the list of modules that need init calling
# on them. It is set up by each 'use'ed module
# that has Gtk stuff in it
-$font = Gtk::Gdk::Font->load("-misc-fixed-medium-r-normal-*-*-130-*-*-c-*-koi8-r");
-
use DXVars;
use DXUtil;
use IO::Socket::INET;
-use Text;
-use DebugHandler;
+#
+# read in gtkconsole file
+
+Gtk2::Rc->set_default_files("$root/gtkconsole/gtkconsolerc", "$ENV{HOME}/.gtkconsolerc", ".gtkconsolerc");
+print join(', ', Gtk2::Rc->get_default_files), "\n";
+Gtk2::Rc->reparse_all;
+
#
# main initialisation
#
die "You cannot connect as your cluster callsign ($main::mycall)\n" if $call eq $main::mycall;
+my $host = 'gb7djk.dxcluster.net';
+my $port = 7300;
-my $sock = IO::Socket::INET->new(PeerAddr=>$main::clusteraddr, PeerPort=>$main::clusterport);
-die "Cannot connect to $main::clusteraddr/$main::clusterport ($!)\n" unless $sock;
-sendmsg('A', 'local');
-sendmsg('G', '2');
+my $sock = IO::Socket::INET->new(PeerAddr=>$host, PeerPort=>$port);
+die "Cannot connect to $host/$port ($!)\n" unless $sock;
+sendmsg('I', $call);
+sendmsg('I', 'set/gtk');
+#sendmsg('A', 'local');
+#sendmsg('G', '2');
sendmsg('I', 'set/page 500');
sendmsg('I', 'set/nobeep');
#
+#
+# +--------+-------+------------------------------------------------------------------------------------+
+# | _File | _Help | |
+# +--------+-------+------------------------------------------------------------------------------------+
+#
# main window
-my $main = new Gtk::Window('toplevel');
-$main->set_default_size(600, 600);
-$main->set_policy(0, 1, 0);
-$main->signal_connect('destroy', sub { Gtk->exit(0); });
-$main->signal_connect('delete_event', sub { Gtk->exit(0); });
+my $main = new Gtk2::Window('toplevel');
+my $scr = $main->get_screen;
+my $scr_width = $scr->get_width;
+my $scr_height = $scr->get_height;
+$main->set_default_size($scr_width, $scr_height/2);
+$main->signal_connect('delete_event', sub { Gtk2->main_quit; });
$main->set_title("gtkconsole - The DXSpider Console - $call");
# the main vbox
-my $vbox = new Gtk::VBox(0, 1);
-$vbox->border_width(1);
+my $vbox = new Gtk2::VBox(0, 1);
$main->add($vbox);
+
# the menu bar
my @menu = (
{path => '/_File', type => '<Branch>'},
- {path => '/_File/Quit', callback => sub {Gtk->exit(0)}},
+ {path => '/_File/Quit', callback => sub {Gtk2->main_quit}},
{path => '/_Help', type => '<LastBranch>'},
{path => '/_Help/About'},
);
-my $accel = new Gtk::AccelGroup();
-my $itemf = new Gtk::ItemFactory('Gtk::MenuBar', '<main>', $accel);
+my $itemf = new Gtk2::ItemFactory('Gtk2::MenuBar', '<main>');
$itemf->create_items(@menu);
-$main->add_accel_group($accel);
my $menu = $itemf->get_widget('<main>');
$vbox->pack_start($menu, 0, 1, 0);
-$menu->show;
-my $top = new Text(1);
-my $toplist = $top->text;
-$toplist->set_editable(0);
-$toplist->sensitive(0);
+# a paned hbox is packed as the bottom of the vbox
+my $bhpane = Gtk2::HPaned->new;
+$vbox->pack_end($bhpane, 1, 1, 0);
+
+# now create the lh and rh panes
+my $lhvpane = Gtk2::VPaned->new;
+my $rhvpane = Gtk2::VPaned->new;
+$bhpane->pack1($lhvpane, 1, 0);
+$bhpane->pack2($rhvpane, 1, 0);
+
+# first add a column type for the QRG
+my $font = 'monospace 9';
+my $oddbg = 'light blue';
+my $evenbg = 'white';
+
+Gtk2::SimpleList->add_column_type( 'qrg',
+ type => 'Glib::Scalar',
+ renderer => 'Gtk2::CellRendererText',
+ attr => sub {
+ my ($treecol, $cell, $model, $iter, $col_num) = @_;
+ my $info = $model->get ($iter, $col_num);
+ $cell->set(text => sprintf("%.1f", $info), font => $font, xalign => 1.0);
+ }
+ );
+
+
+Gtk2::SimpleList->add_column_type( 'tt',
+ type => 'Glib::Scalar',
+ renderer => 'Gtk2::CellRendererText',
+ attr => sub {
+ my ($treecol, $cell, $model, $iter, $col_num) = @_;
+ my $info = $model->get ($iter, $col_num);
+ $cell->set(text => $info, font => $font);
+ }
+ );
+
+Gtk2::SimpleList->add_column_type( 'ttlong',
+ type => 'Glib::Scalar',
+ renderer => 'Gtk2::CellRendererText',
+ attr => sub {
+ my ($treecol, $cell, $model, $iter, $col_num) = @_;
+ my $info = $model->get ($iter, $col_num);
+ $Text::Wrap::columns = 80;
+ $cell->set(text => join("\n",wrap("","",$info)), font => $font);
+ }
+ );
+
+Gtk2::SimpleList->add_column_type( 'ttlesslong',
+ type => 'Glib::Scalar',
+ renderer => 'Gtk2::CellRendererText',
+ attr => sub {
+ my ($treecol, $cell, $model, $iter, $col_num) = @_;
+ my $info = $model->get ($iter, $col_num);
+ $Text::Wrap::columns = 65;
+ $cell->set(text => join("\n",wrap("","",$info)), font => $font);
+ }
+ );
+
+Gtk2::SimpleList->add_column_type( 'ttshort',
+ type => 'Glib::Scalar',
+ renderer => 'Gtk2::CellRendererText',
+ attr => sub {
+ my ($treecol, $cell, $model, $iter, $col_num) = @_;
+ my $info = $model->get ($iter, $col_num);
+ $Text::Wrap::columns = 30;
+ $cell->set(text => join("\n",wrap("","",$info)), font => $font);
+ }
+ );
-# add the handler for incoming messages from the node
-my $tophandler = Gtk::Gdk->input_add($sock->fileno, ['read'], \&tophandler, $sock);
-my $rbuf = ""; # used in handler
-#$toplist->{signalid} = $toplist->signal_connect(insert_text => \&doinsert, $toplist);
-#$bot->{signalid} = $bot->signal_connect(insert_text => \&botinsert, $bot);
-$vbox->pack_start($top, 1, 1, 0);
-$vbox->show;
+#
+# LEFT HAND SIDE
+#
-# the bottom handler
-my $bot = new Gtk::Entry;
-my $style = $toplist->style;
-$style->font($main::font);
-$bot->set_style($style);
-$bot->set_editable(1);
-$bot->signal_connect('activate', \&bothandler);
-$bot->can_default(1);
-$bot->grab_default;
-$bot->show;
+# DX window
+my $dxlist = Gtk2::SimpleList->new(
+ 'RxTime' => 'tt',
+ 'QRG' => 'qrg',
+ 'DX Call' => 'tt',
+ 'Grid' => 'tt',
+ 'Remarks' => 'ttshort',
+ 'By' => 'tt',
+ 'Grid' => 'tt',
+ 'TxTime' => 'tt',
+ );
+$dxlist->set_rules_hint(1);
+my $dxscroll = Gtk2::ScrolledWindow->new (undef, undef);
+$dxscroll->set_shadow_type ('etched-out');
+$dxscroll->set_policy ('never', 'automatic');
+#$dxscroll->set_size_request (700, 400);
+$dxscroll->add($dxlist);
+$dxscroll->set_border_width(5);
+
+$lhvpane->pack1($dxscroll, 1, 0);
+
+# The command list
+my $lhvbox = Gtk2::VBox->new(0, 1);
+my $cmdlist = Gtk2::SimpleList->new(
+ RxTime => 'tt',
+ Information => 'ttlong',
+ );
+my $cmdscroll = Gtk2::ScrolledWindow->new (undef, undef);
+$cmdscroll->set_shadow_type ('etched-out');
+$cmdscroll->set_policy ('automatic', 'automatic');
+#$cmdscroll->set_size_request (700, 400);
+$cmdscroll->add($cmdlist);
+$cmdscroll->set_border_width(5);
+$lhvbox->pack_start($cmdscroll, 1, 1, 0);
-# a horizontal box
-my $hbox = new Gtk::HBox;
-$hbox->show;
# callsign and current date and time
-my $calllabel = new Gtk::Label($call);
-my $date = new Gtk::Label(cldatetime(time));
-Gtk->timeout_add(1000, \&updatetime);
-$calllabel->show;
-$date->show;
-
+my $hbox = new Gtk2::HBox;
+my $calllabel = new Gtk2::Label($call);
+my $date = new Gtk2::Label(cldatetime(time));
+$date->{tick} = Glib::Timeout->add(1000, \&updatetime, 0);
$hbox->pack_start( $calllabel, 0, 1, 0 );
$hbox->pack_end($date, 0, 1, 0);
+$lhvbox->pack_start($hbox, 0, 1, 0);
+$lhvbox->pack_start(Gtk2::HSeparator->new, 0, 1, 0);
+# the bottom handler
+my $bot = new Gtk2::Entry;
+$bot->set_editable(1);
+$bot->signal_connect('activate', \&bothandler);
+$bot->can_default(1);
+$lhvbox->pack_end($bot, 0, 1, 0);
+$lhvpane->pack2($lhvbox, 1, 0);
+$bot->grab_default;
-$vbox->pack_start($hbox, 0, 1, 0);
-
-# nice little separator
-my $separator = new Gtk::HSeparator();
-$vbox->pack_start( $separator, 0, 1, 0 );
-$separator->show();
-$vbox->pack_start($bot, 0, 1, 0);
+#
+# RIGHT HAND SIDE
+#
+# The announce list
+my $annlist = Gtk2::SimpleList->new(
+ RxTime => 'tt',
+ From => 'tt',
+ To => 'tt',
+ Announcement => 'ttlesslong',
+ );
+$annlist->set_rules_hint(1);
+my $annscroll = Gtk2::ScrolledWindow->new (undef, undef);
+$annscroll->set_shadow_type ('etched-out');
+$annscroll->set_policy ('automatic', 'automatic');
+#$annscroll->set_size_request (700, 400);
+$annscroll->add($annlist);
+$annscroll->set_border_width(5);
+$rhvpane->pack1($annscroll, 1, 0);
+
+# The wwv list
+my $rhvbox = Gtk2::VBox->new(0, 1);
+
+my $wwvlist = Gtk2::SimpleList->new(
+ RxTime => 'tt',
+ From => 'tt',
+ SFI => 'int',
+ A => 'int',
+ K => 'int',
+ Remarks => 'ttshort',
+ Hour => 'tt'
+ );
+$wwvlist->set_rules_hint(1);
+my $wwvscroll = Gtk2::ScrolledWindow->new (undef, undef);
+$wwvscroll->set_shadow_type ('etched-out');
+$wwvscroll->set_policy ('never', 'automatic');
+#$wwvscroll->set_size_request (700, 200);
+$wwvscroll->add($wwvlist);
+$wwvscroll->set_border_width(5);
+$rhvbox->pack_start($wwvscroll, 1, 1, 0);
+
+# The wcy list
+my $wcylist = Gtk2::SimpleList->new(
+ RxTime => 'tt',
+ From => 'tt',
+ K => 'int',
+ ExpK => 'int',
+ A => 'int',
+ R => 'int',
+ SFI => 'int',
+ SA => 'tt',
+ GMF => 'tt',
+ Aurora => 'tt',
+ Hour => 'tt'
+ );
+$wcylist->set_rules_hint(1);
+my $wcyscroll = Gtk2::ScrolledWindow->new (undef, undef);
+$wcyscroll->set_shadow_type ('etched-out');
+$wcyscroll->set_policy ('never', 'automatic');
+$wcyscroll->add($wcylist);
+$wcyscroll->set_border_width(5);
+$rhvbox->pack_start($wcyscroll, 1, 1, 0);
+$rhvbox->set_size_request (-1, $scr_height / 4);
+
+
+$rhvpane->pack2($rhvbox, 1, 0);
+
+my $sock_helper = Gtk2::Helper->add_watch($sock->fileno, 'in', \&tophandler, $sock);
+
# the main loop
$main->show_all;
$bot->grab_focus;
-Gtk->main;
+Gtk2->main;
+exit(0);
#
# handlers
1;
}
-sub doinsert {
- my ($self, $text) = @_;
-
- # we temporarily block this handler to avoid recursion
- $self->signal_handler_block($self->{signalid});
- my $pos = $self->insert($self->{font}, $toplist->style->black, $toplist->style->white, $text);
- $self->signal_handler_unblock($self->{signalid});
-
- # we already inserted the text if it was valid: no need
- # for the self to process this signal emission
- $self->signal_emit_stop_by_name('insert-text');
- 1;
-}
-
sub bothandler
{
my ($self, $data) = @_;
senddata($msg);
}
+my $rbuf;
+
sub tophandler
{
- my ($socket, $fd, $flags) = @_;
- if ($flags->{read}) {
- my $offset = length $rbuf;
- my $l = sysread($socket, $rbuf, 1024, $offset);
- if (defined $l) {
- my $freeze;
- if ($l) {
- while ($rbuf =~ s/^([^\015\012]*)\015?\012//) {
- my $msg = $1;
- $msg =~ s/\%([0-9A-F][0-9A-F])/chr(hex($1))/eg;
- $msg =~ s/[\x00-\x08\x0a-\x19\x1b-\x1f\x80-\x9f]/./g; # immutable CSI sequence + control characters
- $toplist->freeze unless $freeze++;
- handlemsg($msg);
- }
- if ($freeze) {
- $toplist->thaw;
- $toplist->vadj->set_value($toplist->vadj->upper);
- $toplist->vadj->value_changed;
- }
- } else {
- Gtk->exit(0);
+ my ($fd, $condx, $socket) = @_;
+
+ my $offset = length $rbuf;
+ my $l = sysread($socket, $rbuf, 1024, $offset);
+ if (defined $l) {
+ if ($l) {
+ while ($rbuf =~ s/^([^\015\012]*)\015?\012//) {
+ my $msg = $1;
+ handlemsg($msg);
}
} else {
- Gtk->exit(0);
+ Gtk2->main_quit;
}
+ } else {
+ Gtk2->main_quit;
}
+ 1;
+
}
sub handlemsg
{
- my $msg = shift;
- my ($sort, $call, $line) = $msg =~ /^(\w)([^\|]+)\|(.*)$/;
- if ($sort eq 'D') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'X') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'T') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'Y') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'V') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'N') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'W') {
- $toplist->insert($toplist->{font}, undef, undef, "$line\n");
- } elsif ($sort eq 'Z') {
- Gtk->exit(0);
+ my $line = shift;
+
+ # this is truely evil and I bet there is a better way...
+ chomp $line;
+ my $list;
+ if ($line =~ /^'\w{2,4}',/) {
+ $list = eval qq([$line]);
+ } else {
+ $list = ['cmd', $line];
}
+ unless ($@) {
+ no strict 'refs';
+ my $cmd = shift @$list;
+ my $handle = "handle_$cmd";
+ if (__PACKAGE__->can($handle)) {
+ __PACKAGE__->$handle($list);
+ } else {
+ unshift @$list, $cmd;
+ __PACKAGE__->handle_def($list);
+ }
+ }
+}
+
+sub handle_cmd
+{
+ my $self = shift;
+ my $ref = shift;
+ my ($t, $ts) = (time, '');
+ my $s;
+ $s = ref $ref ? join ', ',@$ref : $ref;
+
+ if (($cmdscroll->{lasttime}||0) != $t) {
+ $ts = tim($t);
+ $cmdscroll->{lasttime} = $t;
+ }
+
+ chomp $s;
+ push @{$cmdlist->{data}}, [$ts, $s];
+}
+
+sub handle_def
+{
+ my $self = shift;
+ my $ref = shift;
+ my $s;
+ $s = ref $ref ? join ', ',@$ref : $ref;
+ my ($t, $ts) = (time, '');
+
+ if (($cmdscroll->{lasttime}||0) != $t) {
+ $ts = tim($t);
+ $cmdscroll->{lasttime} = $t;
+ }
+
+ chomp $s;
+ push @{$cmdlist->{data}}, [$ts, $s];
+}
+
+sub handle_dx
+{
+ my $self = shift;
+ my $ref = shift;
+ my ($t, $ts) = (time, '');
+
+ if (($dxscroll->{lasttime}||0) != $t) {
+ $ts = tim($t);
+ $dxscroll->{lasttime} = $t;
+ }
+ push @{$dxlist->{data}}, [$ts, @$ref[0,1,15,3,4,16], stim($ref->[2]) ];
+
+}
+
+sub handle_ann
+{
+ my $self = shift;
+ my $ref = shift;
+ my ($t, $ts) = (time, '');
+ my $s;
+ $s = ref $ref ? join ', ',@$ref : $ref;
+
+ if (($annscroll->{lasttime}||0) != $t) {
+ $ts = tim($t);
+ $annscroll->{lasttime} = $t;
+ }
+
+ chomp $s;
+ push @{$annlist->{data}}, [$ts, @$ref[3,1,2]];
+}
+
+sub handle_wcy
+{
+ my $self = shift;
+ my $ref = shift;
+ my $s;
+ $s = ref $ref ? join ', ',@$ref : $ref;
+
+ chomp $s;
+ push @{$wcylist->{data}}, [tim(), @$ref[10,4,5,3,6,2,7,8,9,1] ];
+}
+
+sub handle_wwv
+{
+ my $self = shift;
+ my $ref = shift;
+ my $s;
+ $s = ref $ref ? join ', ',@$ref : $ref;
+
+ chomp $s;
+ push @{$wwvlist->{data}}, [tim(), @$ref[6,2,3,4,5,1] ];
}
#
sub sendmsg
{
my ($let, $msg) = @_;
- $msg =~ s/([\%\x00-\x1f\x7f-\xff])/sprintf("%%%02X", ord($1))/eg;
- $sock->print("$let$call|$msg\n");
+# $msg =~ s/([\%\x00-\x1f\x7f-\xff])/sprintf("%%%02X", ord($1))/eg;
+# $sock->print("$let$call|$msg\n");
+ $sock->print("$msg\n");
+}
+
+sub tim
+{
+ my $t = shift || time;
+ return sprintf "%02d:%02d:%02d", (gmtime($t))[2,1,0];
+}
+
+sub stim
+{
+ my $t = shift || time;
+ return sprintf "%02d:%02d", (gmtime($t))[2,1];
}