#!/usr/bin/perl

# DrakFloppy
# $Id$
# 
# Copyright (C) 2001-2003 MandrakeSoft
# Yves Duret
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2, or (at your option)
# any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, 
# MA 02111-1307, USA.


use strict;
use diagnostics;
use lib qw(/usr/lib/libDrakX);

use standalone;     #- warning, standalone must be loaded very first, for 'explanations'

use common;
use my_gtk qw(:helpers);
use ugtk qw(:helpers);
use detect_devices;

#- languages that can't be displayed with gtk1, so we unset translations
#- for them until this tool is ported to gtk2
if ($ENV{LANGUAGE} =~ /\b(ar|he|hi|ta)/) { $ENV{LANGUAGE} = "C" };

require_root_capability();

my $expert_mode = $::expert;
# we have put here the list in order to do $list->clear() when we have to do
my $fixed_font = Gtk::Gdk::Font->fontset_load(N("-misc-Fixed-Medium-r-*-*-*-140-*-*-*-*-*-*,*"));
my $list = new_with_titles Gtk::CList(N("Module name"), N("Size"));

my $window = my_gtk->new('drakfloppy');
unless ($::isEmbedded) {
    $window->{rwindow}->signal_connect(delete_event => sub { my_gtk->exit(0) });
    $window->{rwindow}->set_title(N("drakfloppy"));
    $window->{rwindow}->set_policy(1, 1, 1);
    $window->{rwindow}->border_width(5);
}

### menus definition
# the menus are not shown
# but they provides shiny shortcut like C-q
my @menu_items = ( 
		  { path => N("/_File"), type => '<Branch>' },
		  { path => N("/File/_Quit"), accelerator => N("<control>Q"), callback => sub { my_gtk->exit(0) } },
		 );
ugtk::create_factory_menu($window->{rwindow}, @menu_items) unless $::isEmbedded;

######### menus end

my $global_vbox = new Gtk::VBox();

$global_vbox->pack_start(new Gtk::Label(N("boot disk creation")), 0, 0, 0) unless $::isEmbedded;

######## up part
my $up_vbox  = new Gtk::VBox(0, 0);

# device part
my $dev_hbox = new Gtk::HBox(1, 0);
my $device_combo = new Gtk::Combo();
my $device_button = new Gtk::Button(N("default"));

$device_combo->set_popdown_strings(map { "/dev/" . $_->{device} } detect_devices::floppies());
$device_button->signal_connect(clicked => sub { $device_combo->entry->set_text("/dev/fd0") });

$dev_hbox->pack_start(new Gtk::Label(N("device")), 0, 0, 0);
$dev_hbox->pack_start($device_combo, 0, 0, 0);
$dev_hbox->pack_start($device_button, 0, 0, 0);
$up_vbox->pack_start($dev_hbox, 0, 0, 0);

# kernel part
my $ker_hbox = new Gtk::HBox(1, 0);
my $kernel_combo = new Gtk::Combo();
my $kernel_button = new Gtk::Button(N("default"));
$kernel_combo->disable_activate();
$kernel_combo->set_popdown_strings(do {
    opendir YREP, "/lib/modules" or die N("DrakFloppy Error: %s", $!);
    my @files_modules = grep !/^\.\.?$/, readdir YREP;
    closedir YREP;
    @files_modules;
});
#$kernel_combo->entry->set_text(`uname -r`);
$kernel_combo->entry->signal_connect(changed => sub { change_tree($kernel_combo->entry->get_text()); $list->clear() });
$kernel_button->signal_connect(clicked => sub { $kernel_combo->entry->set_text(chomp_(`uname -r`)); $list->clear() });

$ker_hbox->pack_start(new Gtk::Label(N("kernel version")), 0, 0, 0);
$ker_hbox->pack_start($kernel_combo, 0, 0, 0);
$ker_hbox->pack_start($kernel_button, 0, 0, 0);
$up_vbox->pack_start($ker_hbox, 0, 0, 5);

# vbox part
my $up_frame = new Gtk::Frame(N("General"));
$up_frame->add($up_vbox);
$global_vbox->pack_start($up_frame, 0, 0, 0);

### expert mode
my $expert_main_frame = new Gtk::Frame(N("Expert Area"));
my $expert_dedans = new Gtk::VBox(0, 5);
$expert_dedans->border_width(5);
my $expert_button_frame = new Gtk::Frame(N("mkinitrd optional arguments"));
my $expert_mod_frame = new Gtk::Frame(N("Add a module"));
my $expert_pane = new Gtk::HPaned();
$expert_pane->set_handle_size(10);
$expert_pane->set_gutter_size(8);

my $expert_button = new Gtk::Button("");
$expert_button->signal_connect(clicked => sub {
    $expert_mode = !$expert_mode;
    toggle_expert_button();
});

my $expert_button_vbox = new Gtk::VBox(0, 5);
my $expert_button_hbox = new Gtk::HBox(0, 5);
my $expert_button_hbox2 = new Gtk::HBox(0, 5);
my $force_button = new Gtk::ToggleButton(N("force"));
my $needed_button = new Gtk::ToggleButton(N("if needed"));
my $scsi_button = new Gtk::ToggleButton(N("omit scsi modules"));
my $raid_button = new Gtk::ToggleButton(N("omit raid modules"));
$expert_button_hbox->pack_start($force_button, 0, 0, 0);
$expert_button_hbox->pack_start($raid_button, 0, 0, 0);

$expert_button_hbox2->pack_start($needed_button, 0, 0, 0);
$expert_button_hbox2->pack_start($scsi_button, 0, 0, 0);

$expert_button_vbox->pack_start($expert_button_hbox, 0, 0, 0);
$expert_button_vbox->pack_start($expert_button_hbox2, 0, 0, 0);
$expert_button_frame->add($expert_button_vbox);
$expert_dedans->pack_start($expert_button_frame, 0, 0, 0);
$expert_mod_frame->add($expert_pane);
$expert_dedans->pack_start($expert_mod_frame, 1, 1, 0);
$expert_main_frame->add($expert_dedans);
$global_vbox->pack_start($expert_main_frame, 1, 1, 0);

### the tree

# Create a ScrolledWindow for the tree
my $tree_scrolled_win = new Gtk::ScrolledWindow();
$tree_scrolled_win->set_usize(200, $::isEmbedded ? 0 : 175);
$expert_pane->add1($tree_scrolled_win);
$tree_scrolled_win->set_policy('automatic', 'automatic');

# Create root tree
my $tree = new Gtk::Tree();
my $leaf;
my $root_dir;
$tree_scrolled_win->add_with_viewport($tree);
$tree->set_selection_mode('single');
$tree->set_view_mode('item');

fill_tree($kernel_combo->entry->get_text());

# Create a ScrolledWindow for the list
my $list_scrolled_win = new Gtk::ScrolledWindow(undef, undef);
my $rmmod_button = new Gtk::Button(N("Remove a module"));
my $expert_inside_pane2 = new Gtk::VBox(0, 0);
my $list_selected_row;

$expert_inside_pane2->pack_start($list_scrolled_win, 1, 1, 0);
$expert_inside_pane2->pack_start($rmmod_button, 0, 0, 0);
$expert_pane->add2($expert_inside_pane2);
$list_scrolled_win->set_policy('automatic', 'automatic');
$rmmod_button->signal_connect(clicked => sub { $list->remove($list_selected_row) });

# Create list box
########################################################## from here my $list
$list->signal_connect(select_row => sub { (undef, $list_selected_row) = @_ });
$list_scrolled_win->add($list);
$list->set_column_justification(1, 'right');
$list->set_column_width(0, 200);
$list->set_column_width(1, 50);
$list->set_selection_mode('single');
$list->set_shadow_type('none');
$list->show();

### output
my $output_frame = new Gtk::Frame(N("Output"));
my $output = new Gtk::Text(undef, undef);
my $vscrollbar = new Gtk::VScrollbar($output->vadj);
my $output_hbox = new Gtk::HBox(0, 0);
$output_hbox->border_width(5);
$output_hbox->set_usize(30, 75);
$output_hbox->pack_start($output, 1, 1, 0);
$output_hbox->pack_start($vscrollbar, 0, 0, 0);
$output_frame->add($output_hbox);
$global_vbox->pack_start($output_frame, 1, 10, 0);

### final buttons
my $build_button = new Gtk::Button(N("Build the disk"));
my $cancel_button = new Gtk::Button(N("Cancel"));
my $fin_hbox = new Gtk::HBox(0, 0);
$cancel_button->signal_connect(clicked => sub { my_gtk->exit(0) });
$build_button->signal_connect(clicked => \&build_it);
$fin_hbox->pack_end($cancel_button, 0, 0, 0);
$fin_hbox->pack_end($build_button,  0, 0, 10);
$fin_hbox->pack_end($expert_button, 0, 0, 10);
$global_vbox->pack_start($fin_hbox, 0, 0, 0);

### back to window
$window->{window}->add($global_vbox);

$window->{rwindow}->show_all();
toggle_expert_button();


$window->main;
my_gtk->exit(0);


sub toggle_expert_button {
    if ($expert_mode) {
        $expert_mod_frame->show();
        $expert_button_frame->show();
        $expert_button->child->set(N("Normal Mode"));
    } else {
        $expert_mod_frame->hide();
        $expert_button_frame->hide();
        $expert_button->child->set(N("Expert Mode"));
    }
}

#-------------------------------------------------------------
# tree functions
#-------------------------------------------------------------
### Subroutines

sub fill_tree {
  ($root_dir) = @_;
  $root_dir = "/lib/modules/" . $root_dir;
  # Create root tree item widget
  $leaf = new_with_label Gtk::TreeItem($root_dir);
  $tree->append($leaf);
  $leaf->signal_connect('select', \&select_item, $root_dir);
  $leaf->set_user_data($root_dir);

  # Create the subtree
  if (has_sub_trees($root_dir)) {
    my $subtree = new Gtk::Tree();
    $leaf->set_subtree($subtree);
    $leaf->signal_connect('expand', \&expand_tree, $subtree);
    $leaf->signal_connect('collapse', \&collapse_tree);
    $leaf->expand();
  }
}

sub change_tree {
  $leaf->destroy();
  fill_tree(@_);
  $leaf->show();
}

# Callback for expanding a tree - find subdirectories, files and add them to tree
sub expand_tree {
    my ($item, $subtree) = @_;

    my $path;
    my $item_new;
    my $new_subtree;

    my $dir = $item->get_user_data();

    chdir($dir);

    foreach my $dir_entry (all(".")) {
      if (-d $dir_entry or $dir_entry =~ /\.o(\.gz)?$/) {
	$path = $dir . "/" . $dir_entry;
	$path =~ s|//|/|g;
	$item_new = new_with_label Gtk::TreeItem($dir_entry);
	$item_new->set_user_data($path);
	$item_new->signal_connect('select', \&select_item, $path);
	$subtree->append($item_new);
	$item_new->show();

	if (has_sub_trees($path)) {
	  $new_subtree = new Gtk::Tree();
	  $item_new->set_subtree($new_subtree);
	  $item_new->signal_connect('expand', \&expand_tree, $new_subtree);
	  $item_new->signal_connect('collapse', \&collapse_tree);
	}
      }
    }
    chdir("..");
  }


# Callback for collapsing a tree -- removes the subtree
sub collapse_tree {
    my ($item) = @_;
    my $subtree = new Gtk::Tree();

    $item->remove_subtree();
    $item->set_subtree($subtree);
    $item->signal_connect('expand', \&expand_tree, $subtree);
  }

# Called whenever an item is clicked on the tree widget.
sub select_item  {
  my ($widget, $file) = @_;
  return if -d $file;
  my  $size = (lstat($file))[7];
  my $lr = $list->rows();
  my $i;
  $file =~ s|/lib/modules/.*?/||g;
  for ($i = 0; $i < $lr; $i++) {
    last if $file eq $list->get_text($i, 0);
  }
  print $file, "\n";
  
  $list->append($file, $size) if $i == $lr or $lr == 0;
}

#-------------------------------------------------------------
# the function
#-------------------------------------------------------------
sub build_it {
  my $y;
  my $co = "/sbin/mkbootdisk --noprompt --verbose --device " . $device_combo->entry->get_text();
  if ($expert_mode) {
    $co .= " --mkinitrdargs -f" if $force_button->get_active;
    $co .= " --mkinitrdargs --ifneeded" if $needed_button->get_active;
    $co .= " --mkinitrdargs --omit-scsi-modules" if $scsi_button->get_active;
    $co .= " --mkinitrdargs --omit-raid-modules" if $raid_button->get_active;
    for (my $i = 0; $i < $list->rows(); $i++) {
      $y = $list->get_text($i, 0);
      $y =~ s|.*?/||g;
      $co .= " --mkinitrdargs --with=" . $y; #. "/usr/lib/" . $kernel_combo->entry->get_text() . "/" . $y;
    }
  }
  $co .= " " . $kernel_combo->entry->get_text();
  $co .= " 2>&1 |";
  create_dialog(N("Be sure a media is present for the device %s",  $device_combo->entry->get_text()), 1) or return;
# we test if the media is present
 test:
  my $a = "dd count=1 if=/dev/null of=" . $device_combo->entry->get_text() . " 2>&1";
  my $b = `$a`;
  if ($b =~ /dd/) { create_dialog(N("There is no medium or it is write-protected for device %s.\nPlease insert one.", $device_combo->entry->get_text()), 1) ? goto test : return 0 }
  
  local *STATUS;
  open STATUS, $co or do { create_dialog(N("Unable to fork: %s", $!), 0); return };
  local $_;
  while (<STATUS>) {
      $output->insert($fixed_font, undef, undef, $_);
  }
  close STATUS or create_dialog(N("Unable to properly close mkbootdisk: \n %s \n %s", $!, $?), 0);
  
  return (0);
}

####
# This is put at the end of the file because any translatable string
# appearing after this will not be found by xgettext, and so wont end in
# the pot file...
####

# Test whether a directory has subdirectories
sub has_sub_trees {
    my ($dir) = @_;
    
    foreach my $file (glob_("$dir/*")) {
        return 1 if -d $file or $file =~ /\.o(\.gz)?$/;
    }
    
    return (0);
}