#!/usr/bin/perl use strict; use lib '/usr/lib/libDrakX'; use Getopt::Long; use ugtk2 qw(:all); use Gtk2::Gdk::Keysyms; use MDK::Common; use Image::Magick; my $kernel_size = 1500; #- in KiB my $initrd_size = 130; #- in KiB my $lilo_block_size = 50; #- in KiB my $isolinux_block_size = 64; #- in KiB my @modes = ( { Vesa => 0x101, X => 640, Y => 480 }, { Vesa => 0x103, X => 800, Y => 600 }, { Vesa => 0x105, X => 1024, Y => 768 }, { Vesa => 0x107, X => 1280, Y => 1024 }, ); my (%image_size, $kernel_and_initrd_size); my ($progress_rect, $progress_color) = ([], {}); my ($timer_pos, $timer_bg, $timer_fg) = ({}, {}, {}); my ($entry_rect, $entry_selected_color, $entry_color) = ([], {}, {}); my $isolinux_mode; my $magick = Image::Magick->new; my ($current_rect, $current_point); my ($image_area, $image_pixbuf); sub move_point { my ($up_down, $direction) = @_; my $wanted = $current_point->{$direction} + $up_down; if (0 <= $wanted && $wanted < $image_size{$direction}) { $current_point->{$direction} += $up_down; } $image_area->queue_draw; } sub create_image_area() { $image_area = Gtk2::DrawingArea->new; $image_area->can_focus(1); $image_area->add_events('button-press-mask'); $image_area->signal_connect(button_press_event => \&image_button_pressed); $image_area->signal_connect(expose_event => \&image_expose); gtkmodify_font($image_area, 'Monospace 12'); my $keys = { $Gtk2::Gdk::Keysyms{Down} => sub { move_point( ($_[0] ? 5 : 1), 'Y') }, $Gtk2::Gdk::Keysyms{Up} => sub { move_point(-($_[0] ? 5 : 1), 'Y') }, $Gtk2::Gdk::Keysyms{Left} => sub { move_point(-($_[0] ? 5 : 1), 'X') }, $Gtk2::Gdk::Keysyms{Right} => sub { move_point( ($_[0] ? 5 : 1), 'X') }, $Gtk2::Gdk::Keysyms{q} => sub { Gtk2->main_quit }, $Gtk2::Gdk::Keysyms{Escape} => sub { Gtk2->main_quit }, }; $image_area->signal_connect(key_press_event => sub { my (undef, $event) = @_; if (my $f = $keys->{$event->keyval}) { $f->(member('control-mask', @{$event->state})); } 1; }); $image_area->grab_focus; $image_area->show; $image_area; } sub image_expose { my ($widget) = @_; my $window = $widget->window; $window->draw_pixbuf($widget->style->white_gc, $image_pixbuf, 0, 0, 0, 0, -1, -1, 'none', 0, 0); if (!$isolinux_mode) { { my $layout = $widget->create_pango_layout('--:--'); my ($width, $height) = $layout->get_pixel_size; $window->draw_rectangle(color_index2gc($window, $timer_bg), 1, $timer_pos->{X}, $timer_pos->{Y}, $width, $height); $window->draw_layout(color_index2gc($window, $timer_fg), $timer_pos->{X}, $timer_pos->{Y}, $layout); } my ($x, $y, $w, $h) = rectangle2xywh($entry_rect); my @std_labels = ('linux', 'failsafe', '2.6.3-7mdk', 'X' x ($w / 8)); for (my $nb = 0; $nb < int($h / 16); $y += 16, $nb++) { my $label = shift(@std_labels) || 'label_' . ($nb+1); my $gc = color_index2gc($window, !$nb ? $entry_selected_color : $entry_color); my $layout = $widget->create_pango_layout($label); $window->draw_layout($gc, $x, $y, $layout); } } if ($current_rect) { my ($x, $y, $w, $h) = rectangle2xywh($current_rect); $window->draw_rectangle($widget->style->black_gc, 0, $x, $y, $w, $h); } if ($current_rect != $progress_rect) { my ($x, $y, $w, $h) = rectangle2xywh($progress_rect); $window->draw_rectangle(color_index2gc($window, $progress_color), 1, $x, $y, $w, $h); } 0; } sub image_button_pressed { my (undef, $event) = @_; if ($event->button eq '3') { create_popup()->popup(undef, undef, undef, undef, '3', $event->time); return 1; } my $point = { X => $event->x, Y => $event->y }; my $chosen_point; if ($current_rect) { if (!@$current_rect) { my @corners = ({ X => 0, Y => 0 }, { X => 0, Y => $image_size{Y} - 1 }, { X => $image_size{X} - 1, Y => 0 }, { X => $image_size{X} - 1, Y => $image_size{Y} - 1 }); @$current_rect = ($corners[farthest($point, @corners)], nearest($point, @corners)); } $current_point = nearest($point, @$current_rect); } elsif ($current_point) { %$current_point or %$current_point = %$point; } else { return; } %$current_point = %$point; $current_point->{Color} = $magick->Get("index[$point->{X},$point->{Y}]"); $image_area->queue_draw; 1; } sub image_set_file { my ($file) = @_; $image_pixbuf = Gtk2::Gdk::Pixbuf->new_from_file($file); %image_size = (X => $image_pixbuf->get_width, Y => $image_pixbuf->get_height); $image_area->set_size_request($image_size{X}, $image_size{Y}); $image_area->queue_draw; } sub color_index2gc { my ($window, $color) = @_; my ($r, $g, $b) = split(',', $magick->Get("colormap[$color->{Color}]")); my $gc = Gtk2::Gdk::GC->new($window); my $gdk_color = Gtk2::Gdk::Color->new($r, $g, $b) or die "bad color for $r,$g,$b"; $window->get_colormap->rgb_find_color($gdk_color); $gc->set_rgb_fg_color($gdk_color); $gc; } sub rectangle2xywh { my ($rect) = @_; my $x = min($rect->[0]{X} , $rect->[1]{X}); my $y = min($rect->[0]{Y} , $rect->[1]{Y}); my $w = abs($rect->[0]{X} - $rect->[1]{X}); my $h = abs($rect->[0]{Y} - $rect->[1]{Y}); ($x, $y, $w, $h); } sub distance { my ($p1, $p2) = @_; sqr($p1->{X} - $p2->{X}) + sqr($p1->{Y} - $p2->{Y}); } sub farthest { my ($point, @others) = @_; my $i = 0; my $dist = 0; my $farthest; foreach (@others) { my $d = distance($point, $_); if ($d >= $dist) { $dist = $d; $farthest = $_; } } $farthest; } sub nearest { my ($point, @others) = @_; my $i = 0; my $dist; my $nearest; foreach (@others) { my $d = distance($point, $_); if (! defined $dist || $d < $dist) { $dist = $d; $nearest = $_; } } $nearest; } sub create_popup() { my %l = my @l = ( 'Progress bar position' => sub { $current_rect = $progress_rect }, 'Progress bar color' => sub { $current_point = $progress_color }, '' => sub {}, $isolinux_mode ? ( 'switch to lilo mode' => sub { $isolinux_mode = 0 }, ) : ( 'Timer position' => sub { $current_point = $timer_pos }, 'Timer text color' => sub { $current_point = $timer_fg }, 'Timer background' => sub { $current_point = $timer_bg }, '' => sub {}, 'Entry position' => sub { $current_rect = $entry_rect }, 'Entry text selected color' => sub { $current_point = $entry_selected_color }, 'Entry text color' => sub { $current_point = $entry_color }, '' => sub {}, 'switch to isolinux mode' => sub { $isolinux_mode = 1 }, ), ); my $popup = Gtk2::Menu->new; foreach (group_by2(@l)) { my ($descr, $f) = @$_; gtkappend($popup, gtksignal_connect(Gtk2::MenuItem->new_with_label($descr), activate => sub { $current_rect = $current_point = undef; $f->(); })); } $popup } sub read_parameters { my ($file) = @_; my %h = getVarsFromSh($file); $entry_rect->[0]{X} = $h{'entry_x'}; $entry_rect->[0]{Y} = $h{'entry_y'}; $entry_rect->[1]{X} = $h{'entry_x'} + $h{'entry_w'}; $entry_rect->[1]{Y} = $h{'entry_y'} + $h{'entry_h'}; $entry_color->{Color} = $h{'entry_bg'} - 64; $entry_selected_color->{Color} = $h{'entry_fg'} - 64; $timer_pos->{X} = $h{'timer_x'}; $timer_pos->{Y} = $h{'timer_y'}; $timer_bg->{Color} = $h{'timer_bg'} - 64; $timer_fg->{Color} = $h{'timer_fg'} - 64; $progress_color->{Color} = $h{'progress_c'} - 64; $progress_rect->[0]{X} = $h{'progress_x'}; $progress_rect->[0]{Y} = $h{'progress_y'}; $progress_rect->[1]{X} = $h{'progress_x'} + $h{'progress_real_w'}; $progress_rect->[1]{Y} = $h{'progress_y'} + $h{'progress_h'}; $isolinux_mode = $h{'isolinux_mode'}; } sub save_parameters { my ($file) = @_; my %h; $h{'mode'} = (find { $image_size{X} eq $_->{X} } @modes)->{Vesa}; $h{'clear_h'} = $image_size{X}; $h{'clear_w'} = $image_size{Y}; if (!$isolinux_mode) { ($h{'entry_x'}, $h{'entry_y'}, $h{'entry_w'}, $h{'entry_h'}) = rectangle2xywh($entry_rect); $h{'entry_w_chr'} = int($h{'entry_w'} / 8); $h{'entry_h_chr'} = int($h{'entry_h'} / 16); $h{'entry_bg'} = $entry_color->{Color} + 64; $h{'entry_fg'} = $entry_selected_color->{Color} + 64; $h{'timer_x'} = $timer_pos->{X}; $h{'timer_y'} = $timer_pos->{Y}; $h{'timer_bg'} = $timer_bg->{Color} + 64; $h{'timer_fg'} = $timer_fg->{Color} + 64; } $h{'progress_c'} = $progress_color->{Color} + 64; ($h{'progress_x'}, $h{'progress_y'}, $h{'progress_real_w'}, $h{'progress_h'}) = rectangle2xywh($progress_rect); my $nb_steps = $kernel_and_initrd_size / ($isolinux_mode ? $isolinux_block_size : $lilo_block_size); $h{'progress_w'} = int($h{'progress_real_w'} / $nb_steps); $h{'isolinux_mode'} = $isolinux_mode; output($file, map { "$_=$h{$_}\n" } sort keys %h); } # MAIN ######################################################################### my $usage = <<EOF; usage: $0 [--kernel <kernel> --initrd <initrd>] [--size <size in KiB>] <image> (kernel and initrd are used to compute the size of data to load) EOF GetOptions('kernel=s' => \ (my $kernel), 'initrd=s' => \ (my $initrd), 'size=s' => \$kernel_and_initrd_size, 'isolinux' => \ (my $force_isolinux_mode), ) or die $usage; my ($file) = @ARGV; @ARGV == 1 && -e $file or die $usage; if ($kernel_and_initrd_size) { $kernel and die "give kernel and initrd or size, not both\n"; } else { if ($kernel) { $initrd or die "give both kernel and initrd\n"; $kernel_size = (-s $kernel) / 1024 or die "bad file $kernel: $!\n"; $initrd_size = (-s $initrd) / 1024 or die "bad file $initrd: $!\n"; } $kernel_and_initrd_size = $kernel_size + $initrd_size; } my $err; $err = $magick->Read($file) and die $err; my $bmp_file_name = $file; $bmp_file_name =~ s/\.\w\w\w?$//; $bmp_file_name .= '.bmp'; if ($file ne $bmp_file_name || $magick->Get('colors') > 128) { warn "writing $bmp_file_name\n"; $err = $magick->Quantize(colors => 128, dither => 'False') and die $err; $err = $magick->Write(filename => "bmp3:$bmp_file_name", compression => 'None') and die $err; } read_parameters("$bmp_file_name.parameters"); $isolinux_mode ||= $force_isolinux_mode; my $window_widget = Gtk2::Window->new('toplevel'); $window_widget->signal_connect(destroy => sub { Gtk2->main_quit }); gtkadd($window_widget, gtkpack(Gtk2::VBox->new(0,0), create_image_area())); image_set_file($bmp_file_name); $window_widget->show; $image_area->window->set_cursor(Gtk2::Gdk::Cursor->new('crosshair')); Gtk2->main; save_parameters("$bmp_file_name.parameters");