]> www.fi.muni.cz Git - slotcarman.git/blobdiff - SCX/GUI.pm
Refactorized GUI layer.
[slotcarman.git] / SCX / GUI.pm
index 9190fe39a328a4660dcb2e612d65893751aeea24..3d500f695f311902be9363ed6e21f498068b1296 100755 (executable)
@@ -5,22 +5,21 @@ package SCX::GUI;
 use strict;
 use utf8;
 
+use Carp;
+use Encode;
 use Gtk2 '-init';
 use Glib qw(TRUE FALSE);
 
-use SCX::Reader;
+our $BASE_SIZE;
 
 sub new {
        my ($class, $args) = @_;
 
        my $self = {
-               throttle_images => [ 
-                       load_image_set('img/throttle%d.svg', 13, 100)
-               ],
-               fuel_images => [
-                       load_image_set('img/fuel%d.svg', 8, 100)
-               ],
+               img_height => $args->{img_height} || 100,
                builder => Gtk2::Builder->new,
+               semaphore_win => Gtk2::Window->new('popup'),
+               semaphore_img => Gtk2::Image->new,
        };
 
        bless $self, $class;
@@ -28,7 +27,24 @@ sub new {
        $self->{builder}->add_from_file('slotcarman.glade');
 
        $self->{builder}->connect_signals(undef);
+       $self->load_all_images;
+       $self->set_font_sizes;
 
+       my $pixbuf = $self->{semaphore_images}->[0];
+       $self->{semaphore_img}->set_from_pixbuf($pixbuf);
+       $self->{semaphore_img}->show;
+       $self->{semaphore_win}->add($self->{semaphore_img});
+       $self->{semaphore_win}->set_type_hint('splashscreen');
+       $self->{semaphore_win}->set_position('center-on-parent');
+       $self->{semaphore_win}->set_transient_for(
+               $self->get_object('slotcarman')
+       );
+
+       $self->get_object('image_car1')->signal_connect(
+               'size_allocate' => \&watch_resize,
+               $self,
+       );
+       
        return $self;
 }
 
@@ -36,11 +52,73 @@ sub show {
        my ($self) = @_;
 
        my $window = $self->{builder}->get_object('slotcarman');
+       $window->signal_connect(delete_event => \&delete_event);
        $window->show();
 }
 
+sub delete_event { Gtk2->main_quit; }
+
+sub quit { Gtk2->main_quit; }
+
+sub fullscreen {
+       my ($self) = @_;
+
+       $self->get_object('slotcarman')->fullscreen;
+       my $h = $self->get_object('image_car1')->allocation->height;
+       print "h=$h\n";
+}
+       
+sub watch_resize {
+       my ($window, $rectangle, $self) = @_;
+       print "w=", $rectangle->width, ", h=", $rectangle->height, "\n";
+       # TODO
+}
+
 sub get_object { return shift->{builder}->get_object(@_); }
 
+sub set_font_sizes {
+       my ($self) = @_;
+
+       if ($self->{img_height} <= 80) {
+               $self->{label_size} = 10;
+               $self->{value_size} = 18;
+               $self->{lap_size} = 24;
+       } elsif ($self->{img_height} <= 100) {
+               $self->{label_size} = 12;
+               $self->{value_size} = 20;
+               $self->{lap_size} = 28;
+       } elsif ($self->{img_height} <= 120) {
+               $self->{label_size} = 14;
+               $self->{value_size} = 24;
+               $self->{lap_size} = 32;
+       } elsif ($self->{img_height} <= 150) {
+               $self->{label_size} = 16;
+               $self->{value_size} = 28;
+               $self->{lap_size} = 36;
+       } else {
+               $self->{label_size} = 18;
+               $self->{value_size} = 32;
+               $self->{lap_size} = 40;
+       }
+}
+
+sub load_all_images {
+       my ($self) = @_;
+
+       my $h = $self->{img_height};
+
+       $self->{throttle_images} = [
+               load_image_set('img/throttle%d.svg', 13, $h)
+       ];
+       $self->{fuel_images} = [
+                       load_image_set('img/fuel%d.svg', 8, int(0.6 * $h))
+       ];
+       $self->{car_images} = load_image_dir('img/SCXCars', $h),
+       $self->{semaphore_images} => [
+                       load_image_set('img/semaphore%d.svg', 5, 2 * $h)
+       ];
+};
+
 sub load_image_set {
        my ($pattern, $limit, $height) = @_;
 
@@ -57,24 +135,153 @@ sub load_image_set {
        return @rv;
 }
 
+sub load_image_dir {
+       my ($dir, $height) = @_;
+
+       my %rv;
+       for my $file (<$dir/*>) {
+               $file = Encode::decode('utf-8', $file);
+               my $name = $file;
+               $name =~ s/.*\///;
+               $name =~ s/\..*?$//;
+               $name =~ s/[-_]/ /g;
+               $name =~ s/\s+/ /g;
+
+
+               my $dummy = Gtk2::Gdk::Pixbuf->new_from_file($file)
+                       or next;
+
+               my $width = $dummy->get_width * $height / $dummy->get_height;
+               $rv{$name} = Gtk2::Gdk::Pixbuf->new_from_file_at_scale(
+                       $file, $width, $height, TRUE
+               );
+       }
+       return \%rv;
+}
+
+sub set_label {
+       my ($self, $prefix, $row, $text, $size) = @_;
+
+       ++$row if defined $row && $row =~ /\A\d+\z/xms;
+       $size //= 'value';
+
+       my $name = defined $row ? $prefix.$row : $prefix;
+       my $label = $self->get_object($name)
+               or croak "Can't get label named $name";
+       $label->set_markup('<span font_desc="' . $self->{$size.'_size'} . '"'
+               . ' weight="bold">' . $text . '</span>');
+}
+
+sub set_image {
+       my ($self, $prefix, $row, $imgset, $value) = @_;
+
+       ++$row if defined $row && $row =~ /\A\d+\z/xms;
+
+       my $name = defined $row ? $prefix.$row : $prefix;
+
+       my $image = $self->get_object($name)
+               or croak "Can't get image named $name";
+
+       my $set = $self->{$imgset.'_images'};
+       my $pixbuf = (ref($set) eq 'ARRAY')
+               ? $set->[$value]
+               : $set->{$value};
+               
+       $image->set_from_pixbuf($pixbuf);
+}
+
 sub set_throttle {
        my ($self, $row, $val) = @_;
 
-       $row++;
        $val = 13 if !defined $val;
-
-       my $image = $self->get_object("image_throttle$row");
-       $image->set_from_pixbuf($self->{throttle_images}->[$val]);
+       $self->set_image('image_throttle', $row, 'throttle', $val);
 }
 
 sub set_fuel {
        my ($self, $row, $val) = @_;
 
-       $row++;
-       $val = 0 if !defined $val;
+       $self->set_image('image_fuel', $row, 'fuel', $val || 0);
+}
+
+sub set_car_icon {
+       my ($self, $row, $name) = @_;
+
+       $self->set_image('image_car', $row, 'car', $name);
+}
+
+sub set_lap {
+       my ($self, $row, $text) = @_;
+
+       $self->set_label('label_lap', $row, $text, 'lap');
+}
+
+sub set_laptime {
+       my ($self, $row, $text) = @_;
+
+       $self->set_label('label_laptime', $row, $text);
+}
+
+sub set_avg_lap {
+       my ($self, $row, $text) = @_;
+
+       $self->set_label('label_avg_lap', $row, $text);
+}
+
+sub set_best_lap {
+       my ($self, $row, $text) = @_;
+
+       $self->set_label('label_best_lap', $row, $text);
+}
+
+sub set_name {
+       my ($self, $row, $text) = @_;
+
+       $self->set_label('label_name', $row, $text);
+}
+
+sub enter_pit_lane {
+       my ($self, $row) = @_;
+       $self->set_laptime($row, 'PIT');
+}
+
+sub leave_pit_lane {
+       my ($self, $row) = @_;
+       $self->set_laptime($row, 'GO!');
+}
+
+sub show_semaphore {
+       my ($self, $num) = @_;
+
+       my $win = $self->{semaphore_win};
+       if (defined $num) {
+               my $img = $self->{semaphore_img};
+               my $pixbuf = $self->{semaphore_images}->[$num];
+               $img->set_from_pixbuf($pixbuf);
+               $win->show;
+       } else {
+               $win->hide;
+       }
+}
+
+sub rounds {
+       my ($self, $text) = @_;
+
+       $self->set_label('label_lap_global', undef, $text);
+}
+
+sub time {
+       my ($self, $text) = @_;
+
+       $self->set_label('label_time_global', undef, $text);
+}
+
+sub best_lap {
+       my ($self, $time, $who) = @_;
 
-       my $image = $self->get_object("image_fuel$row");
-       $image->set_from_pixbuf($self->{fuel_images}->[$val]);
+       my $text = $time;
+       $text .= ' by ' . $who
+               if defined $who;
+       $self->set_label('label_best_lap_global', undef, $text);
 }
 
 1;