Linux ip-172-26-7-228 5.4.0-1103-aws #111~18.04.1-Ubuntu SMP Tue May 23 20:04:10 UTC 2023 x86_64
Your IP : 18.117.71.102
#! /usr/bin/env perl
###############################################################################
# texdoctk - GUI for TeX documentation access
# Copyright (C) 2000-2004 Thomas Ruedas
# Updated in 2010 by Manuel Pegourie-Gonnard
# Trivial non-code updates in 2012 by Karl Berry
# (This program is looking for a maintainer, email tex-live@tug.org.)
#
# This program is provided under the GNU Public License; see the texdoctk
# README for details about requirements, installation, configuration,
# and the full disclaimer.
###############################################################################
my ($version, $date) = ("0.6.1", "2012-12-23");
use strict;
use Getopt::Long;
Getopt::Long::config('bundling');
use File::Basename;
BEGIN {
Win32::SetChildShowWindow(0) if defined &Win32::SetChildShowWindow;
}
my $IsWin32 = ($^O =~ /MSWin32/i);
if ($IsWin32) {
require Win32::API;
# Win32::API->import( qw(
# &something
# ));
}
eval { require Tk; };
if ($@) {
if (-x "/usr/bin/xmessage") {
`xmessage -center -buttons Quit "The program texdoctk needs the package perl-tk, please install it!"`;
} else {
printf STDERR "The program texdoctk needs the package perl-tk, please install it!\n";
}
exit(1);
# that didn't work out, so warn the user and continue with text mode
} else {
Tk->import;
}
# initialization of some internal variables
$|=1;
my $quiet=1;
my $autoview;
my $xfmt_viewer=1,
my @tdcolors;
my %butcol;
my $srchentry;
my $srchflag=0;
my $srchtype; # 0 = database, 1 = texdoc
my $tmpfno;
# system variables
my ($texmfmain,$texmfdist,$texmfdoc,$texmflocal,$texmfhome,
$texdocpath,$distdocpath,$docdocpath,$localdocpath,$homedocpath,
$datadir,
$dvi_viewer,$dvips_conv,$ps_viewer,$pdf_viewer,
$pdfps_conv,$html_viewer,$htmlps_conv,$htmlps_redir,
$txt_viewer,$txtps_conv,$txtps_redir,$print_cmd);
# defaults: unix values
my $qq="'";
my $HomeEnv=$ENV{HOME};
my $TeXDocRcDefaults="texdocrc.defaults";
my $TempDir; $TempDir=$ENV{TMPDIR} or $TempDir="/tmp";
my $CmdConsole="xterm -e ";
if ($IsWin32) {
$qq = "\"";
$HomeEnv = $ENV{'USERPROFILE'};
chomp $HomeEnv;
$TeXDocRcDefaults = "texdocrc-win32.defaults";
$TempDir = &GetTempPath;
$CmdConsole = "cmd /c ";
}
my $myrc="${HomeEnv}/.texdocrc"; # optional personal settings file
my ($line,@dummy);
my @tmpfiles;
# read system-wide defaults from texdocrc.defaults
my $sysrc=`kpsewhich --progname=texdoctk --format=${qq}other text files${qq} ${TeXDocRcDefaults}`;
my $database=`kpsewhich --progname=texdoctk --format=${qq}other text files${qq} texdoctk.dat`;
my $locdatabase=`kpsewhich --progname=texdoctk --format=${qq}other text files${qq} texdoctk-local.dat`;
my $homedatabase=`kpsewhich --progname=texdoctk --format=${qq}other text files${qq} texdoctk-pers.dat`;
chomp $locdatabase;
chomp $homedatabase;
chomp $sysrc;
&readrc($sysrc,1);
@dummy=split("/",$sysrc);
pop @dummy;
$datadir=join('/',@dummy);
# possibly use personal settings instead of defaults
if (-e $myrc) { &readrc($myrc,2); }
# command line options
GetOptions('v'=>\$quiet,'a'=>\$autoview);
my $special=0;
my @button;
$button[17]="Miscellaneous";
my (@packname,@topic,@doc,@keywords,@maxind,%stydoc);
# read database file
my $i=-1;
my $j;
open(DATABASE,"$database") || &fatalmsg("Couldn't open database $database.\n");
while ($line = <DATABASE>) {
LINETYPE: {
($line =~ /^\@/) && do { # category
++$i;
if ($i == $#button) {
pop @button;
push @button,(substr($line,1,-1));
$special=1;
} else {
$button[$i]=substr($line,1,-1);
}
$j=0;
last LINETYPE;
};
($line =~ /^(#|$)/) && do { # comment or empty line
last LINETYPE;
};
# list item
@dummy=split(";",$line);
$packname[$i][$j]=$dummy[0];
$topic[$i][$j]=$dummy[1];
$doc[$i][$j]=$dummy[2];
$keywords[$i][$j]=$dummy[3];
if ($doc[$i][$j] =~ /sty$/ && $keywords[$i][$j]) {
$stydoc{"$doc[$i][$j]"}=substr($keywords[$i][$j],1,1);
unless ($stydoc{"$doc[$i][$j]"} =~ /^\d$/) {
$stydoc{"$doc[$i][$j]"}=0;
}
}
$maxind[$i]=$j;
++$j;
};
}
close(DATABASE);
# Read site-local and user's database(s) if it/they exist(s)
foreach my $dbfile ($locdatabase,$homedatabase) {
if (-e "$dbfile") {
open(DATABASE,"$dbfile") ||
print "Couldn't open additional database $dbfile.\n";
$i=17; # local/user: if no category is specified in database, add to "Misc"
while ($line = <DATABASE>) {
LINETYPE: {
($line =~ /^\@/) && do { # category
$i=&arrindex(@button,substr($line,1,-1));
if ($i >= $#button) {
$i=$#button;
pop @button;
push @button,(substr($line,1,-1));
$special=1;
}
last LINETYPE;
};
($line =~ /^(#|$)/) && do { # comment or empty line
last LINETYPE;
};
# list item
@dummy=split(";",$line);
$j=&arrindex(@{$packname[$i]},$dummy[0]);
if ($j > $#{$packname[$i]}) {
$packname[$i][$j]=$dummy[0];
$topic[$i][$j]=$dummy[1];
$doc[$i][$j]=$dummy[2];
$keywords[$i][$j]=$dummy[3];
if ($doc[$i][$j] =~ /sty$/ && $keywords[$i][$j]) {
$stydoc{"$doc[$i][$j]"}=substr($keywords[$i][$j],1,1);
unless ($stydoc{"$doc[$i][$j]"} =~ /^\d$/) {
$stydoc{"$doc[$i][$j]"}=0;
}
}
$maxind[$i]=$j;
++$j;
}
};
}
close(DATABASE);
$special=1 if ($i == 17 && $j > 0);
}
}
# Tk
# hash table for toplevel windows; used to avoid multiple calls of same window
my %tlwins;
my %buttonlist;
# create frames for main window: commands on top, frame for buttons below
my $main=new MainWindow;
$tlwins{'mainwindow'}{'addr'}=$main;
$main->resizable(0,0);
$main->title("TeX Documentation Browser");
$main->bind('all','<Control-q>'=>\&clean_exit);
$main->bind('all','<Control-m>'=>sub { $main->raise(); });
$main->bind('all','<Control-h>'=>\&helptext);
$main->bind('all','<Control-s>'=>sub { $main->raise(); &mksrch; });
$main->bind('all','<Control-t>'=>\&settings);
my $cmdframe=$main->Frame(-background=>"#ffcc99");
my $buttonframe=$main->Frame;
$cmdframe->pack(-side=>'top',-fill=>'x');
$buttonframe->pack(-side=>'bottom');
# normal cursor
my $defcursor=$main->cget(-cursor);
# make buttons for command frame
my $Qbut=$cmdframe->Button(-text=>'Quit',%butcol,
-command=>\&clean_exit)->pack(-side=>'left');
$tlwins{'mainwindow'}{'buttons'}[0]=$Qbut;
# define common default font for labels and text explicitly
my $deffont=$Qbut->cget(-font);
# ensure readability on high-res screens (suggested by R.Kotucha)
$deffont='Helvetica -16 bold' if &x_resolution > 1200;
$Qbut->configure(-font=>$deffont);
$tlwins{'mainwindow'}{'buttons'}[1]=$cmdframe->Button(
-text=>'Database search',
-font=>$deffont,%butcol,
-command=>[\&mksrch, 0]
)->pack(-side=>'left');
$tlwins{'mainwindow'}{'buttons'}[2]=$cmdframe->Button(
-text=>'File search',
-font=>$deffont,%butcol,
-command=>[\&mksrch, 1]
)->pack(-side=>'left');
$tlwins{'mainwindow'}{'buttons'}[3]=$cmdframe->Button(-text=>'Help/About',
-font=>$deffont,%butcol,
-command=>\&helptext
)->pack(-side=>'right');
$tlwins{'mainwindow'}{'buttons'}[4]=$cmdframe->Button(-text=>'Settings',
-font=>$deffont,%butcol,
-command=>\&settings
)->pack(-side=>'right');
# make buttons for category button frame
my $lbut=0;
my $l;
foreach (@button) { $l=length $_; $lbut=$l if ( $l > $lbut) }
my @catg;
my ($i2,$i3);
my $nbutt=scalar @button;
my $ncols=$nbutt/3-1;
foreach (0..$ncols) {
$i=$_;
$i2=$i+$nbutt/3;
$i3=$i+2*$nbutt/3;
$catg[$i]=$buttonframe->Button(-text=>$button[$i],
-font=>$deffont,%butcol,
-width=>$lbut,
-command=>[\&tpslct,$i,\@dummy])->grid
($catg[$i2]=$buttonframe->Button(-text=>$button[$i2],
-font=>$deffont,%butcol,
-width=>$lbut,
-command=>[\&tpslct,$i2,\@dummy]),
$catg[$i3]=$buttonframe->Button(-text=>$button[$i3],
-font=>$deffont,%butcol,
-width=>$lbut,
-command=>[\&tpslct,$i3,\@dummy]));
}
# disable last button (lower right) if no local specials are found in list
my $ncat;
if ($special == 0) {
$catg[$#button]->configure(-state=>'disabled');
$ncat=$nbutt-1;
} else {
$ncat=$nbutt;
}
for ($i=0,$j=5; $i<@catg; ++$i,++$j) {
$tlwins{'mainwindow'}{'buttons'}[$j]=$catg[$i];
}
MainLoop();
########## SUBROUTINES ########################################################
# toplevel for selecting a topic of a category for viewing or printing
sub tpslct {
my($opt,@srchitems)=@_;
my (@lbitems,@lbdocs,$dspselect,$docselect,$wtitle);
if ($opt >= 0) {
# main window buttons
# see if toplevel window is already there
if (Exists($tlwins{$opt}{'addr'})) {
$tlwins{$opt}{'addr'}->deiconify();
$tlwins{$opt}{'addr'}->raise();
return;
}
for ($j=0; $j <= $maxind[$opt]; ++$j) {
push @lbitems,$topic[$opt][$j];
push @lbdocs,$doc[$opt][$j];
}
$wtitle=$button[$opt];
} else {
# search results
my $spec_wtitle=shift @srchitems;
for ($j=0; $j < $#srchitems; $j+=2) {
my $k=$j+1;
push @lbitems,$topic[$srchitems[$j]][$srchitems[$k]];
push @lbdocs,$doc[$srchitems[$j]][$srchitems[$k]];
}
$wtitle = ($srchtype ? "File" : "Database")
. " search results for $spec_wtitle";
}
# toplevel window of category $opt with two frames
my $tpwin=$main->Toplevel(-title=>$wtitle);
$tlwins{$opt}{'addr'}=$tpwin;
my $tpdsp=$tpwin->Frame(-relief=>'groove')->pack(-side=>'top');
my $tpslc=$tpwin->Frame()->pack(-side=>'top');
# selection frame with listbox and buttons
# label for listbox
my $tplabel=$tpslc->Label(-text=>($srchtype ? "Files" : "Topics"),
-font=>$deffont)->pack(-anchor=>'w',
-side=>'top');
# listbox with optional scrollbar
my $tplist=$tpslc->Scrolled("Listbox",
-font=>$deffont,
-scrollbars=>'osoe',
# -height=>0,
-width=>0,
-selectmode=>'single',
-exportselection=>0,
-cursor=>'hand2')->pack(-side=>'left',
-fill=>'y',
-expand=>1);
$tplist->insert('end',@lbitems); # fill topics into listbox
# buttons frame
my $tpbframe=$tpslc->Frame(-borderwidth=>8);
$tpbframe->pack(-side=>'right');
# make buttons for command frame
my $n_buttons = 0;
$tlwins{$opt}{'buttons'}[$n_buttons++]=
$tpbframe->Button(-text=>'View',
-font=>$deffont,%butcol,
-command=>sub{ &viewslc($docselect,$tpbframe);},
-width=>6)->pack(-side=>'top');
if (! $IsWin32) {
$tlwins{$opt}{'buttons'}[$n_buttons++]=
$tpbframe->Button(-text=>'Print',
-font=>$deffont,%butcol,
-command=>sub{ &prtslc($docselect,$tpbframe);},
-width=>6)->pack(-side=>'top');
}
$tlwins{$opt}{'buttons'}[$n_buttons++]=
$tpbframe->Button(-text=>'Cancel',
-font=>$deffont,%butcol,
-command=>sub{ destroy $tpwin; undef $tlwins{$opt};},
-width=>6)->pack(-side=>'bottom');
# if only 1 item is in the list (most likely in search results), select it
if (scalar @lbitems == 1) {
$tplist->selectionSet(0);
$dspselect=$lbitems[0];
$docselect=$lbdocs[0];
&viewslc($docselect,$tpbframe)
if ($autoview && $wtitle =~ /^Search results/);
}
# display frame
my $dsplabel=$tpdsp->Label(-text=>'Selection:',
-font=>$deffont)->pack(-anchor=>'w',,
-fill=>'x',
-side=>'left');
my $dspslc=$tpdsp->Label(-textvariable=>\$dspselect,
-font=>$deffont,
-borderwidth=>2)->pack(-anchor=>'w',
-fill=>'x',
-side=>'right');
# handle selection; only one selection possible
$tplist->bind('<Button-1>'=>sub{my ($slctind)=$tplist->curselection();
$dspselect=$lbitems[$slctind];
$docselect=$lbdocs[$slctind];});
# key bindings
$tplist->bind('<Double-Button-1>'=>sub{
&viewslc($docselect,$tpbframe);});
$tpwin->bind('<Control-v>'=>sub{
&viewslc($docselect,$tpbframe);});
$tpwin->bind('<Control-p>'=>sub{
&prtslc($docselect,$tpbframe);});
$tpwin->bind('<Control-c>'=>sub{destroy $tpwin; undef $tlwins{$opt};});
$tplist->bind('<Button-3>'=>sub{
if ($docselect) {
&showpath($docselect,$tpbframe);
} else {
&popmsg(2,"No selection;\nuse left mouse button.",$tpbframe);
}});
}
# view document selected in listbox
sub viewslc {
my($slc,$parframe)=@_;
my $viewer;
my $itype=-1;
my $browser=0;
my $styflag=0;
unless (defined $slc) {
&popmsg(2,"No selection was made.",$parframe);
return;
}
$parframe->configure(-cursor=>'watch');
chomp $slc;
my @dummy=split('\.',$slc);
DOC_FORMAT: { # determine document type
($dummy[-1] =~ /dvi/) && do { $viewer=$dvi_viewer; last DOC_FORMAT; };
($dummy[-1] =~ /ps/) && do { $viewer=$ps_viewer; last DOC_FORMAT; };
($dummy[-1] =~ /pdf/) && do { $viewer=$pdf_viewer; last DOC_FORMAT; };
($dummy[-1] =~ /txt/ || $dummy[-1] =~ /faq$/) && do {
$viewer=($txt_viewer eq "TDK_OWN" ||
$txt_viewer =~ /xterm/) ?
$txt_viewer : "${CmdConsole} $txt_viewer";
last DOC_FORMAT;
};
($dummy[-1] =~ /README/i || $dummy[-2] =~ /\/README$/i) && do {
$viewer=($txt_viewer eq "TDK_OWN" ||
$txt_viewer =~ /xterm/) ?
$txt_viewer : "${CmdConsole} $txt_viewer";
last DOC_FORMAT;
};
($dummy[-1] =~ /htm/) && do {
$viewer=$html_viewer;
if ($html_viewer =~ /netscape/i) { # check if Netscape is open
my $lockfile="${HomeEnv}/.netscape/lock";
$browser=1 if (-e $lockfile || -l $lockfile);
} elsif ($html_viewer =~ /mozilla/i) { # check if Mozilla is open
my $mozcheck=
`mozilla -remote 'openFile($texmfmain/doc/index.html)'`;
$browser=1 unless ($?);
}
last DOC_FORMAT;
};
# some packages have no normal documentation but useful info in the .sty
($dummy[-1] =~ /sty$/) && do {
$viewer=($txt_viewer eq "TDK_OWN" ||
$txt_viewer =~ /xterm/) ?
$txt_viewer : "${CmdConsole} $txt_viewer";
$slc="../tex/$slc"; # .sty files are in ${tex,local}docpath/../tex/
$styflag=1;
last DOC_FORMAT;
};
$parframe->configure(-cursor=>$defcursor);
if ($xfmt_viewer != 1) {
&popmsg(2,"$dummy[-1]: not a known document format",$parframe);
return;
}
};
if ($viewer eq "") {
if ($xfmt_viewer == 1) {
&popmsg(-1,'Unknown format; assuming plain text.',$parframe);
$viewer=($txt_viewer eq "TDK_OWN" ||
$txt_viewer =~ /xterm/) ?
$txt_viewer : "${CmdConsole} $txt_viewer";
} else {
&popmsg(2,"$dummy[-1]: no viewer available/specified for this format",
$parframe);
$parframe->configure(-cursor=>$defcursor);
return;
}
}
# build complete path and start viewer if file exists
my $slcdoc;
$slcdoc=&finddoc($slcdoc,$slc,$parframe);
if ($slcdoc ne "") {
# change to doc directory in case there are pictures
my $docpath=substr($slcdoc,0,(rindex($slcdoc,'/',(length $slcdoc))));
chomp $slcdoc;
chdir ($docpath);
# try to get doc out of .sty file
if ($styflag) {
&popmsg(-1,'Trying to extract documentation out of .sty file;
might be cluttered with program comments.',
$parframe);
$slcdoc=&stripsty($slcdoc,substr($slc,7));
}
# for text files use built-in text viewer if set
if ($IsWin32) {
if ($viewer eq "TDK_OWN") {
&ShellExecute('open', $slcdoc);
$parframe->configure(-cursor=>$defcursor);
return;
}
else {
system("$viewer $slcdoc");
}
}
else {
if ($viewer eq "TDK_OWN") {
&textview($slcdoc);
$parframe->configure(-cursor=>$defcursor);
return;
}
if ($quiet == 0) { # show messages
if ($browser == 0) { # normal viewers or new Netscape/Mozilla
system("$viewer $slcdoc &");
} else { # open doc in existing Netscape/Mozilla window
system("$viewer -remote 'openFile($slcdoc)' &");
&popmsg(-1,"Opening document in existing $viewer window.",
$parframe);
}
} else { # viewer messages written to /dev/null instead of terminal
if ($browser == 0) { # normal viewers or new Netscape/Mozilla
system("perl -we 'use strict; \
my \$dump=\"\"; \
open(NULL,\">/dev/null\"); \
open (VIEWQ,\"$viewer $slcdoc |\"); \
while (\$dump=<VIEWQ>) { print NULL \$dump; } \
close(VIEWQ); \
close(NULL);' &");
} else { # open doc in existing window (Netscape or Mozilla)
# not quiet; I didn't get an inline perl call like in
# the other branch working
system("$viewer -remote 'openFile($slcdoc)' &");
&popmsg(-1,"Opening document in existing $viewer window.",
$parframe);
}
}
}
}
$parframe->configure(-cursor=>$defcursor);
}
# Tk widget for text document
sub textview {
my($txtfile)=@_;
# see if toplevel window with this file is already there
if (Exists($tlwins{$txtfile}{'addr'})) {
$tlwins{$txtfile}{'addr'}->deiconify();
$tlwins{$txtfile}{'addr'}->raise();
return;
}
my $tfv_tk=$main->Toplevel(-title=>"Text file viewer");
$tfv_tk->resizable(0,1);
$tlwins{$txtfile}{'addr'}=$tfv_tk;
$tfv_tk->Label(-text=>"File: $txtfile",
-font=>$deffont,
-relief=>'ridge',
-borderwidth=>3)->pack(-side=>'top',
-fill=>'x',
-ipady=>10,
-anchor=>'s');
my $txtbody=$tfv_tk->Scrolled("Text",
-relief=>'flat',
-font=>$deffont,
-height=>20,
-width=>80,
-scrollbars=>"e")->pack(-side=>'top',
-fill=>'y',
-expand=>1);
open(TXTFILE,"$txtfile");
while (<TXTFILE>) { $txtbody->insert('end',$_); }
close(TXTFILE);
$txtbody->configure(-state=>'disabled');
$tlwins{$txtfile}{'buttons'}[0]=
$tfv_tk->Button(-text=>'Close',
-font=>$deffont,%butcol,
-command=>sub{ destroy $tfv_tk;
undef $tlwins{$txtfile};
})->pack(-side=>'bottom',
-fill=>'x');
$tfv_tk->bind('<Control-c>'=>sub{destroy $tfv_tk;
undef $tlwins{$txtfile};
});
}
# print document selected in listbox
sub prtslc {
my($slc,$parframe)=@_;
$tmpfno=&randname;
my $pstmpfile="${TempDir}/texdoc_$tmpfno.ps";
push @tmpfiles,$pstmpfile;
my $status;
chomp $slc;
unless (defined $slc) {
&popmsg(2,"No selection was made.",$parframe);
return;
}
if ($print_cmd eq "") {
&popmsg(2,"No printer specified.",$parframe);
return;
}
# build complete path and check existence of file
my @dummy=split('\.',$slc);
if ($dummy[-1] =~ /sty$/) {
$slc="../tex/$slc"; # .sty files are in ${tex,local,home}docpath/../tex
}
my $slcdoc;
$slcdoc=&finddoc($slcdoc,$slc,$parframe);
return if ($slcdoc eq "");
# change to doc directory in case there are pictures
my $docpath=substr($slcdoc,0,(rindex($slcdoc,'/',(length $slcdoc))));
chdir ($docpath);
DOC_FORMAT: { # determine document type
# convert dvi files to ps before printing
($dummy[-1] =~ /dvi/) && do {
if ($dvips_conv eq "") {
&popmsg(2,"No converter available for dvi->ps conversion.",
$parframe);
return;
}
&popmsg(-1,"Converting dvi to ps for printing and sending file to print...",$parframe);
$status=system("$dvips_conv $slcdoc -o $pstmpfile");
if ($status != 0) {
&popmsg(2,"Error: Conversion dvi->ps failed!",$parframe);
}
last DOC_FORMAT;
};
# PostScript is printed directly
($dummy[-1] =~ /ps/) && do {
&popmsg(-1,"Sending file to $print_cmd...",$parframe);
$pstmpfile=$slcdoc;
last DOC_FORMAT;
};
# convert pdf files to ps before printing
($dummy[-1] =~ /pdf/) && do {
if ($pdfps_conv eq "") {
&popmsg(2,"No converter available for pdf->ps conversion.",
$parframe);
return;
}
&popmsg(-1,"Converting pdf to ps for printing and sending file to print...",$parframe);
system("$pdfps_conv $slcdoc $pstmpfile");
last DOC_FORMAT;
};
# convert html files to ps before printing
($dummy[-1] =~ /htm/) && do {
if ($htmlps_conv eq "") {
&popmsg(2,"No converter available for html->ps conversion.",
$parframe);
return;
}
&popmsg(-1,"Converting html to ps for printing and sending file to print...",$parframe);
my $htmlps_redir_sign=($htmlps_redir == 1) ? ">" :"";
system("$htmlps_conv $slcdoc $htmlps_redir_sign $pstmpfile");
last DOC_FORMAT;
};
# convert txt files to ps before printing
($dummy[-1] =~ /txt/ || $dummy[-1] =~ /faq$/
|| $dummy[-1] =~ /README/i) && do {
if ($txtps_conv eq "") {
&popmsg(2,"No converter available for txt->ps conversion.",
$parframe);
return;
}
&popmsg(-1,"Converting txt to ps for printing and sending file to print...",$parframe);
my $txtps_redir_sign=($txtps_redir == 1) ? ">" :"";
system("$txtps_conv $slcdoc $txtps_redir_sign $pstmpfile");
last DOC_FORMAT;
};
# convert extracted comments from sty files to ps before printing
($dummy[-1] =~ /sty$/) && do {
if ($txtps_conv eq "") {
&popmsg(2,"No converter available for sty(txt)->ps conversion.",
$parframe);
return;
}
&popmsg(-1,"Converting sty(txt) to ps for printing and sending file to print...",$parframe);
# try to get doc out of .sty file
&popmsg(-1,'Trying to extract documentation out of .sty file;
might be cluttered with program comments.',
$parframe);
$slcdoc=&stripsty($slcdoc,substr($slc,7));
my $txtps_redir_sign=($txtps_redir == 1) ? ">" :"";
system("$txtps_conv $slcdoc $txtps_redir_sign $pstmpfile");
last DOC_FORMAT;
};
&popmsg(2,"$dummy[-1]: cannot print document format",$parframe);
return;
};
system("$print_cmd $pstmpfile");
}
# test existence and search a documentation which is not in specified place
sub finddoc {
my($slcdoc,$slc,$parframe)=@_;
my ($dummy,$status);
SLCDOCFIND: foreach ($texdocpath,$distdocpath,$docdocpath) {
$slcdoc="$_/$slc";
if (-e $slcdoc) {
# found where it should be
return $slcdoc;
} else {
# see if the documentation file is in the local doc tree...
my @srchslcdoc=($slcdoc);
if ($localdocpath) {
push @srchslcdoc,"$localdocpath/$slc";
return $srchslcdoc[-1] if (-e $srchslcdoc[-1]);
}
# ... or the user's doc tree...
if ($homedocpath) {
push @srchslcdoc,"$homedocpath/$slc";
return $srchslcdoc[-1] if (-e $srchslcdoc[-1]);
}
# ... otherwise proceed with searching
$status=-1;
# see if the documentation file is compressed
CMPREND: foreach my $cmprtype ("gz","bz2","zip") {
foreach (@srchslcdoc) {
$dummy="$_.$cmprtype";
if (-e $dummy) { # yes, it is compressed
$tmpfno=&randname;
my $rawname=basename($slc,"");
$slcdoc="${TempDir}/texdoc_$tmpfno$rawname";
push @tmpfiles,$slcdoc;
my ($dcmp,$dcmp_opt);
COMPRESS: { # determine compression type
($cmprtype eq "gz") && do {
$dcmp="gzip";
$dcmp_opt="-cd";
last COMPRESS;
};
($cmprtype eq "bz2") && do {
$dcmp="bzip2";
$dcmp_opt="-cd";
last COMPRESS;
};
($cmprtype eq "zip") && do {
$dcmp="zip";
$dcmp_opt="-p";
last COMPRESS;
};
};
# try to decompress
$status=system("$dcmp $dcmp_opt $dummy > $slcdoc");
if ($status != 0) { # failure
&popmsg(2,
"$dcmp: Couldn't decompress file, cancelling.",
$parframe);
$slcdoc="";
return $slcdoc;
}
# last CMPREND; does not work, what we need is:
return $slcdoc;
}
}
}
}
}
if ($status != 0) {
$parframe->configure(-cursor=>'watch');
# try to find it elsewhere with kpsewhich
my $rawname=basename($slc,"");
$slcdoc=($slcdoc =~ /\.sty$/) ?
`kpsewhich $rawname` :
`kpsewhich --format=${qq}TeX system documentation${qq} $rawname`;
if ($slcdoc eq "") { # nothing found; cancel
&popmsg(2,"$rawname not found, cancelling.",$parframe);
$slcdoc="";
}
$parframe->configure(-cursor=>$defcursor);
}
return $slcdoc;
}
# determine existence and show paths of files (Button-3)
sub showpath {
my ($docselect,$tpbframe)=@_;
my $found=0;
if ($docselect =~ /\.sty$/) {
STYPATH: foreach my $fullpath ("$texmfmain/tex/$docselect",
"$texmfdist/tex/$docselect",
"$texmflocal/tex/$docselect",
"$texmfhome/tex/$docselect") {
if (-e $fullpath) {
&popmsg(0,"Selected file:\n$fullpath",$tpbframe);
$found=1;
last STYPATH;
}
}
if (!$found) {
&popmsg(2,"Selected file:\n$docselect\ndoes not exist.",$tpbframe) }
} else {
DOCPATH: foreach my $fullpath ("$texdocpath/$docselect",
"$distdocpath/$docselect",
"$docdocpath/$docselect",
"$localdocpath/$docselect",
"$homedocpath/$docselect") {
foreach ('','.gz','.bz2','.zip') {
my $fullpathext="$fullpath$_";
if (-e $fullpathext) {
&popmsg(0,"Selected file:\n$fullpathext",$tpbframe);
$found=1;
last DOCPATH;
}
}
}
if (!$found) {
&popmsg(2,"Selected file:\n$docselect\ndoes not exist (neither in normal form nor compressed).",$tpbframe) }
}
}
# make or destroy search entry widget
sub mksrch {
my ($type) = @_;
if ($srchflag == 1) {
if ($type == $srchtype) {
destroy $srchentry;
$srchflag=0;
} else {
$srchtype = $type;
}
} else {
$srchtype = $type;
# get the search string
$srchentry=$cmdframe->Entry(-cursor=>'xterm',
-font=>$deffont,
-width=>20,
-takefocus=>1)->pack(-side=>'left');
$srchentry->focus();
$srchentry->bind('<Return>'=>[\&srchstr, $srchentry]);
$srchflag=1;
# key binding
$srchentry->bind('<Control-c>'=>sub{destroy $srchentry; $srchflag=0;});
$srchentry->bind('<Escape>'=>sub{destroy $srchentry; $srchflag=0;});
}
}
# call search routine and display results
sub srchstr {
my $string;
my @reslist;
$string=$srchentry->get();
# search
$main->Busy(-recurse => 1);
if ($srchtype == 0) {
@reslist = &srchdb($string);
} else {
@reslist = &srchtd($string);
}
$main->Unbusy();
# destroy entry widget and show results
destroy $srchentry;
$srchflag=0;
if (scalar @reslist == 0) {
&popmsg(0, ($srchtype == 0 ? 'Database' : 'File').
" search for $string: no matches found.\n".
"You may want to try a ".
($srchtype == 0 ? 'File' : 'Databse') . " search.\n\n".
"If nothing else works, CTAN offers an online search form:\n".
"http://ctan.org/search.html\n",
$cmdframe);
} else {
&tpslct(-1, $string, @reslist);
}
}
# search using texdoc
sub srchtd {
my ($string) = @_;
my @res;
my $tdout = `texdoc -M -l $string`;
if ($?) {
&popmsg(0,"texdoc failed, sorry: $!\n".
"Please report this problem on the texlive mailing list.\n");
return;
} else {
my $i = 0;
for my $line (split /\n/, $tdout) {
my @fields = split /\t/, $line;
$doc[$ncat][$i] = $fields[2];
$topic[$ncat][$i] = basename($fields[2]);
$packname[$ncat][$i] = "no package name";
push @res, ($ncat, $i++);
}
return @res;
}
}
# search a string in @packname, @topic and @keywords
sub srchdb {
my ($string) = @_;
my (@results, @reslist);
if ($string) {
for ($i=0; $i<$ncat; ++$i) {
for ($j=0; $j<$maxind[$i]+1; ++$j) {
SRCH: {
($topic[$i][$j] =~ /$string/i) && do {
push @results,($i,$j);
last SRCH;
};
(defined $keywords[$i][$j]) && do {
if ($keywords[$i][$j] =~ /$string/i) {
push @results,($i,$j);
last SRCH;
}
};
($packname[$i][$j] =~ /$string/i) && do {
push @results,($i,$j);
last SRCH;
};
};
}
}
} else { # return full database
$string='all database entries';
for ($i=0; $i<$ncat; ++$i) {
for ($j=0; $j<$maxind[$i]+1; ++$j) { push @results,($i,$j); }
}
}
unless (scalar @results == 0) {
# cancel multiple entries
my ($omit,$pack1,$pack2);
@reslist=($results[0],$results[1]);
for ($i=0; $i<$#results; $i+=2) {
$omit=0;
$pack1=$packname[$results[$i]][$results[$i+1]];
for ($j=0; $j<$#reslist; $j+=2) {
$pack2=$packname[$reslist[$j]][$reslist[$j+1]];
if ($pack1 eq $pack2) { $omit=1; last; }
}
if ($omit == 0) { push @reslist,($results[$i],$results[$i+1]); }
}
}
return @reslist;
}
# extract documentation of .sty files; a flag in the @keywords array shows
# where the doc is located:
# -0-: no specific place; -1-: at end, behind \endinput; -2-: at beginning,
# terminated by %%%%%%; -3-: at beginning, terminated with blank line
sub stripsty {
my ($slcdoc,$slc)=@_;
my @stydoc;
$tmpfno=&randname;
my $tmpslcdoc="${TempDir}/texdoc_$tmpfno.txt";
push @tmpfiles,$tmpslcdoc;
open (STY,"$slcdoc");
LOC: {
($stydoc{$slc} == 0) && do { # no specific place, suck in everything :-(
while ($line = <STY>) { push @stydoc,$line if ($line =~ /^%/); }
last LOC;
};
($stydoc{$slc} == 1) && do { # at end, behind \endinput
while ($line = <STY>) {
last if ($line =~ /^\s*\\endinput/ && !($line =~ /^%/));
}
while ($line = <STY>) { push @stydoc,$line; }
last LOC;
};
($stydoc{$slc} == 2) && do { # up to a certain %%%%%%%, hopefully the 1st
for (my $i=0; $i<9; ++$i) { $stydoc[$i]=<STY>; } # after the 8th line
while ($line = <STY>) {
last if ($line =~ /^%{4,}/);
push @stydoc,$line;
}
last LOC;
};
($stydoc{$slc} == 3) && do { # up to 1st blank/apparently empty line
while ($line = <STY>) {
last if ($line =~ /^\s*$/);
push @stydoc,$line;
}
last LOC;
};
};
close(STY);
open(TMPSTY,">$tmpslcdoc");
foreach (@stydoc) { print TMPSTY $_; }
close(TMPSTY);
return $tmpslcdoc;
}
# pop up a note or an error or warning message toplevel window
sub popmsg {
my($level,$msg,$parframe)=@_;
my $degree;
LEVEL: {
($level == -1) && do { $degree="Info"; last LEVEL; };
($level == 0) && do { $degree="Note"; last LEVEL; };
($level == 1) && do {
return if ($quiet == 1);
$degree="Warning";
last LEVEL
};
($level == 2) && do { $degree="Error"; last LEVEL; };
};
my $msgwin=$parframe->Toplevel(-title=>$degree);
my $lbl=$msgwin->Label(-text=>uc($degree),
-font=>$deffont)->pack(-side=>'top', -fill=>'x');
# get size of message text
my @dummy=split("\n",$msg);
my $nline=scalar @dummy;
my $msgwidth=0;
my $lline;
foreach (@dummy) {
$lline=length $_;
if ($lline > $msgwidth) { $msgwidth=$lline; }
}
my $message=$msgwin->Text(-relief=>'flat',
-font=>$deffont,
-height=>$nline,
-width=>$msgwidth)->pack(-side=>'top');
$message->insert('end',$msg);
$message->configure(-state=>'disabled');
if ($level < 0) { # transient message window
$msgwin->after(3000,sub{destroy $msgwin});
} else { # persistent message window
$msgwin->Button(-text=>'Close',
-font=>$deffont,%butcol,
-command=>sub{destroy $msgwin})->pack(-side=>'bottom',
-fill=>'x');
$msgwin->bind('<Control-c>'=>sub{destroy $msgwin});
}
}
# show/change settings (don't show this to an expert :-/ )
sub settings {
my ($homedocpath_tmp,$quiet_tmp,$autoview_tmp,$xfmt_viewer_tmp,
$dvi_viewer_tmp,$dvips_conv_tmp,
$ps_viewer_tmp,$pdf_viewer_tmp,$pdfps_conv_tmp,
$html_viewer_tmp,$htmlps_conv_tmp,$htmlps_redir_tmp,
$txt_viewer_tmp,$txtps_conv_tmp,$txtps_redir_tmp,$print_cmd_tmp)=
($homedocpath,$quiet,$autoview,$xfmt_viewer,
$dvi_viewer,$dvips_conv,$ps_viewer,
$pdf_viewer,$pdfps_conv,$html_viewer,
$htmlps_conv,$htmlps_redir,$txt_viewer,
$txtps_conv,$txtps_redir,$print_cmd);
my @tdcolors_tmp=@tdcolors;
# see if toplevel window is already there
if (Exists($tlwins{"setmenu"}{'addr'})) {
$tlwins{"setmenu"}{'addr'}->deiconify();
$tlwins{"setmenu"}{'addr'}->raise();
return;
}
my $setmenu=$main->Toplevel(-title=>'Settings');
$setmenu->resizable(0,0);
$tlwins{"setmenu"}{'addr'}=$setmenu;
# documentation path
my $datlabel;
$datlabel="Database file(s) used: teTeX";
$datlabel.=", local database" if (-e $locdatabase);
$datlabel.=", user database" if (-e $homedatabase);
my $docframe=$setmenu->Frame(-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-expand=>1);
my $docframestring="\nDistribution documentation root path(s):\n$texdocpath";
$docframestring.=", $distdocpath" if (-e $distdocpath);
$docframestring.=", $docdocpath" if (length $docdocpath);
if ($localdocpath && $localdocpath ne $texdocpath) {
$docframestring.="\nLocal documentation root path: $localdocpath\n";
}
$docframe->Label(-text=>"$docframestring",
-font=>$deffont)->pack(-side=>'top',
-anchor=>'w',
-fill=>'x',
-expand=>1);
if (defined $texmfhome && -d $texmfhome) {
$homedocpath_tmp=basename($homedocpath_tmp,"");
my $usrdocframe=$docframe->Frame(-relief=>'flat')->pack(-side=>'top',
-fill=>'x');
my $usrdoclabel=$usrdocframe->Label(-text=>"User's documentation root path: $texmfhome/",
-font=>$deffont)->pack(-side=>'left');
my $usrdocentry=$usrdocframe->Entry(-textvariable=>\$homedocpath_tmp,
-font=>$deffont)->pack(-side=>'left');
}
$docframe->Label(-text=>"$datlabel.",
-font=>$deffont)->pack(-side=>'top',
-anchor=>'w',
-ipady=>2,
-fill=>'x',
-expand=>1);
# General viewer behaviour
my $genvframe=$setmenu->Frame(-label=>'General viewer behaviour',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>6,
-expand=>1);
my $genvframe1=$genvframe->Frame()->pack(-side=>'left');
my $genvsub1=$genvframe1->Frame()->pack(-side=>'top',
-anchor=>'w',
-ipady=>3);
my $genvecbut=$genvsub1->
Checkbutton(-variable=>\$quiet_tmp)->pack(-side=>'left',
-anchor=>'sw');
my $genvelabel=$genvsub1->Label(-text=>'Suppress error messages',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $genvsub2=$genvframe1->Frame()->pack(-side=>'top',
-anchor=>'w');
my $genvscbut=$genvsub2->
Checkbutton(-variable=>\$autoview_tmp)->pack(-side=>'left',
-anchor=>'sw');
my $genvslabel=$genvsub2->Label(-text=>'Autostart viewer for one-item listboxes',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $genvsub3=$genvframe1->Frame()->pack(-side=>'top',
-anchor=>'w');
my $genvxcbut=$genvsub3->
Checkbutton(-variable=>\$xfmt_viewer_tmp)->pack(-side=>'left',
-anchor=>'sw');
my $genvxlabel=$genvsub3->Label(-text=>'Use text viewer for unknown file format',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
# colours of GUI
my @tdcolors_d=@tdcolors_tmp;
$tlwins{'setmenu'}{'buttons'}[0]=
$genvframe->Button(-text=>'Change viewer colours',
-command=>sub {
@tdcolors_tmp=&popcolor($setmenu,@tdcolors_tmp);
@tdcolors_d=@tdcolors_tmp;
},
-font=>$deffont,%butcol)->pack(-side=>'right');
# DVI
my $dviframe=$setmenu->Frame(-label=>'DVI',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>6,
-expand=>1);
my $dvisub1=$dviframe->Frame()->pack(-side=>'top',
-anchor=>'w');
my $dvisub2=$dviframe->Frame()->pack(-side=>'top',
-anchor=>'w',
-ipady=>6);
my $dvivlabel=$dvisub1->Label(-text=>'Viewer command',
-anchor=>'w',
-font=>$deffont)->pack(-side=>'left');
my $dviventry=$dvisub1->Entry(-textvariable=>\$dvi_viewer_tmp,
-font=>$deffont)->pack(-side=>'left');
my $dviclabel=$dvisub2->Label(-text=>'DVI->PS converter command',
-anchor=>'w',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $dvilbwidth=length 'DVI->PS converter command';
$dvivlabel->configure(-width=>1.1*$dvilbwidth);
$dviclabel->configure(-width=>1.1*$dvilbwidth);
my $dvicentry=$dvisub2->Entry(-textvariable=>\$dvips_conv_tmp,
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
# PostScript
my $psframe=$setmenu->Frame(-label=>'PostScript',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>10,
-expand=>1);
my $psvlabel=$psframe->Label(-text=>'Viewer command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'w');
my $psventry=$psframe->Entry(-textvariable=>\$ps_viewer_tmp,
-font=>$deffont)->pack(-side=>'left');
# PDF
my $pdfframe=$setmenu->Frame(-label=>'PDF',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>6,
-expand=>1);
my $pdfsub1=$pdfframe->Frame()->pack(-side=>'top',
-anchor=>'w');
my $pdfsub2=$pdfframe->Frame()->pack(-side=>'top',
-anchor=>'w',
-ipady=>6);
my $pdfvlabel=$pdfsub1->Label(-text=>'Viewer command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'w');
my $pdfventry=$pdfsub1->Entry(-textvariable=>\$pdf_viewer_tmp,
-font=>$deffont)->pack(-side=>'left');
my $pdfclabel=$pdfsub2->Label(-text=>'PDF->PS converter command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $pdfcentry=$pdfsub2->Entry(-textvariable=>\$pdfps_conv_tmp,
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
# HTML
my $htmlframe=$setmenu->Frame(-label=>'HTML',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>6,
-expand=>1);
my $htmlsub1=$htmlframe->Frame()->pack(-side=>'top',
-anchor=>'w');
my $htmlsub2=$htmlframe->Frame()->pack(-side=>'top',
-anchor=>'w',
-ipady=>6);
my $htmlvlabel=$htmlsub1->Label(-text=>'Viewer command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'w');
my $htmlventry=$htmlsub1->Entry(-textvariable=>\$html_viewer_tmp,
-font=>$deffont)->pack(-side=>'left');
my $htmlclabel=$htmlsub2->Label(-text=>'HTML->PS converter command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $htmlcentry=$htmlsub2->Entry(-textvariable=>\$htmlps_conv_tmp,
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $htmlrlabel=$htmlsub2->Label(-text=>'Output redirect needed',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $htmlrcbut=$htmlsub2->
Checkbutton(-variable=>\$htmlps_redir_tmp)->pack(-side=>'left',
-anchor=>'sw');
# plain text
my $txtframe=$setmenu->Frame(-label=>'Plain text',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>6,
-expand=>1);
my $txtsub1=$txtframe->Frame()->pack(-side=>'top',
-anchor=>'w');
my $txtsub2=$txtframe->Frame()->pack(-side=>'top',
-anchor=>'w',
-ipady=>6);
my $txtvlabel=$txtsub1->Label(-text=>'Viewer command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'w');
my $txtventry=$txtsub1->Entry(-textvariable=>\$txt_viewer_tmp,
-font=>$deffont)->pack(-side=>'left');
my $txt_view_flag;
if ($txt_viewer_tmp eq "TDK_OWN") {
$txt_view_flag=1;
$txtventry->configure(-state=>'disabled');
} else {
$txt_view_flag=0;
$txtventry->configure(-state=>'normal');
}
my $txtvilabel=$txtsub1->Label(-text=>"\tUse texdoctk\'s own viewer",
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $txtvcbut=$txtsub1->
Checkbutton(-variable=>\$txt_view_flag,
-command=>sub {
if ($txt_view_flag == 1) {
$txt_viewer_tmp="TDK_OWN";
$txtventry->configure(-state=>'disabled');
} else {
$txt_viewer_tmp=$txt_viewer;
$txtventry->configure(-state=>'normal');
$txtventry->focus();
}
})->pack(-side=>'left',
-anchor=>'sw');
my $txtclabel=$txtsub2->Label(-text=>'Text->PS converter command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $txtcentry=$txtsub2->Entry(-textvariable=>\$txtps_conv_tmp,
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $txtrlabel=$txtsub2->Label(-text=>'Output redirect needed',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'sw');
my $txtrcbut=$txtsub2->
Checkbutton(-variable=>\$txtps_redir_tmp)->pack(-side=>'left',
-anchor=>'sw');
# printer
my $prtframe=$setmenu->Frame(-label=>'Printer',
-relief=>'groove',
-borderwidth=>2)->pack(-side=>'top',
-fill=>'x',
-ipady=>10,
-expand=>1);
my $prtclabel=$prtframe->Label(-text=>'Command',
-font=>$deffont)->pack(-side=>'left',
-anchor=>'w');
my $prtcentry=$prtframe->Entry(-textvariable=>\$print_cmd_tmp,
-font=>$deffont)->pack(-side=>'left');
# buttons and key bindings
my $setbfr=$setmenu->Frame()->pack(-side=>'bottom');
$tlwins{'setmenu'}{'buttons'}[1]=
$setbfr->Button(-text=>'OK',
-font=>$deffont,%butcol,
-command=>sub{
$homedocpath="$texmfhome/$homedocpath_tmp";
($quiet,$autoview,$xfmt_viewer,
$dvi_viewer,$dvips_conv,
$ps_viewer,$pdf_viewer,
$pdfps_conv,$html_viewer,
$htmlps_conv,$htmlps_redir,
$txt_viewer,$txtps_conv,$txtps_redir,$print_cmd)=
($quiet_tmp,$autoview_tmp,$xfmt_viewer_tmp,
$dvi_viewer_tmp,$dvips_conv_tmp,
$ps_viewer_tmp,
$pdf_viewer_tmp,$pdfps_conv_tmp,
$html_viewer_tmp,
$htmlps_conv_tmp,$htmlps_redir_tmp,
$txt_viewer_tmp,$txtps_conv_tmp,
$txtps_redir_tmp,$print_cmd_tmp);
@tdcolors=@tdcolors_tmp;
%butcol=('-background'=>$tdcolors[0],
'-foreground'=>$tdcolors[1],
'-activebackground'=>$tdcolors[2],
'-activeforeground'=>$tdcolors[3]);
foreach my $tlkey (keys %tlwins) {
foreach (@{$tlwins{$tlkey}{'buttons'}}) {
$_->configure(%butcol);
}
}
if ($txt_viewer eq "" &&
defined $ENV{PAGER}) {
$txt_viewer=$ENV{PAGER};
}
destroy $setmenu},
-width=>6)->pack(-side=>'left',
-padx=>10,
-pady=>10);
$tlwins{'setmenu'}{'buttons'}[2]=
$setbfr->Button(-text=>'Save',
-font=>$deffont,%butcol,
-command=>sub{
$setmenu->configure(-cursor=>'watch');
&popmsg(-1,"Writing settings to $myrc",$setmenu);
my $tdcolors_tmp=join(' ',@tdcolors_tmp);
&writerc($homedocpath_tmp,
$quiet_tmp,$autoview_tmp,$xfmt_viewer_tmp,
$tdcolors_tmp,
$dvi_viewer_tmp,$dvips_conv_tmp,
$ps_viewer_tmp,
$pdf_viewer_tmp,$pdfps_conv_tmp,
$html_viewer_tmp,
$htmlps_conv_tmp,$htmlps_redir_tmp,
$txt_viewer_tmp,$txtps_conv_tmp,
$txtps_redir_tmp,$print_cmd_tmp,$setmenu);
$setmenu->configure(-cursor=>$defcursor);
})->pack(-side=>'left',
-padx=>10);
$tlwins{'setmenu'}{'buttons'}[3]=
$setbfr->Button(-text=>'Cancel',
-font=>$deffont,%butcol,
-command=>sub{destroy $setmenu;
undef $tlwins{"setmenu"};
})->pack(-side=>'left',
-padx=>10);
$setmenu->bind('<Control-c>'=>sub{destroy $setmenu;
undef $tlwins{"setmenu"};});
}
# dialog toplevel for color settings
sub popcolor {
my ($parframe,@entryvar)=@_;
# see if toplevel window is already there
if (Exists($tlwins{'chngcol'}{'addr'})) {
$tlwins{'chngcol'}{'addr'}->deiconify();
$tlwins{'chngcol'}{'addr'}->raise();
return;
}
my $dlgwin=$parframe->Toplevel(-title=>'Change colours');
$dlgwin->resizable(0,0);
$tlwins{'chngcol'}{'addr'}=$dlgwin;
# entry widgets for default/active back/foreground
$dlgwin->Label(-text=>'Enter colours as names or RGB #xxxxxx values',
-font=>$deffont)->pack(-side=>'top');
my $dlgdef=$dlgwin->Frame()->pack(-side=>'top');
my $dlgact=$dlgwin->Frame()->pack(-side=>'top');
my $dlgdef1=$dlgdef->Frame()->pack(-side=>'left');
my $dlgtfr0=$dlgdef1->Frame()->pack(-side=>'top');
my $dlgtfr1=$dlgdef1->Frame()->pack(-side=>'top');
my $dlgact1=$dlgact->Frame()->pack(-side=>'left');
my $dlgtfr2=$dlgact1->Frame()->pack(-side=>'top');
my $dlgtfr3=$dlgact1->Frame()->pack(-side=>'top');
$dlgtfr0->Label(-text=>"Default background",
-anchor=>'w',
-width=>18,
-font=>$deffont)->pack(-side=>'left');
$dlgtfr0->Entry(-textvariable=>\$entryvar[0],
-font=>$deffont)->pack(-side=>'left');
$dlgtfr1->Label(-text=>"Default foreground",
-anchor=>'w',
-width=>18,
-font=>$deffont)->pack(-side=>'left');
$dlgtfr1->Entry(-textvariable=>\$entryvar[1],
-font=>$deffont)->pack(-side=>'left');
my $deflabel=$dlgdef->Label(-text=>'Default',
-relief=>'ridge',
-width=>10,
-height=>2)->pack(-side=>'right',
-padx=>7);
$dlgtfr2->Label(-text=>"Active background",
-anchor=>'w',
-width=>18,
-font=>$deffont)->pack(-side=>'left');
$dlgtfr2->Entry(-textvariable=>\$entryvar[2],
-font=>$deffont)->pack(-side=>'left');
$dlgtfr3->Label(-text=>"Active foreground",
-anchor=>'w',
-width=>18,
-font=>$deffont)->pack(-side=>'left');
$dlgtfr3->Entry(-textvariable=>\$entryvar[3],
-font=>$deffont)->pack(-side=>'left');
my $actlabel=$dlgact->Label(-text=>'Active',
-background=>$entryvar[2],
-foreground=>$entryvar[3],
-relief=>'ridge',
-width=>10,
-height=>2)->pack(-side=>'right',
-padx=>7);
# buttons
my $setbfr=$dlgwin->Frame()->pack(-side=>'bottom');
my $chngflag=0;
$tlwins{'chngcol'}{'buttons'}[0]=
$setbfr->Button(-text=>'OK',
-font=>$deffont,%butcol,
-command=>sub {
foreach (@entryvar) { # RGB form with leading #
$_="#$_" if (/^[A-Fa-f0-9]{6}$/);
}
$chngflag=1 if (&colcheck($dlgwin,@entryvar));
},
-width=>6)->pack(-side=>'left',
-padx=>5,
-pady=>10);
$tlwins{'chngcol'}{'buttons'}[1]=
$setbfr->Button(-text=>'Preview',
-font=>$deffont,%butcol,
-command=>sub {
foreach (@entryvar) { # RGB form with leading #
$_="#$_" if (/^[A-Fa-f0-9]{6}$/);
}
if (&colcheck($dlgwin,@entryvar)) {
$deflabel->
configure(-background=>$entryvar[0],
-foreground=>$entryvar[1]);
$actlabel->
configure(-background=>$entryvar[2],
-foreground=>$entryvar[3]);
}
},
-width=>6)->pack(-side=>'left',
-padx=>5,
-pady=>10);
$tlwins{'chngcol'}{'buttons'}[2]=
$setbfr->Button(-text=>'Cancel',
-font=>$deffont,%butcol,
-command=>sub{destroy $dlgwin;
undef $tlwins{'chngcol'};
})->pack(-side=>'right',
-padx=>5,
-pady=>10);
$dlgwin->bind('<Control-c>'=>sub{destroy $dlgwin;
undef $tlwins{'chngcol'}; });
$dlgwin->waitVariable(\$chngflag);
destroy $dlgwin;
undef $tlwins{'chngcol'};
return @entryvar;
}
# check colours set in popcolor; relies on the existence of X11's showrgb
sub colcheck {
my ($dlgwin,@entryvar)=@_;
my $ecnt=0;
foreach (@entryvar) { # all colours defined?
last unless ($_);
++$ecnt;
}
&popmsg(2,'Some colours undefined.',$dlgwin) if ($ecnt != 4);
my $defcol=0;
if (`which showrgb`) { # test for proper name
my @rgb=`showrgb`;
my @norgb;
foreach my $colour (@entryvar) {
if ($colour =~ /^#[A-Fa-f0-9]{6}$/ || grep /\b$colour\b/i,@rgb) {
++$defcol;
} else {
push @norgb,$colour;
}
}
if ($defcol != 4) {
my $norgb=join("\n",@norgb);
&popmsg(2,"Some colours invalid:\n$norgb",$dlgwin);
}
} else { $defcol=4; }
return ($ecnt == 4 && $defcol == 4) ? 1 : 0;
}
# write user's ~/.texdocrc
sub writerc {
use File::Copy qw/cp mv/;
my ($homedocpath_tmp,$quiet_tmp,$autoview_tmp,$xfmt_viewer_tmp,
$tdcolors_tmp,
$dvi_viewer_tmp,$dvips_conv_tmp,
$ps_viewer_tmp,
$pdf_viewer_tmp,$pdfps_conv_tmp,
$html_viewer_tmp,
$htmlps_conv_tmp,$htmlps_redir_tmp,
$txt_viewer_tmp,$txtps_conv_tmp,
$txtps_redir_tmp,$print_cmd_tmp,$parframe)=@_;
my $err=0;
mv $myrc,"$myrc.save" if (-e $myrc);
open(MYRC,">$myrc");
print MYRC "# user's .texdocrc, generated by program\n
# root of user's doc directory (from \$HOMETEXMF)
# can be left empty if \$TEXMFHOME is undefined
HOMEDOCPATH=$homedocpath_tmp
# general viewer behaviour (y or yes to make active)
QUIET=$quiet_tmp
AUTOVIEW=$autoview_tmp
XFMT_VIEWER=$xfmt_viewer_tmp
# button colours: default and active back-/foreground
# names and RGB values are both allowed; use the form #xxxxxx for RGB
TDCOLORS=$tdcolors_tmp
# the following variables contain whole commands, i.e. options can be given
# DVI settings
DVI_VIEWER=$dvi_viewer_tmp
DVIPS_CONV=$dvips_conv_tmp
# PostScript handling
PS_VIEWER=$ps_viewer_tmp
# PDF handling
PDF_VIEWER=$pdf_viewer_tmp
PDFPS_CONV=$pdfps_conv_tmp
# HTML handling
HTML_VIEWER=$html_viewer_tmp
HTMLPS_CONV=$htmlps_conv_tmp
# HTMLPS_REDIR must be on, if the html->ps converter normally writes its output
# to stdout instead of a file (e.g. as html2ps) (y or yes to make active)
HTMLPS_REDIR=$htmlps_redir_tmp
# Plain text handling
# TDK_OWN is texdoctk's own text file viewer
TXT_VIEWER=$txt_viewer_tmp
TXTPS_CONV=$txtps_conv_tmp
# TXTPS_REDIR must be on, if the txt->ps converter normally writes its output
# to stdout instead of a file (e.g. as a2ps) (y or yes to make active)
TXTPS_REDIR=$txtps_redir_tmp
# printer
PRINT_CMD=$print_cmd_tmp
\n# end of config\n";
close(MYRC);
}
# help window (toplevel)
sub helptext {
# see if toplevel window is already there
if (Exists($tlwins{"helptext"}{'addr'})) {
$tlwins{"helptext"}{'addr'}->deiconify();
$tlwins{"helptext"}{'addr'}->raise();
return;
}
my $help=$main->Toplevel(-title=>'Help/About');
$help->resizable(0,1);
$tlwins{'helptext'}{'addr'}=$help;
$help->Label(-text=>"texdoctk
TeX documentation browser
v$version ($date)",
-font=>$deffont)->pack(-side=>'top',
-ipady=>10,
-anchor=>'s');
my $helpbody=$help->Scrolled("Text",
-relief=>'flat',
-font=>$deffont,
-height=>20,
-width=>60,
-scrollbars=>"e")->pack(-side=>'top',
-fill=>'y',
-expand=>1);
my $pos=tell DATA;
while (<DATA>) {
$_ =~ s/\$sysrc/$sysrc/;
last if (/^#/);
$helpbody->insert('end',$_);
}
seek(DATA,$pos,0);
$helpbody->configure(-state=>'disabled');
$tlwins{'helptext'}{'buttons'}[0]=
$help->Button(-text=>'Close',
-font=>$deffont,%butcol,
-command=>sub{destroy $help;
undef $tlwins{"helptext"};
})->pack(-side=>'bottom',
-fill=>'x');
$help->bind('<Control-c>'=>sub{destroy $help; undef $tlwins{"helptext"};});
}
# read config file; personal settings override the default settings
sub readrc {
my($rcfile,$icall)=@_;
my ($var,$val);
my $pathflag=0;
my $pathflagl=0;
my $pathflagh=0;
my $tdcolors='';
# for backward compatibility of converter/printer options
my ($dvips_opts,$pdfps_opts,$print_opts);
# read defaults resp. local/personal settings
CALL: {
($icall == 1) && do {
open (RC,"$rcfile") ||
&fatalmsg("Couldn't open system-wide default config file ${TeXDocRcDefaults}.\n");
last CALL;
};
($icall == 2) && do {
open (RC,"$rcfile") ||
&fatalmsg("Couldn't open personal config file ~/.texdocrc.\n");
last CALL;
};
};
while ($line = <RC>) {
next if ($line =~ /^($|#)/);
chomp $line;
($var,$val)=split("=",$line,2);
RCVARS: {
($var =~ /texdocpath/i) && do { # TEXDOCPATH
if (length $val > 0) {
$pathflag=1;
eval(join("",("\$",lc($var),"=\$val")));
}
last RCVARS;
};
($var =~ /localdocpath/i) && do { # LOCALDOCPATH
if (length $val > 0) {
$pathflagl=1;
eval(join("",("\$",lc($var),"=\$val")));
}
last RCVARS;
};
($var =~ /homedocpath/i) && do { # HOMEDOCPATH
if ($icall == 2 && length $val > 0) { # set only in ~/.texdocrc
$pathflagh=1;
eval(join("",("\$",lc($var),"=\$val")));
}
last RCVARS;
};
# other variables
eval(join("",("\$",lc($var),"=\$val")));
};
}
close(RC);
# find texmf/doc, and possibly other distribution texmf trees
if ($rcfile eq $sysrc || $pathflag == 1) {
$texmfmain=`kpsewhich --expand-path=${qq}\$TEXMFMAIN${qq}`;
chomp $texmfmain;
$texdocpath="$texmfmain/$texdocpath";
$texmfdist=`kpsewhich --expand-path=${qq}\$TEXMFDIST${qq}`;
chomp $texmfdist;
$distdocpath=join('/',"$texmfdist",basename($texdocpath,""))
if (length $texmfdist);
# TeXLive has this texmf-doc
$texmfdoc=join('/',"$texmfmain-doc",basename($texdocpath,""));
if (-e $texmfdoc) {
chomp $texmfdoc;
$docdocpath="$texmfdoc/$docdocpath";
}
}
# find texmf-local/doc, if there is one
if ($rcfile eq $sysrc || $pathflagl == 1) {
$texmflocal=`kpsewhich --expand-path=${qq}\$TEXMFLOCAL${qq}`;
chomp $texmflocal;
if (length $texmflocal) {
# if LOCALDOCPATH is not explicitly defined, it equals TEXDOCPATH
unless (defined $localdocpath) {
$localdocpath=basename($texdocpath,"");
}
$localdocpath="$texmflocal/$localdocpath";
}
}
# find user's $HOMETEXMF/doc, if there is one
if ($icall == 2 || $pathflagh == 1) {
$texmfhome=`kpsewhich --expand-path=${qq}\$HOMETEXMF${qq}`;
chomp $texmfhome;
if (length $texmfhome) {
# if HOMEDOCPATH is not explicitly defined, it equals TEXDOCPATH
unless (defined $homedocpath) {
$homedocpath=basename($texdocpath,"");
}
$homedocpath="$texmfhome/$homedocpath";
}
}
# colours: create array
@tdcolors=split(' ',$tdcolors);
if (scalar @tdcolors != 4) { # if old texdocrc file is used
if (!(-e $myrc) || (-e $myrc && $icall == 2)) {
print "NOTE: Button colours not set in $rcfile; using defaults.\n";
@tdcolors=('#d9d9d9','Black','#ececec','red');
}
}
%butcol=('-background'=>$tdcolors[0],
'-foreground'=>$tdcolors[1],
'-activebackground'=>$tdcolors[2],
'-activeforeground'=>$tdcolors[3]);
# viewer: suppress viewer warnings/use $PAGER if no $txt_viewer is defined
$quiet=($quiet =~ /y/i || $quiet eq "1") ? 1 : 0;
if (!(defined $txt_viewer) && defined $ENV{PAGER}) {
$txt_viewer=$ENV{PAGER};
}
# viewer: autostart viewer if listbox of search results contains only 1 item
$autoview=($autoview =~ /y/i || $autoview eq "1") ? 1 : 0;
# viewer: use text viewer to open files with unrecognized format
$xfmt_viewer=($xfmt_viewer =~ /y/i || $xfmt_viewer eq "1") ? 1 : 0;
# converter: output redirection flags for HTML/Text->PS converters
$htmlps_redir=($htmlps_redir =~ /(y|1)/i) ? 1 : 0;
$txtps_redir=($txtps_redir =~ /(y|1)/i) ? 1 : 0;
# for backward compatibility of converter/printer options
$dvips_conv.=" $dvips_opts" if ($dvips_opts);
$pdfps_conv.=" $pdfps_opts" if ($pdfps_opts);
$print_cmd.=" $print_opts" if ($print_opts);
}
# determine resolution if xwininfo is available
# slightly modified from a patch contributed by Reinhard Kotucha
sub x_resolution {
my $x_res;
if ($IsWin32) {
$x_res = &GetScreenWidth;
}
else {
open(XINF,"xwininfo -root|") or return 1200;
while (<XINF>) {
chop;
if (/Width:/) { ($x_res=$_)=/Width:\s+(\d+)/; last; }
}
close(XINF);
}
return $x_res;
}
# generate a random name for temporary files
sub randname {
my $namestr;
my @chars=("A","B","C","D","E","F","G","H","I","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z","a","b","c","d","e","f","g","h","i","j","k","l","m","n","o","p","q","r","s","t","u","v","w","x","y","z");
my $time=time;
my $maxcnt=3+substr($time,-1);
for (my $i=0; $i<=$maxcnt; $i++) { $namestr.=$chars[rand(51)] }
my @tmp=split(/./,$time/$$);
$namestr.=$tmp[1];
until (length($namestr) > 16) { $namestr.=int(rand(1000000)) }
return $namestr;
}
# find index of an array entry
sub arrindex {
my @array=@_;
my $item=pop @array;
my $i=0;
foreach (@array) {
last if ($_ eq $item);
++$i;
}
return $i;
}
# clean up and exit
sub clean_exit {
foreach (@tmpfiles) { if (-e $_) { unlink $_; } }
exit;
}
# alternative main window, launched for fatal error messages on startup
sub fatalmsg {
my($msg)=@_;
print STDERR $msg; # also print to stderr
# create frame for main window
my $main=new MainWindow;
$main->resizable(0,0);
$main->title("TeX Documentation Browser");
$main->bind('<Control-q>'=>sub { exit });
$main->bind('<Control-k>'=>sub { exit });
my $msgframe=$main->Frame(-background=>"#ffcc99");
my $cmdframe=$main->Frame;
$msgframe->pack(-side=>'top');
$cmdframe->pack(-side=>'bottom',-fill=>'x');
# make buttons for command frame
my $Qbut=$cmdframe->Button(-text=>'Kill',%butcol,
-command=>sub { exit })->pack(-fill=>'x');
# define common default font for labels and text explicitly
my @deffont=$Qbut->configure(-font);
# ensure readability on high-res screens (suggested by R.Kotucha)
$deffont='Helvetica -16 bold' if &x_resolution > 1200;
$Qbut->configure(-font=>$deffont);
$msgframe->Label(-text=>'FATAL ERROR',
-font=>$deffont)->pack(-side=>'top', -fill=>'x');
# get size of message text
my @dummy=split("\n",$msg);
my $nline=scalar @dummy;
my $msgwidth=0;
my $lline;
foreach (@dummy) {
$lline=length $_;
if ($lline > $msgwidth) { $msgwidth=$lline; }
}
my $message=$msgframe->Text(-relief=>'flat',
-font=>$deffont,
-height=>$nline,
-width=>$msgwidth)->pack(-side=>'top');
$message->insert('end',$msg);
$message->configure(-state=>'disabled');
MainLoop();
}
if ($IsWin32) {
sub GetScreenWidth {
my $GetDeviceCaps = new Win32::API('gdi32', 'GetDeviceCaps', 'NN', 'N');
my $HORZRES = 8; # from WinGDI.h
if(not defined $GetDeviceCaps) {
die "Can't import API GetDeviceCaps: $!\n";
}
return $GetDeviceCaps->Call(0, $HORZRES);
}
sub GetTempPath {
my $GetTempPath = new Win32::API('kernel32', 'GetTempPath', 'NP', 'N');
if(not defined $GetTempPath) {
die "Can't import API GetTempPath: $!\n";
}
my $lpBuffer = " " x 260;
$GetTempPath->Call(80, $lpBuffer);
}
sub ShellExecute {
my ($command, $file) = @_;
my $ShellExecute = new Win32::API('shell32', 'ShellExecute', 'NPPPPN', 'N');
if(not defined $ShellExecute) {
die "Can't import API ShellExecute: $!\n";
}
$ShellExecute->Call(0, $command, $file, '', '', 0);
}
sub GetHomeDir {
my $home = $ENV{'HOME'};
# Should check for CSIDL_...
return $home;
}
}
__END__
Many TeX programmers provide more or less detailed manuals for
their programs or packages. They are usually available as .dvi,
.ps, .pdf, .html or plain text files (sometimes included in the
.sty files instead of a separate documentation file) and can be
accessed with this browser, which is simply an interface to find a
documentation more easily. It starts the respective viewer for
reading the selected documentation making use of a database file
which contains the path entries according to the current teTeX
texmf/doc structure; additional usage of a system-wide local and
of individual user texmf trees with corresponding databases is also
possible.
The documentations are grouped in several categories shown in the
main window; pressing one of its buttons lists all documentations
belonging to this topic.
The topic window lets you select one documentation file, view or
send it to the default printer. By right-clicking on the selected
item you get the complete path of the file.
The search button of the main window allows you to search the
database for a string; it does not search file names. Enter the
string and hit <Return> to start the search or <Control-c> to
cancel. Just hitting <Return> without typing something in will
show the full list of files in the database.
Defaults for the documentation root directory, the viewers, the
converters, certain options and the printer are set in the global
configuration file $sysrc.
However, each user can put a copy of it as .texdocrc into his home
directory to modify them according to his needs; modification or
generation of ~/.texdocrc can also be done with the Settings menu.
Additionally, the settings can be changed temporarily with this
menu.
The following key shortcuts are defined for use with the browser:
<Ctrl-q> Quit browser
<Ctrl-m> Raise the Main window to the foreground
<Ctrl-s> Search a keyword in database (case insensitive)
<Ctrl-c> Cancel/close subwindow or search entry widget
<Ctrl-v> View selected document (topic windows)
<Ctrl-p> Print selected document (topic windows)
<Ctrl-t> Open settings menu
<Ctrl-k> Kill fatal error message window (same as
<Ctrl-q>)
<Ctrl-h> Open this help
<Return> Equivalent to button press;
OK in selection menu and search
<Tab> Next/previous widget
There are the following command line options:
-a autostart viewer if a listbox of search results
contains only one item; can also be set in a
configuration file
-v disable suppression of viewer messages sent to
stderr and of certain permanent message popup
windows, in case this was not set in a
configuration file; default: off
Some more information is available in the manpage of texdoctk.
Comments and suggestions to:
Thomas Ruedas
tr@dlc.ku.dk or tex-k@tug.org
# Local Variables:
# mode: perl
# auto-fill-hook: do-auto-fill
# End:
# vim: ts=8 sw=4 noexpandtab
|