File:  [Local Repository] / processmail / createtable.pl
Revision 3.0: download - view: text, annotated - select for diffs - revision graph
Sun Mar 23 17:50:49 2008 UTC (16 years, 2 months ago) by boris
Branches: MAIN
CVS tags: HEAD
Added Lev's corrections

#!/usr/local/bin/perl
#$Id: createtable.pl,v 3.0 2008/03/23 17:50:49 boris Exp $
#
#
#
# Исходные файлы и параметры
#

use Getopt::Std; 

$usage = 
"Usage: createtable.pl [-d] [-s|-m] [-t answers_file]  [-o results] \n";

require 'parameters.pl';
require 'subroutines.pl';

die $usage unless getopts('t:o:dsm');

$DEBUG=$opt_d;
$SHORT=$opt_s;
$MEDIUM=$opt_m;

my %teams;
my @answers;
my @ratings;
#
# Читаем старые результаты
#
if ($opt_t) {
    die "Cannot open $opt_t\n" unless open(INFILE,$opt_t);
} else {
   *INFILE=*STDIN;
}
    
readhash(\@answers);
close(INFILE);


#
# Открываем файл для записи
#
if ($opt_o) { 
    die "Cannot open $opt_o\n" unless open(OUTFILE,">$opt_o");
    select OUTFILE;
}

# Находим максимальный вопрос
$MAXQUEST=1;
for (my $i=1; $i<=$ROUNDS; $i++) {
    if ($MAXQUEST<$MAXQUEST[$i]) {
	$MAXQUEST=$MAXQUEST[$i];
    }
}


#
# Таблица подводится отдельно по каждому раунду
#
my @nopoint_questions;
for (my $r=1; $r<=$ROUNDS; $r++) {
    if ($DEBUG) {
	print STDERR "Раунд $r\n";
    }
#
#  Читаем ответы в данном раунде
#

    foreach my $file ( @{$FILES[$r]} ) {
	open (INFILE, $file);
	readmail(\%teams);
	collect_answers(\%teams,\@answers);
	close (INFILE);
    }

#
# Дополняем (добавляем к имеющемуся) список вопросов, идущих вне зачёта
# в данном туре.  Для них проставляются плюсы и минусы, но они не входят
# в суммарный результат и рейтинг команд.
#
    foreach my $num ( @{$NOCOUNT[$r]} )
    {
	# Если номер не начинается с минуса, то включить его в список
	# внезачётных, иначе - исключить.  Проверяем регэкспом, а не на
	# "> 0", чтобы можно было использовать и "-0".
    	if ($num =~ /^\d+/) {
	    push @nopoint_questions, $num; 
	} else {
            @nopoint_questions = grep( $_ != -$num, @nopoint_questions );
	}
    }
    # И удаляем возможные дубликаты, чтобы всё было красиво.
    my %seen;
    @nopoint_questions = grep( !$seen{$_}++, @nopoint_questions );

#
# Подсчитываем рейтинги вопросов
#
    rate_questions(\%teams,\@answers,\@ratings,$r);
#
# Вычисляем число ответов и рейтинги команд
#
    find_scores(\%teams,\@answers,\@ratings,$r,0.01,\@nopoint_questions);
#
# Ну а теперь печатаем саму таблицу...
#
#
# Печатаем заголовок
#
    # Столько столбцов до пробела в среднем формате ("++-++ +-+-+ -+--+")
    my $COL=5;

    print "ЗАЧЁТ  $NAME[$r]\n";
    if ( @nopoint_questions ) {
	# Есть незачётные вопросы - печатаем их список.
	print  "Вне зачёта вопрос" . (scalar(@nopoint_questions)>1 ? "ы" : "") 
		. ": " . join(", ", sort {$a<=>$b} @nopoint_questions) . "\n";
    }

    printf "%5s ", "N";
    for ($i=$MINQUEST[$r];$i<=$MAXQUEST[$r];$i++)
    {
	if ($SHORT) {
	    printf("%1d",$i%10);
	} elsif ($MEDIUM) {
            if ($i == int($i/$COL)*$COL || $i==$MAXQUEST[$r]) {
		printf "%-3d", $i;
            } else {
		printf "%s", ".";
            }
	} else {
	    printf "%3d",$i;
	}
    }
    printf "%6s","О";
    printf "%4s","Р";
    printf " КОМАНДА";
    print "\n";
#
# Печатаем команды построчно
#
    foreach $team (sort 
		   {
		       $teams{$b}->{score} <=> $teams{$a}->{score}
		       or
			   $teams{$b}->{rating} <=> $teams{$a}->{rating}
		   } keys %teams
		   )
    {
	printf "%5d ",$teams{$team}->{regnum};
	for ($i=$MINQUEST[$r];$i<=$MAXQUEST[$r];$i++)
	{
	    my $answer = $teams{$team}->{answers}[$i];
	    my $score = $answers[$i]->{$answer}->{score};
	    $score = '-' unless $score;
	    $score = 'X' 
		if ( $score eq '+' && @nopoint_questions && 
		     grep($_==$i, @nopoint_questions)
		   );
	    if ($SHORT) {
		printf "%1s",$score;
	    } elsif ($MEDIUM) {
		printf "%1s",$score;
		print "  " if ($i == int($i/$COL)*$COL || $i==$MAXQUEST[$r]);
	    } else {
		printf "%3s", $score;
	    }
	}
	printf "%6.2f",$teams{$team}->{score};
	printf "%4s",$teams{$team}->{rating};
	print " $team";
	print "\n";
    }
#
# Печатаем рейтинги вопросов - длинный или короткий варианты.
#
    if ($SHORT || $MEDIUM) {
        my $nline = 12; 			# По 12 рейтингов в строке
	print "\nРейтинг\n";
	for ($j=0; $j<$MAXQUEST[$r]/$nline; $j++) {
	    my $max = $j*$nline + $nline;
	    if ($max>$MAXQUEST[$r]) {
		$max=$MAXQUEST[$r];
	    }
	    for ($i=$j*$nline+1; $i<=$max; $i++){
		printf("%3s",$i);
	    }
	    printf "\n";
	    for ($i=$j*$nline+1; $i<=$max; $i++){
		printf("%3s",$ratings[$i]);
	    }
	    printf "\n";
	}
    } else {
	# Длинный вариант - под каждым вопросом.
	printf "%5s ", "Р";
	for ($i=$MINQUEST[$r]; $i<=$MAXQUEST[$r]; $i++)
	{
	    printf "%3s", $ratings[$i];
	}
	print "\n";
    }
    print "\n";
}

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>