texlive[44704] trunk: latexdiff (26jun17)

commits+karl at tug.org commits+karl at tug.org
Mon Jun 26 23:22:49 CEST 2017


Revision: 44704
          http://tug.org/svn/texlive?view=revision&revision=44704
Author:   karl
Date:     2017-06-26 23:22:49 +0200 (Mon, 26 Jun 2017)
Log Message:
-----------
latexdiff (26jun17)

Modified Paths:
--------------
    trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff-vc.pl
    trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff.pl
    trunk/Master/texmf-dist/doc/man/man1/latexdiff-vc.1
    trunk/Master/texmf-dist/doc/man/man1/latexdiff-vc.man1.pdf
    trunk/Master/texmf-dist/doc/man/man1/latexdiff.1
    trunk/Master/texmf-dist/doc/man/man1/latexdiff.man1.pdf
    trunk/Master/texmf-dist/doc/man/man1/latexrevise.man1.pdf
    trunk/Master/texmf-dist/doc/support/latexdiff/README
    trunk/Master/texmf-dist/doc/support/latexdiff/doc/latexdiff-man.pdf
    trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff
    trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff-fast
    trunk/Master/texmf-dist/scripts/latexdiff/latexdiff-vc.pl
    trunk/Master/texmf-dist/scripts/latexdiff/latexdiff.pl

Modified: trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff-vc.pl
===================================================================
--- trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff-vc.pl	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff-vc.pl	2017-06-26 21:22:49 UTC (rev 44704)
@@ -27,7 +27,12 @@
 #
 # TODO/IDEAS: - option to call external pre-processing codes
 #             - choose type of latex processor / bibtex (luatex, xelatex etc)
-# versio 1.2.0:
+# version 1.2.1 (22 June 2017)
+#    - bug fix: --hg option was not recognised (partially fixes github issue #93 )
+#    - wrap passed-through options to latexdiff in quotation marks (fix github issue #58 )
+#    - program names for latexdiff, latex, dvips, bibtex configurable (fixes issue #40)
+#
+# version 1.2.0:
 #    - depracation fix: left brace in RegEx now needs to be escaped
 #
 # version 1.1.1:
@@ -63,8 +68,8 @@
 use warnings ;
 
 my $versionstring=<<EOF ;
-This is LATEXDIFF-VC 1.2.0
-  (c) 2005-2016 F J Tilmann
+This is LATEXDIFF-VC 1.2.1
+  (c) 2005-2017 F J Tilmann
 EOF
 
 # output debug and intermediate files, set to 0 in final distribution
@@ -72,10 +77,14 @@
 
 
 # Option names
-my ($version,$help,$fast,$so,$postscript,$pdf,$onlychanges,$flatten,$force,$dir,$cvs,$rcs,$svn,$git,$hg,$diffcmd,$patchcmd, at revs);
+my ($version,$help,$fast,$showconfig,$so,$postscript,$pdf,$onlychanges,$flatten,$force,$dir,$cvs,$rcs,$svn,$git,$hg,
+    $run,$rundvi2,$diffcmd,$patchcmd, at revs, at configlist);
+# for passing through config options to latexdiff:
+my ($configlatexdiff, at config,$config,$assign);
+
 # Preset Variables
-my $latexdiff="latexdiff"; # Program for making the comparison
-my $latexcmd="latex"; # latex compiler if not further identified
+#my $latexdiff="latexdiff"; # Program for making the comparison
+#my $latexcmd="latex"; # latex compiler if not further identified
 my $vc="";
 my $tempdir=tempdir(CLEANUP => 1);   # generate a temp dir, which will automatically be deleted at program exit
 # Variables
@@ -82,6 +91,17 @@
 my ($file1,$file2,$diff,$diffbase,$rootdir,$answer,$options,$infile,$append,$dirname,$cwd);
 my (@files, at ldoptions, at tmpfiles, at ptmpfiles, at difffiles,$extracomp); # ,
 
+
+# keep commands to be used in config hash
+my (%CFG)=( 
+   LATEX     => "latex",
+   LATEXDIFF => "latexdiff",
+   DVI2      => "dvips",
+   BIBTEX    => "bibtex"
+   );
+
+
+
 Getopt::Long::Configure('pass_through','bundling');
 
 GetOptions('revision|r:s' => \@revs,
@@ -91,6 +111,8 @@
            'git' => \$git,
            'hg' => \$hg,
            'dir|d:s' => \$dir,
+	   'run' => \$run,
+	   'dvi' => \$rundvi2,
 	   'fast' => \$fast,
 	   'so' => \$so,
            'postscript|ps' => \$postscript,
@@ -98,6 +120,8 @@
            'force' => \$force,
 	   'only-changes' => \$onlychanges,
 	   'flatten:s' => \$flatten,
+	   'config|c=s' => \@configlist,
+	   'show-config' => \$showconfig,
            'version' => \$version,
 	   'help|h' => \$help,
 	   'debug!' => \$debug);
@@ -116,11 +140,11 @@
 }
 
 if ( $so ) {
-  $latexdiff='latexdiff-so';
+  $CFG{LATEXDIFF}='latexdiff-so';
 }
 if ( $fast ) { 
   die "Cannot specify more than one of --fast or --so " if ($so);
-  $latexdiff='latexdiff-fast';
+  $CFG{LATEXDIFF}='latexdiff-fast';
 }
 
 if ( $cvs ) {
@@ -138,8 +162,67 @@
   die "Cannot specify more than one of --cvs, --rcs, --svn --git or --hg." if ($vc);
   $vc="GIT";
 }
+if ( $hg ) {
+  die "Cannot specify more than one of --cvs, --rcs, --svn --git or --hg." if ($vc);
+  $vc="HG";
+}
 
+if ( $pdf ) {
+  $CFG{LATEX} = "pdflatex";
+  $run=1;
+} elsif ( $postscript ) {
+  $CFG{LATEX} = "latex";
+  $rundvi2=1;
+}
 
+if ( $rundvi2 ) {
+  $run=1;
+}
+
+# setting config variables from command line. Explicitly set configuration options take precedence over convenience options,
+# therefore this is executed last
+ at config=();
+foreach $config ( @configlist ) {
+  if (-f $config || lc $config eq '/dev/null' ) {
+    open(FILE,$config) or die ("Couldn't open configuration file $config: $!");
+    while (<FILE>) {
+      chomp;
+      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+      push (@config,$_);
+    }
+    close(FILE);
+  }
+  else {
+#    foreach ( split(",",$config) ) {
+#      push @config,$_;
+#    }
+     push @config,split(",",$config)
+  }
+}
+
+$configlatexdiff="";
+foreach $assign ( @config ) {
+  $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
+  if ( defined($CFG{$1})) {
+    # known latexdiff-vc option
+    $CFG{$1}=$2;
+  } else {
+    # unknown option, assume it is a latexdiff configuration variable
+    $configlatexdiff .= "$1=$2,";
+  }
+}
+# delete last comma inv $configlatexdiff if present
+$configlatexdiff =~ s/,$// ;
+
+if ($showconfig) {
+  print "# Configuration variables (latexdiff-vc):\n";
+  foreach ( keys %CFG ) {
+    print "$_=$CFG{$_}\n";
+  }
+  exit 0;
+}
+
+
 # check whether the first file name or first passed-through option for latexdiff got misinterpreted as an option to an empty -r option
 if ( @revs && ( -f $revs[$#revs] || $revs[$#revs] =~ /^-/ ) ) {
   push @ARGV,$revs[$#revs];
@@ -156,7 +239,7 @@
   $flatten="";
 }
 
-print "DEBUG: PDF $pdf latexdiff-vc command line: ", join(" ", at ARGV), "\n" if $debug; 
+print STDERR "DEBUG: PDF $pdf latexdiff-vc command line: ", join(" ", at ARGV), "\n" if $debug; 
 
 $file2=pop @ARGV;
 ( defined($file2) && $file2 =~ /\.(tex|bbl|flt)$/ ) or pod2usage("Must specify at least one tex, bbl or flt file");
@@ -217,7 +300,7 @@
     # $diffcmd  = "git diff ";
     # $patchcmd = "patch -R -p1";
   } elsif ( $vc eq "HG" ) {
-    $diffcmd  = "hg diff -r";
+    $diffcmd  = "hg diff --root . -r";
     $patchcmd = "patch -R -p1";
   } else {
     print STDERR "Unknown versioning system $vc \n";
@@ -241,6 +324,10 @@
   }
 }
 
+if ( length $configlatexdiff >0 ) {
+  push @ldoptions, "--config $configlatexdiff";
+}
+
 if ( defined($flatten) ) {
   push @ldoptions, "--flatten" ;
 }
@@ -252,15 +339,15 @@
 
 if ( scalar(@revs) == 0 ) {
   pod2usage("When -r option is not used, two .tex files (old and new) must be given on the command line") unless @files==2;
-  warn "Option -flatten should normally be combined with -r option (Results will probably not be as expected)" if $flatten;
+  warn "Option -flatten should normally be combined with -r option (Results will probably not be as expected)" if defined($flatten);
     #  compare two files
   $file1=shift @files ;
 } else {
   # revision control
-  if ( $flatten ) {
+  if ( defined($flatten) ) {
     pod2usage("Only one root file must be given  if --flatten option is used with version control") if  @files != 1;
     $tempdir='.' if ( $flatten eq "keep-intermediate" );
-    print "flatten tempdir |$tempdir|\n";
+    print STDERR "flatten tempdir |$tempdir|\n" if $debug;
     if ( $vc eq "SVN" ) {
       my (@infoout)=`svn info`;
       my (@urlline) = grep(/^URL:/, @infoout);
@@ -293,8 +380,14 @@
   length($revs[0]) > 0 or $revs[0]="HEAD";
 }
 
+if ($vc eq "HG" && scalar(@revs)) {
+  length($revs[$#revs]) > 0 or $revs[$#revs]="tip";
+  length($revs[0]) > 0 or $revs[0]="tip";
+}
 
+print STDERR "DEBUG LDOptions($#ldoptions): " . join(":", at ldoptions) . "\n" if $debug;
 
+
 # cycle through all files
 @difffiles=();
 
@@ -357,7 +450,11 @@
   system("mkdir -p $dirname") unless ( -e $dirname );
 
   # Remaining options are passed to latexdiff
-  $options = join(" ", at ldoptions);
+  if (scalar(@ldoptions) > 0 ) {
+    $options = "\'" . join("\' \'", at ldoptions) . "\'";
+  } else {
+    $options = "";
+  }
    
   if ( -e $diff && ! $force ) {
     print STDERR "OK to overwrite existing file $diff (y/n)? ";
@@ -367,13 +464,13 @@
       die "Abort ... " ;
     }
   }
-  print "Running: $latexdiff  $options \"$file1\" \"$file2\" > \"$diff\"\n";
-  unless ( system("$latexdiff $options \"$file1\" \"$file2\" > \"$diff\"") == 0 ) { 
-    print STDERR  "Something went wrong in $latexdiff. Deleting $diff and abort\n" ; unlink $diff ; exit(5) 
+  print STDERR "Running: $CFG{LATEXDIFF}  $options \"$file1\" \"$file2\" > \"$diff\"\n";
+  unless ( system("$CFG{LATEXDIFF} $options \"$file1\" \"$file2\" > \"$diff\"") == 0 ) { 
+    print STDERR  "Something went wrong in $CFG{LATEXDIFF}. Deleting $diff and abort\n" ; unlink $diff ; exit(5) 
   };
   print "Generated difference file $diff\n";
    
-  if ( ( $postscript or $pdf ) and !( scalar(@revs) && greptex( qr/\\document(?:class|style)/ , $diff ) ) ) {
+  if ( $run and !( scalar(@revs) && greptex( qr/\\document(?:class|style)/ , $diff ) ) ) {
     # save filename for later processing if postscript or pdf conversion is requested and either two-file mode was used (scalar(@revs)==0) or the diff file contains documentclass statement (ie. is a root document)
     push @difffiles, $diff ;
   }
@@ -393,16 +490,13 @@
   # adapt magically changebar styles to [pdftex] display driver if pdf output was selected
   if ( $pdf ) {
     system("sed \"s/Package\\[dvips\\]/Package[pdftex]/\" \"$diff\" > \"$diff.tmp$$\" ; \\mv \"$diff.tmp$$\" \"$diff\"");
-    $latexcmd = "pdflatex";
-  } elsif ( $postscript ) {
-    $latexcmd = "latex";
   }
 
-  if ( $pdf | $postscript ) {
+  if ( $run ) {
     print STDERR "PDF: $pdf Postscript: $postscript cwd $cwd\n" if $debug;
 
     if ( system("grep -q \'^[^%]*\\\\bibliography{\' \"$diff\"") == 0 ) { 
-      system("$latexcmd --interaction=batchmode \"$diff\"; bibtex \"$diffbase\";");
+      system("$CFG{LATEX} --interaction=batchmode \"$diff\"; $CFG{BIBTEX} \"$diffbase\";");
       push @ptmpfiles, "$diffbase.bbl","$diffbase.bbl" ; 
     }
 
@@ -409,30 +503,40 @@
     # if special needs, as CHANGEBAR
     if ( $extracomp ) {
       # print "Extracomp\n";
-      system("$latexcmd --interaction=batchmode \"$diff\";");
+      system("$CFG{LATEX} --interaction=batchmode \"$diff\";");
     }
 
     # final compilation
-    system("$latexcmd --interaction=batchmode \"$diff\";"); # needed if cross-refs
-    system("$latexcmd \"$diff\";"); # final, with possible error messages
+    system("$CFG{LATEX} --interaction=batchmode \"$diff\";"); # needed if cross-refs
+    system("$CFG{LATEX} \"$diff\";"); # final, with possible error messages
 
-    if ( $postscript ) {
+    if ( $rundvi2 ) {
       my $dvi="$diffbase.dvi";
-      my $ps="$diffbase.ps";
+      #my $ps="$diffbase.ps";
       my $ppoption="";
       
       if ( $onlychanges ) {
 	$ppoption="-pp ".join(",",findchangedpages("$diffbase.aux"));
       }
-      system("dvips $ppoption -o $ps $dvi");
+      #system("$CFG{DVI2} $ppoption -o $ps $dvi");
+      system("$CFG{DVI2} $ppoption $dvi");
       push @ptmpfiles, "$diffbase.aux","$diffbase.log",$dvi ;
-      print "Generated postscript file $ps\n";
-    } elsif ( $pdf ) {
+      #print STDERR "Generated postscript file $ps\n";
+    } elsif ( $run ) {
       if ( $onlychanges ) {
 	my @pages=findchangedpages("$diffbase.aux");
 	###      print ("Running pdftk \"$diffbase.pdf\" cat " . join(" ", at pages) . " output \"$diffbase-changedpage.pdf\"\n") or 
-	system ("pdftk \"$diffbase.pdf\" cat " . join(" ", at pages) . " output \"$diffbase-changedpage.pdf\"")==0 or 
-	  die ("Could not execute <pdftk $diffbase.pdf cat " . join(" ", at pages) . " output $diffbase-changedpage.pdf> . Return code: $?");
+        my $qpdf = `which qpdf`;
+        $qpdf =~ s/^\s+|\s+$//g;
+        my $pdftk = `which pdftk`;
+        $pdftk =~ s/^\s+|\s+$//g;
+        if (-x $qpdf) {
+          system("qpdf --linearize \"$diffbase.pdf\" --pages \"$diffbase.pdf\" " . join(",", @pages) . " -- \"$diffbase-changedpage.pdf\" ") == 0
+            or die("could not execute qpdf to strip pages. Return code: $?");
+        } elsif (-x $pdftk) {
+          system ("pdftk \"$diffbase.pdf\" cat " . join(" ", at pages) . " output \"$diffbase-changedpage.pdf\"")==0 or
+            die ("Could not execute <pdftk $diffbase.pdf cat " . join(" ", at pages) . " output $diffbase-changedpage.pdf> . Return code: $?");
+        }
 	move("$diffbase-changedpage.pdf","$diffbase.pdf");
       }
       push @ptmpfiles, "$diffbase.aux","$diffbase.log";
@@ -591,6 +695,51 @@
 
 With C<--flatten=keep-intermediate>, the intermediate revision snapshots are kept in the current directory (Default is to store them in a temporary directory and delete them after generating the diff file.)
 
+=item B<--config var1=val1,var2=val2,...> or B<-c var1=val1,..>
+
+
+
+=item B<--only-changes>
+
+Post-process the output such that only pages with changes on them are displayed. This requires the use of subtype ZLABEL 
+in latexdiff, which will be set automatically, but any manually set -s option will be overruled (also requires zref package to 
+be installed). (note that this option must be combined with --ps or --pdf to make sense)
+
+=item B<--force>
+
+Overwrite existing diff files without asking for confirmation. Default 
+behaviour is to ask for confirmation before overwriting an existing difference
+file.
+
+=item B<--run>
+
+run latex command on diff file after generation of diff file.
+
+=item B<--dvi>
+
+run latex and dvixxx commands after generation of diff file.
+
+=item B<-c configfile>
+
+Set configuration variables for latexdiff and latexdiff-vc.  The option can be repeated to set different
+variables (as an alternative to the comma-separated list).
+Available variables for latexdiff-vc:
+
+=over 8
+
+=item C<LATEXDIFF> latexdiff command (e.g. latexdiff-fast, latexdiff-so). This command should support the option C<--interaction=batchmode>
+
+=item C<LATEX> latex command (e.g. pdflatex, lualatex)
+
+=item C<DVI2>  Command for conversion of dvi file (e.g. dvips, dvipdf)
+
+=item C<BIBTEX> Command replacing bibtex
+
+=back
+
+All other config variables are passed to latexdiff. Explicity set configuration changes always override implicit
+changes by the following shortcut options B<--fast>, B<--so>, B<--ps> and B<--pdf>. 
+
 =item B<--fast> or B<--so>
 
 Use C<latexdiff-fast> or C<latexdiff-so>, respectively (instead of C<latexdiff>).
@@ -609,22 +758,16 @@
 Generate pdf output from difference file using C<pdflatex>. This will
 run the sequence C<pdflatex; pdflatex> on the difference file, or
 C<pdflatex; bibtex; pdflatex; pdflatex> for files requiring bibtex.
+Note that this is not just a shortcut for setting configuration variable but also triggers 
+some special behaviour.
 
-=item B<--only-changes>
 
-Post-process the output such that only pages with changes on them are displayed. This requires the use of subtype ZLABEL 
-in latexdiff, which will be set automatically, but any manually set -s option will be overruled (also requires zref package to 
-be installsed). (note that this option must be combined with --ps or --pdf to make sense)
+=item B<--show-config>
 
-=item B<--force>
+Show values of configuration variables.
 
-Overwrite existing diff files without asking for confirmation. Default 
-behaviour is to ask for confirmation before overwriting an existing difference
-file.
+=item B<--help> or B<-h>
 
-=item B<--help> or
-B<-h>
-
 Show help text
 
 =item B<--version>
@@ -654,8 +797,8 @@
 
 =head1 AUTHOR
 
-Version 1.2.0
-Copyright (C) 2005-2016 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2005-2017 Frederik Tilmann
 
 This program is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License Version 3

Modified: trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff.pl
===================================================================
--- trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff.pl	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Build/source/texk/texlive/linked_scripts/latexdiff/latexdiff.pl	2017-06-26 21:22:49 UTC (rev 44704)
@@ -24,6 +24,28 @@
 #
 # ToDo:
 #
+# Version 1.2.1 (22 June 2017)
+#    - add "DeclareOldFontCommand" to styles using \bf or \sf old style font commands (fixies issue #92 )
+#    - improved markup: process lstinline commands in listings package correctly
+#      for styles using colour, \verb and \lstinline arguments are marked up with colour (blue for added, red for deleted)
+#    - bug fix: protecting inline math expressions for mbox did not work as intended (see stack exchange question: http://tex.stackexchange.com/questions/359412/compiling-the-latexdiff-when-adding-a-subscript-before-a-pmatrix-environment-cau)
+#    - bug fix: when deleted \item commands are followed immediately by unsafe commands, they were not restored properly
+#      (thanks to J. Protze for pull request) (pull request #89)
+#    - treat lstlisting and comment as equivalent to verbatim environment
+#      make environments that are treated like verbatim environments configurable (config variable VERBATIMENV)
+#      treat lstinlne as equivalent to verb command
+#      partially addresses issue #38
+#    - refactoring: set default configuration variables in a hash, and those that correspond to lists 
+#    - feature: option --add-to-config used to amend configuration variables, which are regex pattern lists
+#    - bug fix: deleted figures when endfloat package is activated
+#    - bug fix: alignat environment now always processed correctly (fix issues #65)
+#    - bug fix: avoid processing of commands as potential files in routine init_regex_arr (fix issue #70 )
+#    - minimal feature enhancement: treat '@' as allowed character in commands (strictly speaking requires prior \makeatletter statement, but always assuming it to be 
+#       @       a letter if it is part of a command name will usually lead to the correct behaviour (see http://tex.stackexchange.com/questions/346651/latexdiff-and-let)
+#    - new feature/bug fix: --flatten option \endinput in included files now respected but only if \endinput stands right at the beginning of the line (issue #77)
+#    - bug fix: flatten would incorrectly attempt to process commented out \include commands (from discussion in issue #77 )
+#    - introduce an invisible space (\hspace{0pt} after \mbox{..} auxiliary commands (not in math mode), to allow line breaks between added and deleted citations (change should not cause adverse behaviour otherwise)
+#
 # Version 1.2.0:
 #    - highlight new and deleted figures
 #    - bug fix in title mark-up. Previously deleted commands in title (such as \title, \author or \date) were marked up erroneously
@@ -597,34 +619,76 @@
 
 
 my ($versionstring)=<<EOF ;
-This is LATEXDIFF 1.2.0  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
-  (c) 2004-2016 F J Tilmann
+This is LATEXDIFF 1.2.1  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
+  (c) 2004-2017 F J Tilmann
 EOF
 
-# Configuration variables: these have to be visible from the subroutines
-my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
-                     # shorter identical blocks will be merged to the previous word
-my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
-                      # this is only used for --graphics-markup=BOTH option
-my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
-my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
-my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
-my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
-my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
-my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
-my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
-my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+# Hash with defaults for configuration variables. These marked undef have default values constructed from list defined in the DATA block
+# (under tag CONFIG)
+my %CONFIG=(
+   MINWORDSBLOCK => 3, # minimum number of tokens to form an independent block
+                        # shorter identical blocks will be merged to the previous word
+   SCALEDELGRAPHICS => 0.5, # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+                             # this is only used for --graphics-markup=BOTH option
+   FLOATENV => undef ,   # Environments in which FL variants of defined commands are used
+   PICTUREENV => undef ,   # Environments in which all change markup is removed
+   MATHENV => undef ,           # Environments turning on display math mode (code also knows about \[ and \])
+   MATHREPL => 'displaymath',  # Environment introducing deleted maths blocks
+   MATHARRENV => undef ,           # Environments turning on eqnarray math mode
+   MATHARRREPL => 'eqnarray*',  # Environment introducing deleted maths blocks
+   ARRENV => undef , # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+   COUNTERCMD => undef,
+                                        # COUNTERCMD textcmds which are associated with a counter
                                         # If any of these commands occur in a deleted block
                                         # they will be succeeded by an \addtocounter{...}{-1}
                                         # for the associated counter such that the overall numbers
                                         # should be the same as in the new file
+   LISTENV => undef ,  # list making environments - they will generally be kept
+   VERBATIMENV => undef,    # Environments whose content should be treated as verbatim text 
+   ITEMCMD => 'item'                    # command marking item in a list environment
+);
+# Configuration variables: these have to be visible from the subroutines
+my ($ARRENV,
+    $COUNTERCMD,
+    $FLOATENV,
+    $ITEMCMD,
+    $LISTENV,
+    $MATHARRENV,
+    $MATHARRREPL,
+    $MATHENV,
+    $MATHREPL,
+    $MINWORDSBLOCK,
+    $PICTUREENV,
+    $SCALEDELGRAPHICS,
+    $VERBATIMENV
+);
+
+# my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
+#                      # shorter identical blocks will be merged to the previous word
+# my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+#                       # this is only used for --graphics-markup=BOTH option
+# my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
+# my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
+# my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
+# my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
+# my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
+# my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
+# my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+# my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+#                                         # If any of these commands occur in a deleted block
+#                                         # they will be succeeded by an \addtocounter{...}{-1}
+#                                         # for the associated counter such that the overall numbers
+#                                         # should be the same as in the new file
+# my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
+# my $ITEMCMD='item';   # command marking item in a list environment
+
+
+
 my $LABELCMD='(?:label)';                # matching commands are disabled within deleted blocks - mostly useful for maths mode, as otherwise it would be fine to just not add those to SAFECMDLIST
 my @UNSAFEMATHCMD=('qedhere');           # Commands which are definitely unsafe for marking up in math mode (amsmath qedhere only tested to not work with UNDERLINE markup) (only affects WHOLE and COARSE math markup modes). Note that unlike text mode (or FINE math mode0 deleted unsafe commands are not deleted but simply taken outside \DIFdel
 my $MBOXINLINEMATH=0; # if set to 1 then surround marked-up inline maths expression with \mbox ( to get around compatibility
                       # problems between some maths packages and ulem package
 
-my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
-my $ITEMCMD='item';
 
 # Markup strings
 # If at all possible, do not change these as parts of the program
@@ -674,7 +738,8 @@
 
 my ($i,$j,$l);
 my ($old,$new);
-my ($line);
+my ($line,$key);
+my (@dumlist);
 my ($newpreamble,$oldpreamble);
 my (@newpreamble, at oldpreamble, at diffpreamble, at diffbody);
 my ($latexdiffpreamble);
@@ -709,7 +774,7 @@
 
 my ($mboxcmd);
 
-my (@configlist, at labels,
+my (@configlist, at addtoconfiglist, at labels,
     @appendsafelist, at excludesafelist,
     @appendmboxsafelist, at excludemboxsafelist,
     @appendtextlist, at excludetextlist,
@@ -751,6 +816,7 @@
 	   'subtype|s=s' => \$subtype, 
 	   'floattype|f=s' => \$floattype, 
 	   'config|c=s' => \@configlist,
+	   'add-to-config|c=s' => \@addtoconfiglist,
 	   'preamble|p=s' => \$preamblefile,
 	   'encoding|e=s' => \$encoding,
 	   'label|L=s' => \@labels,
@@ -917,21 +983,64 @@
 }
 foreach $assign ( @config ) {
   $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
-  if ( $1 eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $2; }
-  elsif ( $1 eq "FLOATENV" ) { $FLOATENV = $2 ; }
-  elsif ( $1 eq "ITEMCMD" ) { $ITEMCMD = $2 ; }
-  elsif ( $1 eq "LISTENV" ) { $LISTENV = $2 ; }
-  elsif ( $1 eq "PICTUREENV" ) { $PICTUREENV = $2 ; }
-  elsif ( $1 eq "MATHENV" ) { $MATHENV = $2 ; }
-  elsif ( $1 eq "MATHREPL" ) { $MATHREPL = $2 ; }
-  elsif ( $1 eq "MATHARRENV" ) { $MATHARRENV = $2 ; }
-  elsif ( $1 eq "MATHARRREPL" ) { $MATHARRREPL = $2 ; }
-  elsif ( $1 eq "ARRENV" ) { $ARRENV = $2 ; }
-  elsif ( $1 eq "COUNTERCMD" ) { $COUNTERCMD = $2 ; }
-  elsif ( $1 eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $2 ; }
-  else { die "Unknown variable $1 in assignment.";}
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}=$2;
 }
 
+my @addtoconfig=();
+foreach $config ( @addtoconfiglist ) {
+  if (-f $config || lc $config eq '/dev/null' ) {
+    open(FILE,$config) or die ("Couldn't open addd-to-config file $config: $!");
+    while (<FILE>) {
+      chomp;
+      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+      push (@addtoconfig,$_);
+    }
+    close(FILE);
+  }
+  else {
+#    foreach ( split(",",$config) ) {
+#      push @addtoconfig,$_;
+#    }
+     push @addtoconfig,split(",",$config)
+  }
+}
+
+# initialise default lists from DATA
+# for those configuration variables, which have not been set explicitly, initiate from list in document
+foreach $key ( keys(%CONFIG) ) {
+  if (!defined $CONFIG{$key}) {
+    @dumlist=();
+    init_regex_arr_data(\@dumlist,"$key CONFIG");
+    $CONFIG{$key}=join(";", at dumlist)
+  }
+}
+
+
+foreach $assign ( @addtoconfig ) {
+  $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}.=";$2";
+}
+
+# Map from hash to variables (we do this to have more concise code later, change from comma-separeted list)
+foreach  ( keys(%CONFIG) ) {
+  if ( $_ eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $CONFIG{$_}; }
+  elsif ( $_ eq "FLOATENV" ) { $FLOATENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "ITEMCMD" ) { $ITEMCMD = $CONFIG{$_} ; }
+  elsif ( $_ eq "LISTENV" ) { $LISTENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "PICTUREENV" ) { $PICTUREENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHENV" ) { $MATHENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHREPL" ) { $MATHREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "MATHARRENV" ) { $MATHARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHARRREPL" ) { $MATHARRREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "ARRENV" ) { $ARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "VERBATIMENV" ) { $VERBATIMENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "COUNTERCMD" ) { $COUNTERCMD = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $CONFIG{$_} ; }
+  else { die "Unknown configuration variable $_.";}
+}
+
 if ( $mathmarkup == COARSE || $mathmarkup == WHOLE ) {
   push(@MATHTEXTCMDLIST,qr/^MATHBLOCK(?:$MATHENV|$MATHARRENV|SQUAREBRACKET)$/);
 }
@@ -944,46 +1053,11 @@
 
 
 
-if ($showpreamble) {
-  print "\nPreamble commands:\n";
-  print $latexdiffpreamble ;
+if ($showconfig || $showtext || $showsafe || $showpreamble) {
+  show_configuration();
+  exit 0; 
 }
 
-if ($showsafe) {
-  print "\nCommands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
-  print_regex_arr(@SAFECMDLIST);
-  print "\nCommands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
-  print_regex_arr(@SAFECMDEXCL);
-}
-
-if ($showtext) {
-  print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
-  print_regex_arr(@TEXTCMDLIST);
-  print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
-  print_regex_arr(@CONTEXT1CMDLIST);
-  print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
-  print_regex_arr(@CONTEXT2CMDLIST);  
-  print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
-  print_regex_arr(@TEXTCMDEXCL);
-}
-
-
-if ($showconfig) {
-  print "Configuration variables:\n";
-  print "ARRENV=$ARRENV\n";
-  print "COUNTERCMD=$COUNTERCMD\n";
-  print "FLOATENV=$FLOATENV\n";
-  print "ITEMCMD=$ITEMCMD\n";
-  print "LISTENV=$LISTENV\n";
-  print "MATHARRENV=$MATHARRENV\n";
-  print "MATHARRREPL=$MATHARRREPL\n";
-  print "MATHENV=$MATHENV\n";
-  print "MATHREPL=$MATHREPL\n";
-  print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
-  print "PICTUREENV=$PICTUREENV\n";
-}
-if ($showconfig || $showtext || $showsafe || $showpreamble) {
-  exit 0; }
 if ( @ARGV != 2 ) { 
   print STDERR "2 and only 2 non-option arguments required.  Write latexdiff -h to get help\n";
   exit(2);
@@ -1033,7 +1107,7 @@
   my $word_ja='\p{Han}+|\p{InHiragana}+|\p{InKatakana}+';
   my $word='(?:' . $word_ja . '|(?:(?:[-\w\d*]|\\\\[\"\'\`~^][A-Za-z\*])(?!(?:' . $word_ja . ')))+)';
   my $cmdleftright='\\\\(?:left|right|[Bb]igg?[lrm]?|middle)\s*(?:[<>()\[\]|\.]|\\\\(?:[|{}]|\w+))';
-  my $cmdoptseq='\\\\[\w\d\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
+  my $cmdoptseq='\\\\[\w\d@\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
   my $backslashnl='\\\\\n';
   my $oneletcmd='\\\\.\*?(?:\['.$brat0.'\]|\{'. $pat_n . '\})*';
   my $math='\$(?:[^$]|\\\$)*?\$|\\\\[(](?:.|\n)*?\\\\[)]';
@@ -1056,18 +1130,30 @@
 
 
 # set the labels to be included into the file
+# first find out which file name is longer for correct alignment
+my ($diff,$oldlabel_n_spaces,$newlabel_n_spaces);
+$oldlabel_n_spaces = 0;
+$newlabel_n_spaces = 0;
+$diff = length($newfile) - length($oldfile);
+if ($diff > 0) {
+  $oldlabel_n_spaces = $diff;
+}
+if ($diff < 0) {
+  $newlabel_n_spaces = abs($diff);
+}
+
 my ($oldtime,$newtime,$oldlabel,$newlabel);
 if (defined($labels[0])) {
   $oldlabel=$labels[0] ;
 } else {
   $oldtime=localtime((stat($oldfile))[9]); 
-  $oldlabel="$oldfile   " . " "x(length($newfile)-length($oldfile)) . $oldtime;
+  $oldlabel="$oldfile   " . " "x($oldlabel_n_spaces) . $oldtime;
 }
 if (defined($labels[1])) {
   $newlabel=$labels[1] ;
 } else {
   $newtime=localtime((stat($newfile))[9]);
-  $newlabel="$newfile   " . " "x(length($oldfile)-length($newfile)) . $newtime;
+  $newlabel="$newfile   " . " "x($newlabel_n_spaces) . $newtime;
 }
 
 $encoding=guess_encoding($newfile) unless defined($encoding);
@@ -1088,7 +1174,6 @@
 exetime(1);
 ($oldpreamble,$oldbody,$oldpost)=splitdoc($old,'\\\\begin\{document\}','\\\\end\{document\}');
 
-
 ($newpreamble,$newbody,$newpost)=splitdoc($new,'\\\\begin\{document\}','\\\\end\{document\}');
 
 
@@ -1167,22 +1252,28 @@
   # is not compatible with this.
   # (A more elegant solution would be to suppress the redefinitions of the \DIFaddbeginFL etc commands, but for this narrow use case
   #  I currently don't see this as an efficient use of time)
-  foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
-    @matches=( $latexdiffpreamble =~ m/command{\\DIFaddbeginFL}{($pat_n)}/sg ) ;
-    # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one elemen
-    if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
-      die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
-      exit 10;
-    }
+  ### The foreach loop does not make sense here. I don't know why I put this in -  (F Tilmann)
+  ###foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
+  @matches=( $latexdiffpreamble =~ m/command\{\\DIFaddbeginFL}\{($pat_n)}/sg ) ;
+  # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one element
+  if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
+    die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
+    exit 10;
   }
+  ###}
   $latexdiffpreamble .= join "\n",("\\newcommand{\\DIFscaledelfig}{$SCALEDELGRAPHICS}",extrapream("HIGHLIGHTGRAPHICS"),"");
 
   # only change required for highlighting both is to declare \includegraphics safe, as preamble already contains commands for deleted environment
   if ( $graphicsmarkup == BOTH ) {
-    init_regex_arr_ext(\@SAFECMDLIST,'includegraphics');
+    init_regex_arr_list(\@SAFECMDLIST,'includegraphics');
   }
 }
 
+# If listings is being used and latexdiffpreamble uses color markup
+if (defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/)) {
+  $latexdiffpreamble .= join "\n",(extrapream("LISTINGS"),"");
+}
+
 # adding begin and end marker lines to preamble
 $latexdiffpreamble = "%DIF PREAMBLE EXTENSION ADDED BY LATEXDIFF\n" . $ latexdiffpreamble . "%DIF END PREAMBLE EXTENSION ADDED BY LATEXDIFF\n";
 
@@ -1244,11 +1335,11 @@
   # this is done to get around an incompatibility between the ulem and siunitx package
   print STDERR "siunitx package detected.\n" if $verbose ;
   my $mboxcmds='SI,ang,numlist,numrange,SIlist,SIrange';
-  init_regex_arr_ext(\@SAFECMDLIST,'num,si');
+  init_regex_arr_list(\@SAFECMDLIST,'num,si');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -1258,9 +1349,9 @@
   print STDERR "cleveref package detected.\n" if $verbose ;
   my $mboxcmds='[Cc]ref(?:range)?\*?,labelcref,(?:lc)?name[cC]refs?' ;
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -1269,17 +1360,17 @@
   # this is done to get around an incompatibility between ulem and glossaries package 
   print STDERR "glossaries package detected.\n" if $verbose ;
   my $mboxcmds='[gG][lL][sS](?:|pl|disp|link|first|firstplural|desc|user[iv][iv]?[iv]?),[aA][cC][rR](?:long|longpl|full|fullpl),[aA][cC][lfp]?[lfp]?';
-  init_regex_arr_ext(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
+  init_regex_arr_list(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
 if (defined $packages{"chemformula"} or defined $packages{"chemmacros"} ) {
   print STDERR "chemformula package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ch');
+  init_regex_arr_list(\@SAFECMDLIST,'ch');
   push(@UNSAFEMATHCMD,'ch');
   # The next command would be needed to allow highlighting the interior of \ch commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -1288,7 +1379,7 @@
 
 if (defined $packages{"mhchem"} ) {
   print STDERR "mhchem package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ce');
+  init_regex_arr_list(\@SAFECMDLIST,'ce');
   push(@UNSAFEMATHCMD,'ce','cee');
   # The next command would be needed to allow highlighting the interior of \cee commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -1330,7 +1421,7 @@
 
 # add commands in MBOXCMDLIST to SAFECMDLIST
 foreach $mboxcmd ( @MBOXCMDLIST ) {
-  init_regex_arr_ext(\@SAFECMDLIST, $mboxcmd);
+  init_regex_arr_list(\@SAFECMDLIST, $mboxcmd);
 }
 
 # check if \label is in SAFECMDLIST, and if yes replace "label" in $LABELCMD by something that never matches (we hope!)
@@ -1376,7 +1467,71 @@
 print STDERR "(",exetime()," s)\n","Done.\n" if $verbose;
 
 
+# liststringtoregex(liststring)
+# expands string with semi-colon separated list into a regular expression corresponding 
+# matching any of the elements
+sub liststringtoregex {
+  my ($liststring)=@_;
+  my @elements=grep /\S/,split(";",$liststring);
+  if ( @elements) {
+    return('(?:(?:' . join(')|(?:', at elements) .'))');
+  } else {
+    return "";
+  }
+}
 
+# show_configuration
+# note that this is not encapsulated but uses variables from the main program 
+# It is provided for convenience because in the future it is planned to allow output
+# to be modified based on what packages are read etc - this works only if the input files are actually red
+# whether or not additional files are provided
+sub show_configuration {
+  if ($showpreamble) {
+    print "\nPreamble commands:\n";
+    print $latexdiffpreamble ;
+  }
+
+  if ($showsafe) {
+    print "\nsafecmd: Commands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
+    print_regex_arr(@SAFECMDLIST);
+    print "\nsafecmd-exlude: Commands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
+    print_regex_arr(@SAFECMDEXCL);
+    print "\nmboxsafecmd:  Commands safe only if they are surrounded by \\mbox command:\n";
+    print_regex_arr(@MBOXCMDLIST);
+    print "\nnmboxsafecmd: Commands not safe:\n";
+    print_regex_arr(@MBOXCMDEXCL);
+  }
+
+  if ($showtext) {
+    print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
+    print_regex_arr(@TEXTCMDLIST);
+    print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
+    print_regex_arr(@CONTEXT1CMDLIST);
+    print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
+    print_regex_arr(@CONTEXT2CMDLIST);  
+    print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
+    print_regex_arr(@TEXTCMDEXCL);
+  }
+
+
+  if ($showconfig) {
+    print "Configuration variables:\n";
+    print "ARRENV=$ARRENV\n";
+    print "COUNTERCMD=$COUNTERCMD\n";
+    print "FLOATENV=$FLOATENV\n";
+    print "ITEMCMD=$ITEMCMD\n";
+    print "LISTENV=$LISTENV\n";
+    print "MATHARRENV=$MATHARRENV\n";
+    print "MATHARRREPL=$MATHARRREPL\n";
+    print "MATHENV=$MATHENV\n";
+    print "MATHREPL=$MATHREPL\n";
+    print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
+    print "PICTUREENV=$PICTUREENV\n";
+  }
+}
+
+
+
 ## guess_encoding(filename)
 ## reads the first 20 lines of filename and looks for call of inputenc package
 ## if found, return the option of this package (encoding), otherwise return utf8
@@ -1480,46 +1635,44 @@
 # and adds the created commands which are clearly safe to @SAFECMDLIST
 sub add_safe_commands {
   my ($preamble)=@_;
-  my $added_command = 1;
 
+  # get rid of comments
   $preamble=~s/(?<!\\)%.*$//mg ;
 
-  while ($added_command) { 
-    $added_command  = 0;
-      # get rid of comments
-    my $to_test = "";
-      # test for \DeclareMathOperator{\foo}{myoperator}
-    while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
-      $to_test=$1;
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
-      }
+  my $to_test = "";
+  # test for \DeclareMathOperator{\foo}{myoperator}
+  while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
+    $to_test=$1;
+    if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+      # one should add $to_test to the list of safe commands.
+      init_regex_arr_list(\@SAFECMDLIST, $to_test);
+      print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
     }
+  }
 
-    while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
-      my $maybe_to_test  = $1;
-      my $should_be_safe = $2;
-      my $success = 0;
-      # test if all latex commands inside it are safe
-      $success = 1;
-      if ($should_be_safe =~ m/\\\\/) {
-	$success = 0;
-      } else {
-	while ($should_be_safe =~ m/\\(\w+)/g) {
-	  $success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
-	}
+  while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
+    my $maybe_to_test  = $1;
+    my $should_be_safe = $2;
+    print STDERR "DEBUG Checking new command: maybe_to_test, should_be_safe: $1 $2\n" if $debug;
+    my $success = 0;
+    # test if all latex commands inside it are safe
+    $success = 1;
+    if ($should_be_safe =~ m/\\\\/) {
+      $success = 0;
+    } else {
+      while ($should_be_safe =~ m/\\(\w+)/g) {
+	###	  print STDERR "DEBUG: Testing command $1 " if $debug;
+	$success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
+	###        print STDERR " success=$success\n" if $debug;
       }
-      ###      }
-      $to_test = $maybe_to_test if $success;
-      
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-#        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
+    }
+    ###      }
+    if ($success) {
+      $to_test = $maybe_to_test;
+      if (  not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+	#        # one should add $to_test to the list of safe commands.
+	init_regex_arr_list(\@SAFECMDLIST, $to_test);
+	print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
       }
     }
   }
@@ -1526,6 +1679,19 @@
 }
 
 
+# helper function for flatten
+# remove \endinput at beginning of line and everything
+# following it, # if \endinput is not at the beginning of
+# the line, nothing will be removed. It is assumed that
+# this case is most common when \endinput is part of a
+# conditional clause.  The file will only be processed
+# correctly if the conditional is always false,
+# i.e. \endinput # not actually reached
+sub remove_endinput {
+  # s/// operates on default input
+  $_[0] =~ s/^\\endinput.*\Z//ms ;
+  return($_[0]);
+}
 
 # flatten($text,$preamble,$filename,$encoding)
 # expands \input and \include commands within text
@@ -1553,7 +1719,7 @@
   print STDERR "DEBUG: includeonly $includeonly\n" if $debug;
 
   # recursively replace \\input and \\include files
-  1 while $text=~s/(^(?:[^%\n]|\\%)*)\\input{(.*?)}|\\include{(${includeonly}(?:\.tex)?)}/{ 
+  1 while $text=~s/(^(?:[^%\n]|\\%)*)(?:\\input\{(.*?)\}|\\include\{(${includeonly}(?:\.tex)?)\})/{ 
 	    $begline=(defined($1)? $1 : "") ;
 	    $fname = $2 if defined($2) ;
 	    $fname = $3 if defined($3) ;
@@ -1565,6 +1731,7 @@
             # content of file becomes replacement value (use recursion), add \newpage if the command was include
             ###$replacement=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
 	    $replacement=flatten(read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding), $preamble,$filename,$encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
+	    $replacement = remove_endinput($replacement); 
 	    # \include always starts a new page; use explicit \newpage command to simulate this
 	    $newpage=(defined($3)? " \\newpage " : "") ;
 	    "$begline$newpage$replacement$newpage";
@@ -1593,6 +1760,7 @@
            $subfile=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
            ($subpreamble,$subbody,$subpost)=splitdoc($subfile,'\\\\begin\{document\}','\\\\end\{document\}');
 	   $replacement=flatten($subbody, $preamble,$filename,$encoding);
+	   $replacement = remove_endinput($replacement); 
 	   "$begline$replacement";
   }/exgm;
 
@@ -2236,7 +2404,7 @@
       if ( $word =~ /^\\(?!\()(\\|[`'^"~=.]|[\w*@]+)(.*?)(\s*)$/s &&  iscmd($1,\@MBOXCMDLIST,\@MBOXCMDEXCL)) {
 	# $word is a safe command in MBOXCMDLIST
 	###print STDERR "DEBUG Mboxsafecmd detected:$word:\n" if $debug ;
-	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}$AUXCMD\n" );
+	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}\\hspace{0pt}$AUXCMD\n" );
       } else {
 	# $word is a normal word or a safe command (not in MBOXCMDLIST)
 	push (@$retval,$word);
@@ -2253,47 +2421,49 @@
 
 #used in preprocess
 sub take_comments_and_enter_from_frac() {
-      ###*************take the \n and % between frac and {}***********
-      ###notice all of the substitution are made none global
-      while( m/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/s ) {
-	### if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
-	if( $2 !~ m/\\end{$1}/s ) {
-	  ### take out % and \n from the next match only (none global)
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
+      #*************take the \n and % between frac and {}***********
+      #notice all of the substitution are made none global
+      while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/s ) {
+	# if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
+	### print STDERR "Match the following in take_comments and_enter_from_frac(1):\n****$&****\n" if $debug;
+	if( $2 !~ m/\\end\{$1}/s ) {
+	  # take out % and \n from the next match only (none global)
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
 	}
 	else{
-	  ###there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+	  #there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
 	}
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       ###*************take the \n and % between frac and {}***********
       
       ###**********take the \n and % between {} and {} of the frac***************
       while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/s ) {
-	### if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	# if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	###print STDERR "Match the following in take_comments and_enter_from_frac(2):\n****$&****\n" if $debug;
         if( $2 !~ m/\\end\{$1\}/s ) {
-          ### from now on CURRFRAC is the frac we are looking at
+          # from now on CURRFRAC is the frac we are looking at
           s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\\end\{$1\}/s;
           while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{(.*?)\\end\{\1\}/s ) {
-             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end{\1}/s ) {
+             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1}/s ) {
                    s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\}\{$6\\end\{$1\}/s;
                } 
               else { # there is no comment or \n between the two brackets {}{}
-              ### change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
-                   s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
+              # change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
+                   s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end\{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
               }
             }
         }
         else{
           ###there are no more frac in $2, we want to find the next one so we clear the begin-end from the pattern
-          s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+          s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
         }
         
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       s/FRACSTART/\\frac/g;
       ###***************take the \n and % between {} and {} of the frac*********************
 }
@@ -2306,8 +2476,8 @@
 # 2. mark all first empty line (in block of several) with \PAR tokens
 # 3. Convert all '\%' into '\PERCENTAGE ' and all '\$' into \DOLLAR to make parsing regular expressions easier
 # 4. Convert all \verb|some verbatim text| commands (where | can be an arbitrary character)
-#    into \verb{hash}
-# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}
+#    into \verb{hash}  (also lstinline)
+# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}  (not only verbatim, all patterns matching VERBATIMENV)
 # 6. Convert _n into \SUBSCRIPTNB{n} and _{nnn} into \SUBSCRIPT{nn}
 # 7. Convert ^n into \SUPERSCRIPTNB{n} and ^{nnn} into \SUPERSCRIPT{nn}
 # 8. a. Convert $$ $$ into \begin{DOLLARDOLLAR} \end{DOLLARDOLLAR}
@@ -2323,6 +2493,17 @@
 # Returns: leading white space removed in step 1
 sub preprocess {
   for (@_) { 
+
+
+    # change in \verb and similar commands - note that I introduce an extra space here so that the 
+    #       already hashed variants do not trigger again
+    # transform \lstinline{...}
+#    s/\\lstinline(\[$brat0\])?(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+#    s/\\lstinline(\[$brat0\])?((\S).*?\2)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)((\S).*?\3)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\(verb\*?|lstinline)(\S)(.*?)\2/"\\DIF${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
+
     #    Change \{ to \QLEFTBRACE, \} to \QRIGHTBRACE, and \& to \AMPERSAND
     s/(?<!\\)\\\{/\\QLEFTBRACE /sg;
     s/(?<!\\)\\\}/\\QRIGHTBRACE /sg;
@@ -2330,12 +2511,10 @@
 # replace {,} in comments with \\CLEFTBRACE,\\CRIGHTBRACE
     1 while s/((?<!\\)%.*)\{(.*)$/$1\\CLEFTBRACE $2/mg ;
     1 while s/((?<!\\)%.*)\}(.*)$/$1\\CRIGHTBRACE $2/mg ;
-
     s/\n(\s*?)\n((?:\s*\n)*)/\n$1\\PAR\n$2/g ;
     s/(?<!\\)\\%/\\PERCENTAGE /g ;  # (?<! is negative lookbehind assertion to prevent \\% from being converted
     s/(?<!\\)\\\$/\\DOLLAR /g ;  # (?<! is negative lookbehind assertion to prevent \\$ from being converted
-    s/(\\verb\*?)(\S)(.*?)\2/"${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
-    s/\\begin\{(verbatim\*?)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
+    s/\\begin\{($VERBATIMENV)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
     # Convert _n or _\cmd into \SUBSCRIPTNB{n} or \SUBSCRIPTNB{\cmd} and _{nnn} into \SUBSCRIPT{nn}
     1 while s/(?<!\\)_(\s*([^{\\\s]|\\\w+))/\\SUBSCRIPTNB{$1}/g ;
     1 while s/(?<!\\)_(\s*{($pat_n)})/\\SUBSCRIPT$1/g ;
@@ -2351,17 +2530,20 @@
     s/(?<!\\)\\\[/\\begin{SQUAREBRACKET}/sg;
     s/\\\]/\\end{SQUAREBRACKET}/sg;
     # Convert all picture environmentent (\begin{PICTUREENV} .. \end{PICTUREENV} \PICTUREBLOCKenv
-    s/\\begin{($PICTUREENV)}(.*?)\\end{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
+    s/\\begin\{($PICTUREENV)}(.*?)\\end\{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
     #    For --block-math-markup option -convert all \begin{MATH} .. \end{MATH}
     #    into \MATHBLOCKMATH{...} commands, where MATH is any valid math environment
     #    Also convert all array environments into ARRAYBLOCK environments
+
     if ( $mathmarkup != FINE ) {
-      s/\\begin{($ARRENV)}(.*?)\\end{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
+      s/\\begin\{($ARRENV)}(.*?)\\end\{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
       
       take_comments_and_enter_from_frac();
       
-      s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/\\MATHBLOCK$1\{$2\}/sg;
+
+      s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\end\{\1\}/\\MATHBLOCK$1\{$2\}/sg;
     }
+
     # add final token " STOP"
     $_ .= " STOP"
   }
@@ -2470,6 +2652,7 @@
     # Replace \RIGHTBRACE by }    
     s/\\RIGHTBRACE/}/g;
 
+
     # Check all deleted blocks: where a deleted block contains a matching \begin and
     #    \end environment (these will be disabled by a %DIFDELCMD statements), enable
     #    these commands again (such that for example displayed math in a deleted equation
@@ -2561,15 +2744,17 @@
 #    bb. disable active labels within deleted blocks (as these are not safe commands, this should normally only
 #        happen within deleted maths blocks
       $delblock=~ s/(?<!$DELCMDOPEN)(\\$LABELCMD(?:${extraspace})\{(?:[^{}])*\}[\t ]*)\n?/${DELCMDOPEN}$1${DELCMDCLOSE}/smg ;
+      # previous line causes trouble as by issue #90 I might need to modify this
 
 
 #     c. If in-line math mode contains array environment, enclose the whole environment in \mbox'es
       while ( $delblock =~ m/($math)(\s*)/sg ) {
+#	      print STDERR "DEBUG Delblock Match math $& at ",pos,"\n";
 	$cnt2=0;
 	$len2=length($&);
 	$begin2=pos($delblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/ );
 	substr($delblock,$begin2,$len2)=$mathblock;
 	pos($delblock) = $begin2 + length($mathblock);
       }
@@ -2578,11 +2763,22 @@
 	# note additional \newline after command is omitted from output if right at the end of deleted block (otherwise a spurious empty line is generated)
 	$delblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+      ###  #     change included verbatim environments
+      ###  $delblock =~ s/\\DIFverb\{/\\DIFDIFdelverb\{/g;
+      ###  $delblock =~ s/\\DIFlstinline/\\DIFDIFdellstinline/g;
+      ###}
+      # Mark deleted verbose commands
+      $delblock =~ s/(${DELCMDOPEN}\\DIF((?:verb\*?|lstinline(?:\[$brat0\])?)\{([-\d]*?)\}\s*).*)$/%\n\\DIFDIFdel$2${AUXCMD}\n$1/gm;
 
 #     splice in modified delblock
       substr($_,$begin,$len)=$delblock;
       pos = $begin + length($delblock);
     }
+
+    ### print STDERR "<<<$_>>>\n" if $debug;
+
+
     # make the array modification in added blocks
     while ( m/\\DIFaddbegin.*?\\DIFaddend/sg ) {
       $cnt=0;
@@ -2594,7 +2790,7 @@
 	$len2=length($&);
 	$begin2=pos($addblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/) ;
 	substr($addblock,$begin2,$len2)=$mathblock;
 	pos($addblock) = $begin2 + length($mathblock);
       }
@@ -2603,6 +2799,11 @@
 	##$addblock=~s/($math)/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
 	$addblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+	# mark added verbatim commands
+      $addblock =~ s/\\DIFverb/\\DIFDIFaddverb/g;
+      $addblock =~ s/\\DIFlstinline/\\DIFDIFaddlstinline/g;
+      ###}
 #     splice in modified addblock
       substr($_,$begin,$len)=$addblock;
       pos = $begin + length($addblock);
@@ -2646,8 +2847,8 @@
     # environments) are between the \begin{$MATHENV} and \end{MATHMODE} commands. This is necessary as the minimal matching 
     # is not globally minimal but only 'locally' (matching is beginning from the left side of the string)
     if ( $mathmarkup == FINE ) {
-      1 while s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin{MATHMODE})))*?)\\end{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
-      1 while s/\\begin{MATHMODE}((?:.(?!\\end{MATHMODE}))*?)\\end{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
+      1 while s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end\{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin\{MATHMODE})))*?)\\end\{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
+      1 while s/\\begin\{MATHMODE}((?:.(?!\\end\{MATHMODE}))*?)\\end\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
       # convert remaining \begin{MATHMODE} \end{MATHMODE} (and not containing & or \\ )into MATHREPL environments
       s/\\begin\{MATHMODE\}((?:(.(?!(?<!\\)\&|\\\\))*)?)\\end\{MATHMODE\}/\\begin{$MATHREPL}$1\\end{$MATHREPL}/sg;
       # others into MATHARRREPL
@@ -2654,7 +2855,7 @@
       s/\\begin\{MATHMODE\}(.*?)\\end\{MATHMODE\}/\\begin{$MATHARRREPL}$1\\end{$MATHARRREPL}/sg;
 
       # now look for AUXCMD math-mode pairs which have only comments (or empty lines between them), and remove the added commands
-      s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
+      s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
     } else {
       #   math modes OFF,WHOLE,COARSE: Convert \MATHBLOCKmath{..} commands back to environments
       s/\\MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{($pat_n)\}/\\begin{$1}$2\\end{$1}/sg;
@@ -2702,13 +2903,8 @@
     s/^\Q${DELCMDOPEN}\E\n//msg;
 
     # Expand hashes of verb and verbatim environments (note negative look behind assertion to not leak out of DIFDELCMD comments
-    s/(\\verb\*?)\{([-\d]*?)\}/"${1}". fromhash(\%verbhash,$2)/esg;
-    s/${DELCMDOPEN}\\(verbatim\*?)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
-    s/\\(verbatim\*?)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
-    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
-    s/\\PERCENTAGE ?/\\%/g;
-    # Convert '\DOLLAR ' back into '\$'			       
-    s/\\DOLLAR /\\\$/g;
+    s/${DELCMDOPEN}\\($VERBATIMENV)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
+    s/\\($VERBATIMENV)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
     # remove all \PAR tokens (taking care to properly keep commented out PAR's
     # from introducing uncommented newlines - next line)
     s/(%DIF < )([^\n]*?)\\PAR\n/$1$2\n$1\n/sg;
@@ -2724,11 +2920,17 @@
       print STDERR "endfloat package detected.\n" if $verbose ;
       # eliminate whitespace before and after
       s/^(\s*)(\\(?:begin|end)\{(?:figure|table)\})(\s*)$/$2/mg;
-      # split lines with remaining characters before float enviroment conmmand
-      s/^(.+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
-      # split lines with remaining characters after float enviroment conmmand
-      s/(\\(?:begin|end)\{(?:figure|table)\})(.+)$/$1\n$2/mg;
+      # split lines with remaining characters before float environment conmmand
+      s/^([^%]+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
+      # split lines with remaining characters after float environment conmmand
+      s/^((?:[^%]+)\\(?:begin|end)\{(?:figure|table)\}(?:\[[a-zA-Z]+\])?)(.+)((?:%.*)?)$/$1\n$2$3/mg;
     }
+
+    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
+    s/\\PERCENTAGE ?/\\%/g;
+    # Convert '\DOLLAR ' back into '\$'			       
+    s/\\DOLLAR /\\\$/g;
+
     # undo renaming of the \begin and \end,{,}  and dollars in comments 
     1 while s/(%.*)DOLLARDIF/$1\$/mg ;
 #   Convert \begin{SQUAREBRACKET} \end{SQUAREBRACKET} into \[ \]
@@ -2754,6 +2956,21 @@
     s/\\QLEFTBRACE /\\\{/sg;
     s/\\QRIGHTBRACE /\\\}/sg;
     s/\\AMPERSAND /\\&/sg;
+    # Highligh added inline verbatim commands if possible
+    if ( $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/ )   { 
+      # wrap added verb commands with color commands
+      s/\\DIFDIFadd((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*)/\{\\color{blue}$AUXCMD\n\\DIF$1%\n\}$AUXCMD\n/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD)/\{\\color{red}${AUXCMD}\n\\DIF$1\n\}${AUXCMD}/sg;
+    } else {
+      # currently if colour markup is not used just remove the added mark
+      s/\\DIFDIFadd(verb\*?|lstinline)/\\DIF$1/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD\n)//sg;
+    }
+    # expand \verb and friends inline arguments
+    s/\\DIF((?:DIFadd|DIFdel)?(?:verb\*?|lstinline(?:\[$brat0\])?))\{([-\d]*?)\}/"\\${1}". fromhash(\%verbhash,$2)/esg;
+    # add basicstyle color{blue} to added lstinline commands
+    # finally add the comment to the ones not having an optional argument before
+    ###s/\\DIFaddlstinline(?!\[)/\\lstinline\n[basicstyle=\\color{blue}]$AUXCMD\n/g;
 
   return;
   }
@@ -2764,7 +2981,7 @@
 sub restore_item_commands {
   my ($string)=@_ ;
   my ($itemarg, at itemargs);
-  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*?(?:\n|$DELCMDCLOSE))/
+  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*((?:${cmdoptseq}\s*?)*)(?:\n|$DELCMDCLOSE))/
      # if \item has an []argument, then mark up the argument as deleted)
      if (length($4)>0) {
        # use substr to exclude square brackets at end points
@@ -2773,7 +2990,8 @@
      } else {
        $itemarg="";
      }
-     "$1$2$3$itemarg$AUXCMD\n"/sge;
+     "$1$2$3$itemarg$AUXCMD\n";  ###.((length($5)>0) ? "%DIFDELCMD $5 $DELCMDCLOSE\n" : "")
+     /sge;
   return($string);
 }
 
@@ -2956,16 +3174,32 @@
 # init_regex_arr_data(\@array,"TOKEN INIT")
 # scans DATA file handel for line "%% TOKEN INIT" line
 # then appends each line not beginning with % into array (as a quoted regex)
+# This is used for command lists and configuration variables, but the processing is slightly 
+# different: 
+# For lists, the regular expression is extended to include beginning (^) and end ($) markers, to require full-string matching
+# For configuration variables (and all others), simply an unadorned list is copied
 sub init_regex_arr_data {
   my ($arr,$token)=@_;
-  my ($copy);
+  my $copy=0;
+  my ($mode);
+  if ($token =~ m/COMMANDS/ ) { 
+    $mode=0;
+  } else {
+    $mode=1;
+  }
+
   while (<DATA>) {
     if ( m/^%%BEGIN $token\s*$/ ) {
-      $copy=1; }
-    elsif ( m/^%%END $token\s*/ )  {
+      $copy=1; 
+      next;
+    } elsif ( m/^%%END $token\s*$/ )  {
       last; }
     chomp;
-    push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    if ( $mode==0 ) {
+      push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    } elsif ($mode==1) {
+      push (@$arr,"$_") if ( $copy && !/^%/ ) ;
+    }
   }
   seek DATA,0,0;    # rewind DATA handle to file begin
 }
@@ -2978,25 +3212,42 @@
 # Otherwise treat arg as a comma separated list of regular expressions
 sub init_regex_arr_ext {
   my ($arr,$arg)=@_;
-  my $regex;
-  if ( -f $ arg ) {
-    open(FILE,"$arg") or die ("Couldn't open $arg: $!");
-    while (<FILE>) {
-      chomp;
-      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
-      push (@$arr,qr/^$_$/);
-    }
-    close(FILE);
+  if ( -f $arg ) {
+    init_regex_arr_file($arr,$arg);
+  } else {
+    init_regex_arr_list($arr,$arg);
   }
-  else {
-    # assume it is a comma-separated list of reg-ex
-    foreach $regex (split(qr/(?<!\\),/,$arg)) {
-      $regex =~ s/\\,/,/g;
-      push (@$arr,qr/^$regex$/);
-    }
+}
+
+# init_regex_arr_file(\@array,$fname)
+# appends array with regular expressions.
+# Read in list of regular expressions from $fname
+# (one expression per line)
+sub init_regex_arr_file {
+  my ($arr,$fname)=@_;
+  open(FILE,"$fname") or die ("Couldn't open $fname: $!");
+  while (<FILE>) {
+    chomp;
+    next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+    push (@$arr,qr/^$_$/);
   }
+  close(FILE);
 }
 
+# init_regex_arr_list(\@array,$arg)
+# appends array with regular expressions.
+# read from comma separated list of regular expressions ($arg)
+sub init_regex_arr_list {
+  my ($arr,$arg)=@_;
+  my $regex;
+  ###    print STDERR "DEBUG init_regex_arr_list arg >$arg<\n" if $debug;
+  foreach $regex (split(qr/(?<!\\),/,$arg)) {
+    $regex =~ s/\\,/,/g;
+    push (@$arr,qr/^$regex$/);
+  }
+}
+
+
 #exetime() returns time since last execution of this command
 #exetime(1) resets this time
 my $lasttime=-1;   # global variable for persistence
@@ -3138,6 +3389,14 @@
                           SCALEDELGRAPHICS (Float)
                        This option can be repeated.
 
+--add-to-config  varenv1=pattern1,varenv2=pattern2
+                       For configuration variables containing a regular expression (essentially those ending
+                       in ENV, and COUNTERCMD) this provides an alternative way to modify the configuration 
+                       variables. Instead of setting the complete pattern, with this option it is possible to add an
+                       alternative pattern. varenv must be one of the variables listed above that take a regular
+                       expression as argument, and pattern is any regular expression (which might need to be 
+                       protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+                       to separate them, e.g. --add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote"
 
 --packages=pkg1,pkg2,..
                        Tell latexdiff that .tex file is processed with the packages in list 
@@ -3556,11 +3815,12 @@
 =item B<--replace-context1cmd=replace-file> 
 
 =item B<--append-context1cmd=append-file> or
+
 =item B<--append-context1cmd="cmd1,cmd2,...">
 
 Replace or append to the list of regex matching commands
 whose last argument is text but which require a particular
-context to work, e.g. \caption will only work within a figure
+context to work, e.g. C<\caption> will only work within a figure
 or table.  These commands behave like text commands, except when 
 they occur in a deleted section, when they are disabled, but their
 argument is shown as deleted text.
@@ -3568,6 +3828,7 @@
 =item B<--replace-context2cmd=replace-file> 
 
 =item B<--append-context2cmd=append-file> or
+
 =item B<--append-context2cmd="cmd1,cmd2,...">
 
 As corresponding commands for context1.  The only difference is that
@@ -3580,7 +3841,7 @@
 =item B<--append-mboxsafecmd=append-file> or B<--append-mboxsafecmd="cmd1,cmd2,...">
 
 Define safe commands, which additionally need to be protected by encapsulating
-in an \\mbox{..}. This is sometimes needed to get around incompatibilities 
+in an C<\mbox{..}>. This is sometimes needed to get around incompatibilities 
 between external packages and the ulem package, which is  used for highlighting
 in the default style UNDERLINE as well as CULINECHBAR CFONTSTRIKE
 
@@ -3620,6 +3881,16 @@
 
 C<SCALEDELGRAPHICS> (Float)
 
+=item B<--add-to-config varenv1=pattern1,varenv2=pattern2,...>
+
+For configuration variables, which are a regular expression (essentially those ending
+in ENV, and COUNTERCMD, see list above) this provides an alternative way to modify the configuration 
+variables. Instead of setting the complete pattern, with this option it is possible to add an
+alternative pattern. C<varenv> must be one of the variables listed above that take a regular
+expression as argument, and pattern is any regular expression (which might need to be 
+protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+to separate them, e.g. C<--add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote">
+
 =item B<--show-safecmd>
 
 Print list of RegEx matching and excluding safe commands.
@@ -3680,16 +3951,17 @@
 =item B<--graphics-markup=level>
 
  Change highlight style for graphics embedded with C<\includegraphics> commands.
-                      Possible values for level:
 
-                      C<none, C<off> or C<0>: no highlighting for figures
+Possible values for level:
 
-                      C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
+C<none>, C<off> or C<0>: no highlighting for figures
 
-                      C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
-                                scale, and crossed out with a red diagonal cross. Use configuration
-                                variable SCALEDELGRAPHICS to set size of deleted figures.
+C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
 
+C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
+scale, and crossed out with a red diagonal cross. Use configuration
+variable SCALEDELGRAPHICS to set size of deleted figures.
+
 Note that changes to the optional parameters will make the figure appear as changed 
 to latexdiff, and this figure will thus be highlighted
 
@@ -3745,7 +4017,7 @@
 
 Suppress inclusion of old and new file names as comment in output file
 
-=item B<--visble-label>
+=item B<--visible-label>
 
 Include old and new filenames (or labels set with C<--label> option) as 
 visible output.
@@ -4054,8 +4326,8 @@
 
 =head1 AUTHOR
 
-Version 1.2.0
-Copyright (C) 2004-2015 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2004-2017 Frederik Tilmann
 
 This program is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License Version 3
@@ -4337,9 +4609,68 @@
 institute
 %%END CONTEXT2 COMMANDS 
 
+%% CONFIGURATION variabe defaults
+%%BEGIN LISTENV CONFIG
+itemize
+description
+enumerate
+%%END LISTENV CONFIG
 
-%% TYPES (Commands for highlighting changed blocks)
+%%BEGIN FLOATENV CONFIG
+figure[\w\d*@]*
+table[\w\d*@]*
+plate[\w\d*@]*
+%%END FLOATENV CONFIG
 
+%%BEGIN PICTUREENV CONFIG
+picture[\w\d*@]*
+tikzpicture[\w\d*@]*
+DIFnomarkup
+%%END PICTUREENV CONFIG
+
+%%BEGIN MATHENV CONFIG
+equation[*]?
+displaymath
+DOLLARDOLLAR
+%%END MATHENV CONFIG
+
+%%BEGIN MATHARRENV CONFIG
+eqnarray[*]?
+align[*]?
+alignat[*]?
+gather[*]?
+multline[*]?
+flalign[*]?
+%%END MATHARRENV CONFIG
+
+%%BEGIN ARRENV CONFIG
+aligned
+array
+[pbvBV]?matrix
+smallmatrix
+cases
+split
+%%END ARRENV CONFIG
+
+%%BEGIN COUNTERCMD CONFIG
+footnote
+part
+chapter
+section
+subsection
+subsubsection
+paragraph
+subparagraph
+%%END COUNTERCMD CONFIG
+
+%%BEGIN VERBATIMENV CONFIG
+verbatim[*]?
+lstlisting
+comment
+%%END VERBATIMENV CONFIG
+
+%%% TYPES (Commands for highlighting changed blocks)
+
 %DIF UNDERLINE PREAMBLE
 \RequirePackage[normalem]{ulem}
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
@@ -4350,6 +4681,7 @@
 %DIF CTRADITIONAL PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} [..\footnote{removed: #1} ]}}
 %DIF END CTRADITIONAL PREAMBLE
@@ -4356,6 +4688,7 @@
 
 %DIF TRADITIONAL PREAMBLE
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{[..\footnote{removed: #1} ]}}
 %DIF END TRADITIONAL PREAMBLE
@@ -4362,6 +4695,7 @@
 
 %DIF CFONT PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} \scriptsize #1}}
 %DIF END CFONT PREAMBLE
@@ -4368,6 +4702,7 @@
 
 %DIF FONTSTRIKE PREAMBLE
 \RequirePackage[normalem]{ulem}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{\footnotesize \sout{#1}}}
 %DIF END FONTSTRIKE PREAMBLE
@@ -4406,6 +4741,7 @@
 %DIF END INVISIBLE PREAMBLE
 
 %DIF BOLD PREAMBLE
+\DeclareOldFontCommand{\bf}{\normalfont\bfseries}{\mathbf}
 \providecommand{\DIFadd}[1]{{\bf #1}}
 \providecommand{\DIFdel}[1]{}
 %DIF END BOLD PREAMBLE
@@ -4430,7 +4766,7 @@
 \providecommand{\DIFaddend}{\protect\marginpar{]}}
 \providecommand{\DIFdelbegin}{\protect\marginpar{d[}}
 \providecommand{\DIFdelend}{\protect\marginpar{]}}
-%DIF END BRACKET PREAMBLE
+%DIF END MARGIN PREAMBLE
 
 %DIF DVIPSCOL PREAMBLE
 %Note: only works with dvips converter
@@ -4623,4 +4959,18 @@
 %DIF HYPERREF PREAMBLE
 \providecommand{\DIFadd}[1]{\texorpdfstring{\DIFaddtex{#1}}{#1}}
 \providecommand{\DIFdel}[1]{\texorpdfstring{\DIFdeltex{#1}}{}}
-%DIF END HYPERREF PACKAGE
+%DIF END HYPERREF PREAMBLE
+
+%DIF LISTINGS PREAMBLE
+\lstdefinelanguage{codediff}{
+  moredelim=**[is][\color{red}]{*!----}{----!*},
+  moredelim=**[is][\color{blue}]{*!++++}{++++!*}
+}
+\lstdefinestyle{codediff}{
+	belowcaptionskip=.25\baselineskip,
+	language=codediff,
+	basicstyle=\ttfamily,
+	columns=fullflexible,
+	keepspaces=true,
+}
+%DIF END LISTINGS PREAMBLE

Modified: trunk/Master/texmf-dist/doc/man/man1/latexdiff-vc.1
===================================================================
--- trunk/Master/texmf-dist/doc/man/man1/latexdiff-vc.1	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/doc/man/man1/latexdiff-vc.1	2017-06-26 21:22:49 UTC (rev 44704)
@@ -133,7 +133,7 @@
 .\" ========================================================================
 .\"
 .IX Title "LATEXDIFF-VC 1"
-.TH LATEXDIFF-VC 1 "2016-08-15" "perl v5.18.2" " "
+.TH LATEXDIFF-VC 1 "2017-06-22" "perl v5.18.2" " "
 .\" For nroff, turn off justification.  Always turn off hyphenation; it makes
 .\" way too many mistakes in technical documents.
 .if n .ad l
@@ -207,6 +207,52 @@
 The generic usage of this function is : \f(CW\*(C`latexdiff\-vc \-\-flatten \-r rev1 [\-r rev2] master.tex\*(C'\fR where master.tex is the project file containing the highest level of includes etc.
 .Sp
 With \f(CW\*(C`\-\-flatten=keep\-intermediate\*(C'\fR, the intermediate revision snapshots are kept in the current directory (Default is to store them in a temporary directory and delete them after generating the diff file.)
+.IP "\fB\-\-config var1=val1,var2=val2,...\fR or \fB\-c var1=val1,..\fR" 4
+.IX Item "--config var1=val1,var2=val2,... or -c var1=val1,.."
+.PD 0
+.IP "\fB\-\-only\-changes\fR" 4
+.IX Item "--only-changes"
+.PD
+Post-process the output such that only pages with changes on them are displayed. This requires the use of subtype \s-1ZLABEL \s0
+in latexdiff, which will be set automatically, but any manually set \-s option will be overruled (also requires zref package to 
+be installed). (note that this option must be combined with \-\-ps or \-\-pdf to make sense)
+.IP "\fB\-\-force\fR" 4
+.IX Item "--force"
+Overwrite existing diff files without asking for confirmation. Default 
+behaviour is to ask for confirmation before overwriting an existing difference
+file.
+.IP "\fB\-\-run\fR" 4
+.IX Item "--run"
+run latex command on diff file after generation of diff file.
+.IP "\fB\-\-dvi\fR" 4
+.IX Item "--dvi"
+run latex and dvixxx commands after generation of diff file.
+.IP "\fB\-c configfile\fR" 4
+.IX Item "-c configfile"
+Set configuration variables for latexdiff and latexdiff-vc.  The option can be repeated to set different
+variables (as an alternative to the comma-separated list).
+Available variables for latexdiff-vc:
+.RS 4
+.ie n .IP """LATEXDIFF"" latexdiff command (e.g. latexdiff-fast, latexdiff-so). This command should support the option ""\-\-interaction=batchmode""" 8
+.el .IP "\f(CWLATEXDIFF\fR latexdiff command (e.g. latexdiff-fast, latexdiff-so). This command should support the option \f(CW\-\-interaction=batchmode\fR" 8
+.IX Item "LATEXDIFF latexdiff command (e.g. latexdiff-fast, latexdiff-so). This command should support the option --interaction=batchmode"
+.PD 0
+.ie n .IP """LATEX"" latex command (e.g. pdflatex, lualatex)" 8
+.el .IP "\f(CWLATEX\fR latex command (e.g. pdflatex, lualatex)" 8
+.IX Item "LATEX latex command (e.g. pdflatex, lualatex)"
+.ie n .IP """DVI2""  Command for conversion of dvi file (e.g. dvips, dvipdf)" 8
+.el .IP "\f(CWDVI2\fR  Command for conversion of dvi file (e.g. dvips, dvipdf)" 8
+.IX Item "DVI2 Command for conversion of dvi file (e.g. dvips, dvipdf)"
+.ie n .IP """BIBTEX"" Command replacing bibtex" 8
+.el .IP "\f(CWBIBTEX\fR Command replacing bibtex" 8
+.IX Item "BIBTEX Command replacing bibtex"
+.RE
+.RS 4
+.PD
+.Sp
+All other config variables are passed to latexdiff. Explicity set configuration changes always override implicit
+changes by the following shortcut options \fB\-\-fast\fR, \fB\-\-so\fR, \fB\-\-ps\fR and \fB\-\-pdf\fR.
+.RE
 .IP "\fB\-\-fast\fR or \fB\-\-so\fR" 4
 .IX Item "--fast or --so"
 Use \f(CW\*(C`latexdiff\-fast\*(C'\fR or \f(CW\*(C`latexdiff\-so\*(C'\fR, respectively (instead of \f(CW\*(C`latexdiff\*(C'\fR).
@@ -223,16 +269,11 @@
 Generate pdf output from difference file using \f(CW\*(C`pdflatex\*(C'\fR. This will
 run the sequence \f(CW\*(C`pdflatex; pdflatex\*(C'\fR on the difference file, or
 \&\f(CW\*(C`pdflatex; bibtex; pdflatex; pdflatex\*(C'\fR for files requiring bibtex.
-.IP "\fB\-\-only\-changes\fR" 4
-.IX Item "--only-changes"
-Post-process the output such that only pages with changes on them are displayed. This requires the use of subtype \s-1ZLABEL \s0
-in latexdiff, which will be set automatically, but any manually set \-s option will be overruled (also requires zref package to 
-be installsed). (note that this option must be combined with \-\-ps or \-\-pdf to make sense)
-.IP "\fB\-\-force\fR" 4
-.IX Item "--force"
-Overwrite existing diff files without asking for confirmation. Default 
-behaviour is to ask for confirmation before overwriting an existing difference
-file.
+Note that this is not just a shortcut for setting configuration variable but also triggers 
+some special behaviour.
+.IP "\fB\-\-show\-config\fR" 4
+.IX Item "--show-config"
+Show values of configuration variables.
 .IP "\fB\-\-help\fR or \fB\-h\fR" 4
 .IX Item "--help or -h"
 Show help text
@@ -257,8 +298,8 @@
 (option \f(CW\*(C`\-\-version\*(C'\fR).
 .SH "AUTHOR"
 .IX Header "AUTHOR"
-Version 1.2.0
-Copyright (C) 2005\-2016 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2005\-2017 Frederik Tilmann
 .PP
 This program is free software; you can redistribute it and/or modify
 it under the terms of the \s-1GNU\s0 General Public License Version 3

Modified: trunk/Master/texmf-dist/doc/man/man1/latexdiff-vc.man1.pdf
===================================================================
(Binary files differ)

Modified: trunk/Master/texmf-dist/doc/man/man1/latexdiff.1
===================================================================
--- trunk/Master/texmf-dist/doc/man/man1/latexdiff.1	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/doc/man/man1/latexdiff.1	2017-06-26 21:22:49 UTC (rev 44704)
@@ -133,7 +133,7 @@
 .\" ========================================================================
 .\"
 .IX Title "LATEXDIFF 1"
-.TH LATEXDIFF 1 "2016-08-15" "perl v5.18.2" " "
+.TH LATEXDIFF 1 "2017-06-22" "perl v5.18.2" " "
 .\" For nroff, turn off justification.  Always turn off hyphenation; it makes
 .\" way too many mistakes in technical documents.
 .if n .ad l
@@ -416,13 +416,15 @@
 .IP "\fB\-\-replace\-context1cmd=replace\-file\fR" 4
 .IX Item "--replace-context1cmd=replace-file"
 .PD 0
-.ie n .IP "\fB\-\-append\-context1cmd=append\-file\fR or =item \fB\-\-append\-context1cmd=""cmd1,cmd2,...""\fR" 4
-.el .IP "\fB\-\-append\-context1cmd=append\-file\fR or =item \fB\-\-append\-context1cmd=``cmd1,cmd2,...''\fR" 4
-.IX Item "--append-context1cmd=append-file or =item --append-context1cmd=cmd1,cmd2,..."
+.IP "\fB\-\-append\-context1cmd=append\-file\fR or" 4
+.IX Item "--append-context1cmd=append-file or"
+.ie n .IP "\fB\-\-append\-context1cmd=""cmd1,cmd2,...""\fR" 4
+.el .IP "\fB\-\-append\-context1cmd=``cmd1,cmd2,...''\fR" 4
+.IX Item "--append-context1cmd=cmd1,cmd2,..."
 .PD
 Replace or append to the list of regex matching commands
 whose last argument is text but which require a particular
-context to work, e.g. \ecaption will only work within a figure
+context to work, e.g. \f(CW\*(C`\ecaption\*(C'\fR will only work within a figure
 or table.  These commands behave like text commands, except when 
 they occur in a deleted section, when they are disabled, but their
 argument is shown as deleted text.
@@ -429,9 +431,11 @@
 .IP "\fB\-\-replace\-context2cmd=replace\-file\fR" 4
 .IX Item "--replace-context2cmd=replace-file"
 .PD 0
-.ie n .IP "\fB\-\-append\-context2cmd=append\-file\fR or =item \fB\-\-append\-context2cmd=""cmd1,cmd2,...""\fR" 4
-.el .IP "\fB\-\-append\-context2cmd=append\-file\fR or =item \fB\-\-append\-context2cmd=``cmd1,cmd2,...''\fR" 4
-.IX Item "--append-context2cmd=append-file or =item --append-context2cmd=cmd1,cmd2,..."
+.IP "\fB\-\-append\-context2cmd=append\-file\fR or" 4
+.IX Item "--append-context2cmd=append-file or"
+.ie n .IP "\fB\-\-append\-context2cmd=""cmd1,cmd2,...""\fR" 4
+.el .IP "\fB\-\-append\-context2cmd=``cmd1,cmd2,...''\fR" 4
+.IX Item "--append-context2cmd=cmd1,cmd2,..."
 .PD
 As corresponding commands for context1.  The only difference is that
 context2 commands are completely disabled in deleted sections, including
@@ -445,7 +449,7 @@
 .IX Item "--append-mboxsafecmd=append-file or --append-mboxsafecmd=cmd1,cmd2,..."
 .PD
 Define safe commands, which additionally need to be protected by encapsulating
-in an \e\embox{..}. This is sometimes needed to get around incompatibilities 
+in an \f(CW\*(C`\embox{..}\*(C'\fR. This is sometimes needed to get around incompatibilities 
 between external packages and the ulem package, which is  used for highlighting
 in the default style \s-1UNDERLINE\s0 as well as \s-1CULINECHBAR CFONTSTRIKE\s0
 .IP "\fB\-\-config var1=val1,var2=val2,...\fR or \fB\-c var1=val1,..\fR" 4
@@ -481,6 +485,15 @@
 \&\f(CW\*(C`PICTUREENV\*(C'\fR (RegEx)
 .Sp
 \&\f(CW\*(C`SCALEDELGRAPHICS\*(C'\fR (Float)
+.IP "\fB\-\-add\-to\-config varenv1=pattern1,varenv2=pattern2,...\fR" 4
+.IX Item "--add-to-config varenv1=pattern1,varenv2=pattern2,..."
+For configuration variables, which are a regular expression (essentially those ending
+in \s-1ENV,\s0 and \s-1COUNTERCMD,\s0 see list above) this provides an alternative way to modify the configuration 
+variables. Instead of setting the complete pattern, with this option it is possible to add an
+alternative pattern. \f(CW\*(C`varenv\*(C'\fR must be one of the variables listed above that take a regular
+expression as argument, and pattern is any regular expression (which might need to be 
+protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+to separate them, e.g. \f(CW\*(C`\-\-add\-to\-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote"\*(C'\fR
 .IP "\fB\-\-show\-safecmd\fR" 4
 .IX Item "--show-safecmd"
 Print list of RegEx matching and excluding safe commands.
@@ -530,19 +543,20 @@
 expected, e.g. correction of typos.
 .IP "\fB\-\-graphics\-markup=level\fR" 4
 .IX Item "--graphics-markup=level"
-.Vb 2
+.Vb 1
 \& Change highlight style for graphics embedded with C<\eincludegraphics> commands.
-\&                      Possible values for level:
-\&
-\&                      C<none, C<off> or C<0>: no highlighting for figures
-\&
-\&                      C<new\-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
-\&
-\&                      C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
-\&                                scale, and crossed out with a red diagonal cross. Use configuration
-\&                                variable SCALEDELGRAPHICS to set size of deleted figures.
 .Ve
 .Sp
+Possible values for level:
+.Sp
+\&\f(CW\*(C`none\*(C'\fR, \f(CW\*(C`off\*(C'\fR or \f(CW0\fR: no highlighting for figures
+.Sp
+\&\f(CW\*(C`new\-only\*(C'\fR or \f(CW1\fR: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
+.Sp
+\&\f(CW\*(C`both\*(C'\fR or \f(CW2\fR:     highlight new figures with a blue frame and show deleted figures at reduced 
+scale, and crossed out with a red diagonal cross. Use configuration
+variable \s-1SCALEDELGRAPHICS\s0 to set size of deleted figures.
+.Sp
 Note that changes to the optional parameters will make the figure appear as changed 
 to latexdiff, and this figure will thus be highlighted
 .IP "\fB\-\-disable\-citation\-markup\fR or \fB\-\-disable\-auto\-mbox\fR" 4
@@ -585,8 +599,8 @@
 .IP "\fB\-\-no\-label\fR" 4
 .IX Item "--no-label"
 Suppress inclusion of old and new file names as comment in output file
-.IP "\fB\-\-visble\-label\fR" 4
-.IX Item "--visble-label"
+.IP "\fB\-\-visible\-label\fR" 4
+.IX Item "--visible-label"
 Include old and new filenames (or labels set with \f(CW\*(C`\-\-label\*(C'\fR option) as 
 visible output.
 .IP "\fB\-\-flatten\fR" 4
@@ -849,8 +863,8 @@
 \&\fIlatexdiff-fast\fR requires the \fIdiff\fR command to be present.
 .SH "AUTHOR"
 .IX Header "AUTHOR"
-Version 1.2.0
-Copyright (C) 2004\-2015 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2004\-2017 Frederik Tilmann
 .PP
 This program is free software; you can redistribute it and/or modify
 it under the terms of the \s-1GNU\s0 General Public License Version 3

Modified: trunk/Master/texmf-dist/doc/man/man1/latexdiff.man1.pdf
===================================================================
(Binary files differ)

Modified: trunk/Master/texmf-dist/doc/man/man1/latexrevise.man1.pdf
===================================================================
(Binary files differ)

Modified: trunk/Master/texmf-dist/doc/support/latexdiff/README
===================================================================
--- trunk/Master/texmf-dist/doc/support/latexdiff/README	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/doc/support/latexdiff/README	2017-06-26 21:22:49 UTC (rev 44704)
@@ -5,7 +5,9 @@
   Various options are available for visual markup using standard latex 
 packages such as "color.sty". Changes not directly affecting visible 
 text, for example in formatting commands, are still marked in 
-the latex source.
+the latex source. Note that only files conforming to latex syntax will 
+be processed correctly, not generic TeX files. Some further 
+minor restrictions apply, see documentation.
  
 A rudimentary revision facilility is provided by another Perl script,
 latexrevise, which accepts or rejects all changes.  Manual

Modified: trunk/Master/texmf-dist/doc/support/latexdiff/doc/latexdiff-man.pdf
===================================================================
(Binary files differ)

Modified: trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff
===================================================================
--- trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff	2017-06-26 21:22:49 UTC (rev 44704)
@@ -24,6 +24,28 @@
 #
 # ToDo:
 #
+# Version 1.2.1 (22 June 2017)
+#    - add "DeclareOldFontCommand" to styles using \bf or \sf old style font commands (fixies issue #92 )
+#    - improved markup: process lstinline commands in listings package correctly
+#      for styles using colour, \verb and \lstinline arguments are marked up with colour (blue for added, red for deleted)
+#    - bug fix: protecting inline math expressions for mbox did not work as intended (see stack exchange question: http://tex.stackexchange.com/questions/359412/compiling-the-latexdiff-when-adding-a-subscript-before-a-pmatrix-environment-cau)
+#    - bug fix: when deleted \item commands are followed immediately by unsafe commands, they were not restored properly
+#      (thanks to J. Protze for pull request) (pull request #89)
+#    - treat lstlisting and comment as equivalent to verbatim environment
+#      make environments that are treated like verbatim environments configurable (config variable VERBATIMENV)
+#      treat lstinlne as equivalent to verb command
+#      partially addresses issue #38
+#    - refactoring: set default configuration variables in a hash, and those that correspond to lists 
+#    - feature: option --add-to-config used to amend configuration variables, which are regex pattern lists
+#    - bug fix: deleted figures when endfloat package is activated
+#    - bug fix: alignat environment now always processed correctly (fix issues #65)
+#    - bug fix: avoid processing of commands as potential files in routine init_regex_arr (fix issue #70 )
+#    - minimal feature enhancement: treat '@' as allowed character in commands (strictly speaking requires prior \makeatletter statement, but always assuming it to be 
+#       @       a letter if it is part of a command name will usually lead to the correct behaviour (see http://tex.stackexchange.com/questions/346651/latexdiff-and-let)
+#    - new feature/bug fix: --flatten option \endinput in included files now respected but only if \endinput stands right at the beginning of the line (issue #77)
+#    - bug fix: flatten would incorrectly attempt to process commented out \include commands (from discussion in issue #77 )
+#    - introduce an invisible space (\hspace{0pt} after \mbox{..} auxiliary commands (not in math mode), to allow line breaks between added and deleted citations (change should not cause adverse behaviour otherwise)
+#
 # Version 1.2.0:
 #    - highlight new and deleted figures
 #    - bug fix in title mark-up. Previously deleted commands in title (such as \title, \author or \date) were marked up erroneously
@@ -140,34 +162,76 @@
 
 
 my ($versionstring)=<<EOF ;
-This is LATEXDIFF 1.2.0  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
-  (c) 2004-2016 F J Tilmann
+This is LATEXDIFF 1.2.1  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
+  (c) 2004-2017 F J Tilmann
 EOF
 
-# Configuration variables: these have to be visible from the subroutines
-my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
-                     # shorter identical blocks will be merged to the previous word
-my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
-                      # this is only used for --graphics-markup=BOTH option
-my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
-my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
-my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
-my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
-my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
-my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
-my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
-my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+# Hash with defaults for configuration variables. These marked undef have default values constructed from list defined in the DATA block
+# (under tag CONFIG)
+my %CONFIG=(
+   MINWORDSBLOCK => 3, # minimum number of tokens to form an independent block
+                        # shorter identical blocks will be merged to the previous word
+   SCALEDELGRAPHICS => 0.5, # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+                             # this is only used for --graphics-markup=BOTH option
+   FLOATENV => undef ,   # Environments in which FL variants of defined commands are used
+   PICTUREENV => undef ,   # Environments in which all change markup is removed
+   MATHENV => undef ,           # Environments turning on display math mode (code also knows about \[ and \])
+   MATHREPL => 'displaymath',  # Environment introducing deleted maths blocks
+   MATHARRENV => undef ,           # Environments turning on eqnarray math mode
+   MATHARRREPL => 'eqnarray*',  # Environment introducing deleted maths blocks
+   ARRENV => undef , # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+   COUNTERCMD => undef,
+                                        # COUNTERCMD textcmds which are associated with a counter
                                         # If any of these commands occur in a deleted block
                                         # they will be succeeded by an \addtocounter{...}{-1}
                                         # for the associated counter such that the overall numbers
                                         # should be the same as in the new file
+   LISTENV => undef ,  # list making environments - they will generally be kept
+   VERBATIMENV => undef,    # Environments whose content should be treated as verbatim text 
+   ITEMCMD => 'item'                    # command marking item in a list environment
+);
+# Configuration variables: these have to be visible from the subroutines
+my ($ARRENV,
+    $COUNTERCMD,
+    $FLOATENV,
+    $ITEMCMD,
+    $LISTENV,
+    $MATHARRENV,
+    $MATHARRREPL,
+    $MATHENV,
+    $MATHREPL,
+    $MINWORDSBLOCK,
+    $PICTUREENV,
+    $SCALEDELGRAPHICS,
+    $VERBATIMENV
+);
+
+# my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
+#                      # shorter identical blocks will be merged to the previous word
+# my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+#                       # this is only used for --graphics-markup=BOTH option
+# my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
+# my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
+# my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
+# my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
+# my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
+# my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
+# my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+# my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+#                                         # If any of these commands occur in a deleted block
+#                                         # they will be succeeded by an \addtocounter{...}{-1}
+#                                         # for the associated counter such that the overall numbers
+#                                         # should be the same as in the new file
+# my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
+# my $ITEMCMD='item';   # command marking item in a list environment
+
+
+
 my $LABELCMD='(?:label)';                # matching commands are disabled within deleted blocks - mostly useful for maths mode, as otherwise it would be fine to just not add those to SAFECMDLIST
 my @UNSAFEMATHCMD=('qedhere');           # Commands which are definitely unsafe for marking up in math mode (amsmath qedhere only tested to not work with UNDERLINE markup) (only affects WHOLE and COARSE math markup modes). Note that unlike text mode (or FINE math mode0 deleted unsafe commands are not deleted but simply taken outside \DIFdel
 my $MBOXINLINEMATH=0; # if set to 1 then surround marked-up inline maths expression with \mbox ( to get around compatibility
                       # problems between some maths packages and ulem package
 
-my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
-my $ITEMCMD='item';
 
 # Markup strings
 # If at all possible, do not change these as parts of the program
@@ -217,7 +281,8 @@
 
 my ($i,$j,$l);
 my ($old,$new);
-my ($line);
+my ($line,$key);
+my (@dumlist);
 my ($newpreamble,$oldpreamble);
 my (@newpreamble, at oldpreamble, at diffpreamble, at diffbody);
 my ($latexdiffpreamble);
@@ -252,7 +317,7 @@
 
 my ($mboxcmd);
 
-my (@configlist, at labels,
+my (@configlist, at addtoconfiglist, at labels,
     @appendsafelist, at excludesafelist,
     @appendmboxsafelist, at excludemboxsafelist,
     @appendtextlist, at excludetextlist,
@@ -294,6 +359,7 @@
 	   'subtype|s=s' => \$subtype, 
 	   'floattype|f=s' => \$floattype, 
 	   'config|c=s' => \@configlist,
+	   'add-to-config|c=s' => \@addtoconfiglist,
 	   'preamble|p=s' => \$preamblefile,
 	   'encoding|e=s' => \$encoding,
 	   'label|L=s' => \@labels,
@@ -460,21 +526,64 @@
 }
 foreach $assign ( @config ) {
   $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
-  if ( $1 eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $2; }
-  elsif ( $1 eq "FLOATENV" ) { $FLOATENV = $2 ; }
-  elsif ( $1 eq "ITEMCMD" ) { $ITEMCMD = $2 ; }
-  elsif ( $1 eq "LISTENV" ) { $LISTENV = $2 ; }
-  elsif ( $1 eq "PICTUREENV" ) { $PICTUREENV = $2 ; }
-  elsif ( $1 eq "MATHENV" ) { $MATHENV = $2 ; }
-  elsif ( $1 eq "MATHREPL" ) { $MATHREPL = $2 ; }
-  elsif ( $1 eq "MATHARRENV" ) { $MATHARRENV = $2 ; }
-  elsif ( $1 eq "MATHARRREPL" ) { $MATHARRREPL = $2 ; }
-  elsif ( $1 eq "ARRENV" ) { $ARRENV = $2 ; }
-  elsif ( $1 eq "COUNTERCMD" ) { $COUNTERCMD = $2 ; }
-  elsif ( $1 eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $2 ; }
-  else { die "Unknown variable $1 in assignment.";}
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}=$2;
 }
 
+my @addtoconfig=();
+foreach $config ( @addtoconfiglist ) {
+  if (-f $config || lc $config eq '/dev/null' ) {
+    open(FILE,$config) or die ("Couldn't open addd-to-config file $config: $!");
+    while (<FILE>) {
+      chomp;
+      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+      push (@addtoconfig,$_);
+    }
+    close(FILE);
+  }
+  else {
+#    foreach ( split(",",$config) ) {
+#      push @addtoconfig,$_;
+#    }
+     push @addtoconfig,split(",",$config)
+  }
+}
+
+# initialise default lists from DATA
+# for those configuration variables, which have not been set explicitly, initiate from list in document
+foreach $key ( keys(%CONFIG) ) {
+  if (!defined $CONFIG{$key}) {
+    @dumlist=();
+    init_regex_arr_data(\@dumlist,"$key CONFIG");
+    $CONFIG{$key}=join(";", at dumlist)
+  }
+}
+
+
+foreach $assign ( @addtoconfig ) {
+  $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}.=";$2";
+}
+
+# Map from hash to variables (we do this to have more concise code later, change from comma-separeted list)
+foreach  ( keys(%CONFIG) ) {
+  if ( $_ eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $CONFIG{$_}; }
+  elsif ( $_ eq "FLOATENV" ) { $FLOATENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "ITEMCMD" ) { $ITEMCMD = $CONFIG{$_} ; }
+  elsif ( $_ eq "LISTENV" ) { $LISTENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "PICTUREENV" ) { $PICTUREENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHENV" ) { $MATHENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHREPL" ) { $MATHREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "MATHARRENV" ) { $MATHARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHARRREPL" ) { $MATHARRREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "ARRENV" ) { $ARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "VERBATIMENV" ) { $VERBATIMENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "COUNTERCMD" ) { $COUNTERCMD = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $CONFIG{$_} ; }
+  else { die "Unknown configuration variable $_.";}
+}
+
 if ( $mathmarkup == COARSE || $mathmarkup == WHOLE ) {
   push(@MATHTEXTCMDLIST,qr/^MATHBLOCK(?:$MATHENV|$MATHARRENV|SQUAREBRACKET)$/);
 }
@@ -487,46 +596,11 @@
 
 
 
-if ($showpreamble) {
-  print "\nPreamble commands:\n";
-  print $latexdiffpreamble ;
+if ($showconfig || $showtext || $showsafe || $showpreamble) {
+  show_configuration();
+  exit 0; 
 }
 
-if ($showsafe) {
-  print "\nCommands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
-  print_regex_arr(@SAFECMDLIST);
-  print "\nCommands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
-  print_regex_arr(@SAFECMDEXCL);
-}
-
-if ($showtext) {
-  print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
-  print_regex_arr(@TEXTCMDLIST);
-  print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
-  print_regex_arr(@CONTEXT1CMDLIST);
-  print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
-  print_regex_arr(@CONTEXT2CMDLIST);  
-  print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
-  print_regex_arr(@TEXTCMDEXCL);
-}
-
-
-if ($showconfig) {
-  print "Configuration variables:\n";
-  print "ARRENV=$ARRENV\n";
-  print "COUNTERCMD=$COUNTERCMD\n";
-  print "FLOATENV=$FLOATENV\n";
-  print "ITEMCMD=$ITEMCMD\n";
-  print "LISTENV=$LISTENV\n";
-  print "MATHARRENV=$MATHARRENV\n";
-  print "MATHARRREPL=$MATHARRREPL\n";
-  print "MATHENV=$MATHENV\n";
-  print "MATHREPL=$MATHREPL\n";
-  print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
-  print "PICTUREENV=$PICTUREENV\n";
-}
-if ($showconfig || $showtext || $showsafe || $showpreamble) {
-  exit 0; }
 if ( @ARGV != 2 ) { 
   print STDERR "2 and only 2 non-option arguments required.  Write latexdiff -h to get help\n";
   exit(2);
@@ -576,7 +650,7 @@
   my $word_ja='\p{Han}+|\p{InHiragana}+|\p{InKatakana}+';
   my $word='(?:' . $word_ja . '|(?:(?:[-\w\d*]|\\\\[\"\'\`~^][A-Za-z\*])(?!(?:' . $word_ja . ')))+)';
   my $cmdleftright='\\\\(?:left|right|[Bb]igg?[lrm]?|middle)\s*(?:[<>()\[\]|\.]|\\\\(?:[|{}]|\w+))';
-  my $cmdoptseq='\\\\[\w\d\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
+  my $cmdoptseq='\\\\[\w\d@\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
   my $backslashnl='\\\\\n';
   my $oneletcmd='\\\\.\*?(?:\['.$brat0.'\]|\{'. $pat_n . '\})*';
   my $math='\$(?:[^$]|\\\$)*?\$|\\\\[(](?:.|\n)*?\\\\[)]';
@@ -599,18 +673,30 @@
 
 
 # set the labels to be included into the file
+# first find out which file name is longer for correct alignment
+my ($diff,$oldlabel_n_spaces,$newlabel_n_spaces);
+$oldlabel_n_spaces = 0;
+$newlabel_n_spaces = 0;
+$diff = length($newfile) - length($oldfile);
+if ($diff > 0) {
+  $oldlabel_n_spaces = $diff;
+}
+if ($diff < 0) {
+  $newlabel_n_spaces = abs($diff);
+}
+
 my ($oldtime,$newtime,$oldlabel,$newlabel);
 if (defined($labels[0])) {
   $oldlabel=$labels[0] ;
 } else {
   $oldtime=localtime((stat($oldfile))[9]); 
-  $oldlabel="$oldfile   " . " "x(length($newfile)-length($oldfile)) . $oldtime;
+  $oldlabel="$oldfile   " . " "x($oldlabel_n_spaces) . $oldtime;
 }
 if (defined($labels[1])) {
   $newlabel=$labels[1] ;
 } else {
   $newtime=localtime((stat($newfile))[9]);
-  $newlabel="$newfile   " . " "x(length($oldfile)-length($newfile)) . $newtime;
+  $newlabel="$newfile   " . " "x($newlabel_n_spaces) . $newtime;
 }
 
 $encoding=guess_encoding($newfile) unless defined($encoding);
@@ -631,7 +717,6 @@
 exetime(1);
 ($oldpreamble,$oldbody,$oldpost)=splitdoc($old,'\\\\begin\{document\}','\\\\end\{document\}');
 
-
 ($newpreamble,$newbody,$newpost)=splitdoc($new,'\\\\begin\{document\}','\\\\end\{document\}');
 
 
@@ -710,22 +795,28 @@
   # is not compatible with this.
   # (A more elegant solution would be to suppress the redefinitions of the \DIFaddbeginFL etc commands, but for this narrow use case
   #  I currently don't see this as an efficient use of time)
-  foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
-    @matches=( $latexdiffpreamble =~ m/command{\\DIFaddbeginFL}{($pat_n)}/sg ) ;
-    # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one elemen
-    if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
-      die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
-      exit 10;
-    }
+  ### The foreach loop does not make sense here. I don't know why I put this in -  (F Tilmann)
+  ###foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
+  @matches=( $latexdiffpreamble =~ m/command\{\\DIFaddbeginFL}\{($pat_n)}/sg ) ;
+  # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one element
+  if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
+    die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
+    exit 10;
   }
+  ###}
   $latexdiffpreamble .= join "\n",("\\newcommand{\\DIFscaledelfig}{$SCALEDELGRAPHICS}",extrapream("HIGHLIGHTGRAPHICS"),"");
 
   # only change required for highlighting both is to declare \includegraphics safe, as preamble already contains commands for deleted environment
   if ( $graphicsmarkup == BOTH ) {
-    init_regex_arr_ext(\@SAFECMDLIST,'includegraphics');
+    init_regex_arr_list(\@SAFECMDLIST,'includegraphics');
   }
 }
 
+# If listings is being used and latexdiffpreamble uses color markup
+if (defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/)) {
+  $latexdiffpreamble .= join "\n",(extrapream("LISTINGS"),"");
+}
+
 # adding begin and end marker lines to preamble
 $latexdiffpreamble = "%DIF PREAMBLE EXTENSION ADDED BY LATEXDIFF\n" . $ latexdiffpreamble . "%DIF END PREAMBLE EXTENSION ADDED BY LATEXDIFF\n";
 
@@ -787,11 +878,11 @@
   # this is done to get around an incompatibility between the ulem and siunitx package
   print STDERR "siunitx package detected.\n" if $verbose ;
   my $mboxcmds='SI,ang,numlist,numrange,SIlist,SIrange';
-  init_regex_arr_ext(\@SAFECMDLIST,'num,si');
+  init_regex_arr_list(\@SAFECMDLIST,'num,si');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -801,9 +892,9 @@
   print STDERR "cleveref package detected.\n" if $verbose ;
   my $mboxcmds='[Cc]ref(?:range)?\*?,labelcref,(?:lc)?name[cC]refs?' ;
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -812,17 +903,17 @@
   # this is done to get around an incompatibility between ulem and glossaries package 
   print STDERR "glossaries package detected.\n" if $verbose ;
   my $mboxcmds='[gG][lL][sS](?:|pl|disp|link|first|firstplural|desc|user[iv][iv]?[iv]?),[aA][cC][rR](?:long|longpl|full|fullpl),[aA][cC][lfp]?[lfp]?';
-  init_regex_arr_ext(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
+  init_regex_arr_list(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
 if (defined $packages{"chemformula"} or defined $packages{"chemmacros"} ) {
   print STDERR "chemformula package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ch');
+  init_regex_arr_list(\@SAFECMDLIST,'ch');
   push(@UNSAFEMATHCMD,'ch');
   # The next command would be needed to allow highlighting the interior of \ch commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -831,7 +922,7 @@
 
 if (defined $packages{"mhchem"} ) {
   print STDERR "mhchem package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ce');
+  init_regex_arr_list(\@SAFECMDLIST,'ce');
   push(@UNSAFEMATHCMD,'ce','cee');
   # The next command would be needed to allow highlighting the interior of \cee commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -873,7 +964,7 @@
 
 # add commands in MBOXCMDLIST to SAFECMDLIST
 foreach $mboxcmd ( @MBOXCMDLIST ) {
-  init_regex_arr_ext(\@SAFECMDLIST, $mboxcmd);
+  init_regex_arr_list(\@SAFECMDLIST, $mboxcmd);
 }
 
 # check if \label is in SAFECMDLIST, and if yes replace "label" in $LABELCMD by something that never matches (we hope!)
@@ -919,7 +1010,71 @@
 print STDERR "(",exetime()," s)\n","Done.\n" if $verbose;
 
 
+# liststringtoregex(liststring)
+# expands string with semi-colon separated list into a regular expression corresponding 
+# matching any of the elements
+sub liststringtoregex {
+  my ($liststring)=@_;
+  my @elements=grep /\S/,split(";",$liststring);
+  if ( @elements) {
+    return('(?:(?:' . join(')|(?:', at elements) .'))');
+  } else {
+    return "";
+  }
+}
 
+# show_configuration
+# note that this is not encapsulated but uses variables from the main program 
+# It is provided for convenience because in the future it is planned to allow output
+# to be modified based on what packages are read etc - this works only if the input files are actually red
+# whether or not additional files are provided
+sub show_configuration {
+  if ($showpreamble) {
+    print "\nPreamble commands:\n";
+    print $latexdiffpreamble ;
+  }
+
+  if ($showsafe) {
+    print "\nsafecmd: Commands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
+    print_regex_arr(@SAFECMDLIST);
+    print "\nsafecmd-exlude: Commands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
+    print_regex_arr(@SAFECMDEXCL);
+    print "\nmboxsafecmd:  Commands safe only if they are surrounded by \\mbox command:\n";
+    print_regex_arr(@MBOXCMDLIST);
+    print "\nnmboxsafecmd: Commands not safe:\n";
+    print_regex_arr(@MBOXCMDEXCL);
+  }
+
+  if ($showtext) {
+    print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
+    print_regex_arr(@TEXTCMDLIST);
+    print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
+    print_regex_arr(@CONTEXT1CMDLIST);
+    print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
+    print_regex_arr(@CONTEXT2CMDLIST);  
+    print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
+    print_regex_arr(@TEXTCMDEXCL);
+  }
+
+
+  if ($showconfig) {
+    print "Configuration variables:\n";
+    print "ARRENV=$ARRENV\n";
+    print "COUNTERCMD=$COUNTERCMD\n";
+    print "FLOATENV=$FLOATENV\n";
+    print "ITEMCMD=$ITEMCMD\n";
+    print "LISTENV=$LISTENV\n";
+    print "MATHARRENV=$MATHARRENV\n";
+    print "MATHARRREPL=$MATHARRREPL\n";
+    print "MATHENV=$MATHENV\n";
+    print "MATHREPL=$MATHREPL\n";
+    print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
+    print "PICTUREENV=$PICTUREENV\n";
+  }
+}
+
+
+
 ## guess_encoding(filename)
 ## reads the first 20 lines of filename and looks for call of inputenc package
 ## if found, return the option of this package (encoding), otherwise return utf8
@@ -1023,46 +1178,44 @@
 # and adds the created commands which are clearly safe to @SAFECMDLIST
 sub add_safe_commands {
   my ($preamble)=@_;
-  my $added_command = 1;
 
+  # get rid of comments
   $preamble=~s/(?<!\\)%.*$//mg ;
 
-  while ($added_command) { 
-    $added_command  = 0;
-      # get rid of comments
-    my $to_test = "";
-      # test for \DeclareMathOperator{\foo}{myoperator}
-    while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
-      $to_test=$1;
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
-      }
+  my $to_test = "";
+  # test for \DeclareMathOperator{\foo}{myoperator}
+  while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
+    $to_test=$1;
+    if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+      # one should add $to_test to the list of safe commands.
+      init_regex_arr_list(\@SAFECMDLIST, $to_test);
+      print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
     }
+  }
 
-    while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
-      my $maybe_to_test  = $1;
-      my $should_be_safe = $2;
-      my $success = 0;
-      # test if all latex commands inside it are safe
-      $success = 1;
-      if ($should_be_safe =~ m/\\\\/) {
-	$success = 0;
-      } else {
-	while ($should_be_safe =~ m/\\(\w+)/g) {
-	  $success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
-	}
+  while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
+    my $maybe_to_test  = $1;
+    my $should_be_safe = $2;
+    print STDERR "DEBUG Checking new command: maybe_to_test, should_be_safe: $1 $2\n" if $debug;
+    my $success = 0;
+    # test if all latex commands inside it are safe
+    $success = 1;
+    if ($should_be_safe =~ m/\\\\/) {
+      $success = 0;
+    } else {
+      while ($should_be_safe =~ m/\\(\w+)/g) {
+	###	  print STDERR "DEBUG: Testing command $1 " if $debug;
+	$success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
+	###        print STDERR " success=$success\n" if $debug;
       }
-      ###      }
-      $to_test = $maybe_to_test if $success;
-      
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-#        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
+    }
+    ###      }
+    if ($success) {
+      $to_test = $maybe_to_test;
+      if (  not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+	#        # one should add $to_test to the list of safe commands.
+	init_regex_arr_list(\@SAFECMDLIST, $to_test);
+	print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
       }
     }
   }
@@ -1069,6 +1222,19 @@
 }
 
 
+# helper function for flatten
+# remove \endinput at beginning of line and everything
+# following it, # if \endinput is not at the beginning of
+# the line, nothing will be removed. It is assumed that
+# this case is most common when \endinput is part of a
+# conditional clause.  The file will only be processed
+# correctly if the conditional is always false,
+# i.e. \endinput # not actually reached
+sub remove_endinput {
+  # s/// operates on default input
+  $_[0] =~ s/^\\endinput.*\Z//ms ;
+  return($_[0]);
+}
 
 # flatten($text,$preamble,$filename,$encoding)
 # expands \input and \include commands within text
@@ -1096,7 +1262,7 @@
   print STDERR "DEBUG: includeonly $includeonly\n" if $debug;
 
   # recursively replace \\input and \\include files
-  1 while $text=~s/(^(?:[^%\n]|\\%)*)\\input{(.*?)}|\\include{(${includeonly}(?:\.tex)?)}/{ 
+  1 while $text=~s/(^(?:[^%\n]|\\%)*)(?:\\input\{(.*?)\}|\\include\{(${includeonly}(?:\.tex)?)\})/{ 
 	    $begline=(defined($1)? $1 : "") ;
 	    $fname = $2 if defined($2) ;
 	    $fname = $3 if defined($3) ;
@@ -1108,6 +1274,7 @@
             # content of file becomes replacement value (use recursion), add \newpage if the command was include
             ###$replacement=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
 	    $replacement=flatten(read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding), $preamble,$filename,$encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
+	    $replacement = remove_endinput($replacement); 
 	    # \include always starts a new page; use explicit \newpage command to simulate this
 	    $newpage=(defined($3)? " \\newpage " : "") ;
 	    "$begline$newpage$replacement$newpage";
@@ -1136,6 +1303,7 @@
            $subfile=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
            ($subpreamble,$subbody,$subpost)=splitdoc($subfile,'\\\\begin\{document\}','\\\\end\{document\}');
 	   $replacement=flatten($subbody, $preamble,$filename,$encoding);
+	   $replacement = remove_endinput($replacement); 
 	   "$begline$replacement";
   }/exgm;
 
@@ -1779,7 +1947,7 @@
       if ( $word =~ /^\\(?!\()(\\|[`'^"~=.]|[\w*@]+)(.*?)(\s*)$/s &&  iscmd($1,\@MBOXCMDLIST,\@MBOXCMDEXCL)) {
 	# $word is a safe command in MBOXCMDLIST
 	###print STDERR "DEBUG Mboxsafecmd detected:$word:\n" if $debug ;
-	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}$AUXCMD\n" );
+	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}\\hspace{0pt}$AUXCMD\n" );
       } else {
 	# $word is a normal word or a safe command (not in MBOXCMDLIST)
 	push (@$retval,$word);
@@ -1796,47 +1964,49 @@
 
 #used in preprocess
 sub take_comments_and_enter_from_frac() {
-      ###*************take the \n and % between frac and {}***********
-      ###notice all of the substitution are made none global
-      while( m/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/s ) {
-	### if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
-	if( $2 !~ m/\\end{$1}/s ) {
-	  ### take out % and \n from the next match only (none global)
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
+      #*************take the \n and % between frac and {}***********
+      #notice all of the substitution are made none global
+      while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/s ) {
+	# if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
+	### print STDERR "Match the following in take_comments and_enter_from_frac(1):\n****$&****\n" if $debug;
+	if( $2 !~ m/\\end\{$1}/s ) {
+	  # take out % and \n from the next match only (none global)
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
 	}
 	else{
-	  ###there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+	  #there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
 	}
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       ###*************take the \n and % between frac and {}***********
       
       ###**********take the \n and % between {} and {} of the frac***************
       while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/s ) {
-	### if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	# if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	###print STDERR "Match the following in take_comments and_enter_from_frac(2):\n****$&****\n" if $debug;
         if( $2 !~ m/\\end\{$1\}/s ) {
-          ### from now on CURRFRAC is the frac we are looking at
+          # from now on CURRFRAC is the frac we are looking at
           s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\\end\{$1\}/s;
           while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{(.*?)\\end\{\1\}/s ) {
-             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end{\1}/s ) {
+             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1}/s ) {
                    s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\}\{$6\\end\{$1\}/s;
                } 
               else { # there is no comment or \n between the two brackets {}{}
-              ### change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
-                   s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
+              # change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
+                   s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end\{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
               }
             }
         }
         else{
           ###there are no more frac in $2, we want to find the next one so we clear the begin-end from the pattern
-          s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+          s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
         }
         
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       s/FRACSTART/\\frac/g;
       ###***************take the \n and % between {} and {} of the frac*********************
 }
@@ -1849,8 +2019,8 @@
 # 2. mark all first empty line (in block of several) with \PAR tokens
 # 3. Convert all '\%' into '\PERCENTAGE ' and all '\$' into \DOLLAR to make parsing regular expressions easier
 # 4. Convert all \verb|some verbatim text| commands (where | can be an arbitrary character)
-#    into \verb{hash}
-# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}
+#    into \verb{hash}  (also lstinline)
+# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}  (not only verbatim, all patterns matching VERBATIMENV)
 # 6. Convert _n into \SUBSCRIPTNB{n} and _{nnn} into \SUBSCRIPT{nn}
 # 7. Convert ^n into \SUPERSCRIPTNB{n} and ^{nnn} into \SUPERSCRIPT{nn}
 # 8. a. Convert $$ $$ into \begin{DOLLARDOLLAR} \end{DOLLARDOLLAR}
@@ -1866,6 +2036,17 @@
 # Returns: leading white space removed in step 1
 sub preprocess {
   for (@_) { 
+
+
+    # change in \verb and similar commands - note that I introduce an extra space here so that the 
+    #       already hashed variants do not trigger again
+    # transform \lstinline{...}
+#    s/\\lstinline(\[$brat0\])?(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+#    s/\\lstinline(\[$brat0\])?((\S).*?\2)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)((\S).*?\3)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\(verb\*?|lstinline)(\S)(.*?)\2/"\\DIF${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
+
     #    Change \{ to \QLEFTBRACE, \} to \QRIGHTBRACE, and \& to \AMPERSAND
     s/(?<!\\)\\\{/\\QLEFTBRACE /sg;
     s/(?<!\\)\\\}/\\QRIGHTBRACE /sg;
@@ -1873,12 +2054,10 @@
 # replace {,} in comments with \\CLEFTBRACE,\\CRIGHTBRACE
     1 while s/((?<!\\)%.*)\{(.*)$/$1\\CLEFTBRACE $2/mg ;
     1 while s/((?<!\\)%.*)\}(.*)$/$1\\CRIGHTBRACE $2/mg ;
-
     s/\n(\s*?)\n((?:\s*\n)*)/\n$1\\PAR\n$2/g ;
     s/(?<!\\)\\%/\\PERCENTAGE /g ;  # (?<! is negative lookbehind assertion to prevent \\% from being converted
     s/(?<!\\)\\\$/\\DOLLAR /g ;  # (?<! is negative lookbehind assertion to prevent \\$ from being converted
-    s/(\\verb\*?)(\S)(.*?)\2/"${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
-    s/\\begin\{(verbatim\*?)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
+    s/\\begin\{($VERBATIMENV)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
     # Convert _n or _\cmd into \SUBSCRIPTNB{n} or \SUBSCRIPTNB{\cmd} and _{nnn} into \SUBSCRIPT{nn}
     1 while s/(?<!\\)_(\s*([^{\\\s]|\\\w+))/\\SUBSCRIPTNB{$1}/g ;
     1 while s/(?<!\\)_(\s*{($pat_n)})/\\SUBSCRIPT$1/g ;
@@ -1894,17 +2073,20 @@
     s/(?<!\\)\\\[/\\begin{SQUAREBRACKET}/sg;
     s/\\\]/\\end{SQUAREBRACKET}/sg;
     # Convert all picture environmentent (\begin{PICTUREENV} .. \end{PICTUREENV} \PICTUREBLOCKenv
-    s/\\begin{($PICTUREENV)}(.*?)\\end{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
+    s/\\begin\{($PICTUREENV)}(.*?)\\end\{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
     #    For --block-math-markup option -convert all \begin{MATH} .. \end{MATH}
     #    into \MATHBLOCKMATH{...} commands, where MATH is any valid math environment
     #    Also convert all array environments into ARRAYBLOCK environments
+
     if ( $mathmarkup != FINE ) {
-      s/\\begin{($ARRENV)}(.*?)\\end{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
+      s/\\begin\{($ARRENV)}(.*?)\\end\{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
       
       take_comments_and_enter_from_frac();
       
-      s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/\\MATHBLOCK$1\{$2\}/sg;
+
+      s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\end\{\1\}/\\MATHBLOCK$1\{$2\}/sg;
     }
+
     # add final token " STOP"
     $_ .= " STOP"
   }
@@ -2013,6 +2195,7 @@
     # Replace \RIGHTBRACE by }    
     s/\\RIGHTBRACE/}/g;
 
+
     # Check all deleted blocks: where a deleted block contains a matching \begin and
     #    \end environment (these will be disabled by a %DIFDELCMD statements), enable
     #    these commands again (such that for example displayed math in a deleted equation
@@ -2104,15 +2287,17 @@
 #    bb. disable active labels within deleted blocks (as these are not safe commands, this should normally only
 #        happen within deleted maths blocks
       $delblock=~ s/(?<!$DELCMDOPEN)(\\$LABELCMD(?:${extraspace})\{(?:[^{}])*\}[\t ]*)\n?/${DELCMDOPEN}$1${DELCMDCLOSE}/smg ;
+      # previous line causes trouble as by issue #90 I might need to modify this
 
 
 #     c. If in-line math mode contains array environment, enclose the whole environment in \mbox'es
       while ( $delblock =~ m/($math)(\s*)/sg ) {
+#	      print STDERR "DEBUG Delblock Match math $& at ",pos,"\n";
 	$cnt2=0;
 	$len2=length($&);
 	$begin2=pos($delblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/ );
 	substr($delblock,$begin2,$len2)=$mathblock;
 	pos($delblock) = $begin2 + length($mathblock);
       }
@@ -2121,11 +2306,22 @@
 	# note additional \newline after command is omitted from output if right at the end of deleted block (otherwise a spurious empty line is generated)
 	$delblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+      ###  #     change included verbatim environments
+      ###  $delblock =~ s/\\DIFverb\{/\\DIFDIFdelverb\{/g;
+      ###  $delblock =~ s/\\DIFlstinline/\\DIFDIFdellstinline/g;
+      ###}
+      # Mark deleted verbose commands
+      $delblock =~ s/(${DELCMDOPEN}\\DIF((?:verb\*?|lstinline(?:\[$brat0\])?)\{([-\d]*?)\}\s*).*)$/%\n\\DIFDIFdel$2${AUXCMD}\n$1/gm;
 
 #     splice in modified delblock
       substr($_,$begin,$len)=$delblock;
       pos = $begin + length($delblock);
     }
+
+    ### print STDERR "<<<$_>>>\n" if $debug;
+
+
     # make the array modification in added blocks
     while ( m/\\DIFaddbegin.*?\\DIFaddend/sg ) {
       $cnt=0;
@@ -2137,7 +2333,7 @@
 	$len2=length($&);
 	$begin2=pos($addblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/) ;
 	substr($addblock,$begin2,$len2)=$mathblock;
 	pos($addblock) = $begin2 + length($mathblock);
       }
@@ -2146,6 +2342,11 @@
 	##$addblock=~s/($math)/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
 	$addblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+	# mark added verbatim commands
+      $addblock =~ s/\\DIFverb/\\DIFDIFaddverb/g;
+      $addblock =~ s/\\DIFlstinline/\\DIFDIFaddlstinline/g;
+      ###}
 #     splice in modified addblock
       substr($_,$begin,$len)=$addblock;
       pos = $begin + length($addblock);
@@ -2189,8 +2390,8 @@
     # environments) are between the \begin{$MATHENV} and \end{MATHMODE} commands. This is necessary as the minimal matching 
     # is not globally minimal but only 'locally' (matching is beginning from the left side of the string)
     if ( $mathmarkup == FINE ) {
-      1 while s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin{MATHMODE})))*?)\\end{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
-      1 while s/\\begin{MATHMODE}((?:.(?!\\end{MATHMODE}))*?)\\end{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
+      1 while s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end\{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin\{MATHMODE})))*?)\\end\{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
+      1 while s/\\begin\{MATHMODE}((?:.(?!\\end\{MATHMODE}))*?)\\end\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
       # convert remaining \begin{MATHMODE} \end{MATHMODE} (and not containing & or \\ )into MATHREPL environments
       s/\\begin\{MATHMODE\}((?:(.(?!(?<!\\)\&|\\\\))*)?)\\end\{MATHMODE\}/\\begin{$MATHREPL}$1\\end{$MATHREPL}/sg;
       # others into MATHARRREPL
@@ -2197,7 +2398,7 @@
       s/\\begin\{MATHMODE\}(.*?)\\end\{MATHMODE\}/\\begin{$MATHARRREPL}$1\\end{$MATHARRREPL}/sg;
 
       # now look for AUXCMD math-mode pairs which have only comments (or empty lines between them), and remove the added commands
-      s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
+      s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
     } else {
       #   math modes OFF,WHOLE,COARSE: Convert \MATHBLOCKmath{..} commands back to environments
       s/\\MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{($pat_n)\}/\\begin{$1}$2\\end{$1}/sg;
@@ -2245,13 +2446,8 @@
     s/^\Q${DELCMDOPEN}\E\n//msg;
 
     # Expand hashes of verb and verbatim environments (note negative look behind assertion to not leak out of DIFDELCMD comments
-    s/(\\verb\*?)\{([-\d]*?)\}/"${1}". fromhash(\%verbhash,$2)/esg;
-    s/${DELCMDOPEN}\\(verbatim\*?)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
-    s/\\(verbatim\*?)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
-    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
-    s/\\PERCENTAGE ?/\\%/g;
-    # Convert '\DOLLAR ' back into '\$'			       
-    s/\\DOLLAR /\\\$/g;
+    s/${DELCMDOPEN}\\($VERBATIMENV)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
+    s/\\($VERBATIMENV)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
     # remove all \PAR tokens (taking care to properly keep commented out PAR's
     # from introducing uncommented newlines - next line)
     s/(%DIF < )([^\n]*?)\\PAR\n/$1$2\n$1\n/sg;
@@ -2267,11 +2463,17 @@
       print STDERR "endfloat package detected.\n" if $verbose ;
       # eliminate whitespace before and after
       s/^(\s*)(\\(?:begin|end)\{(?:figure|table)\})(\s*)$/$2/mg;
-      # split lines with remaining characters before float enviroment conmmand
-      s/^(.+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
-      # split lines with remaining characters after float enviroment conmmand
-      s/(\\(?:begin|end)\{(?:figure|table)\})(.+)$/$1\n$2/mg;
+      # split lines with remaining characters before float environment conmmand
+      s/^([^%]+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
+      # split lines with remaining characters after float environment conmmand
+      s/^((?:[^%]+)\\(?:begin|end)\{(?:figure|table)\}(?:\[[a-zA-Z]+\])?)(.+)((?:%.*)?)$/$1\n$2$3/mg;
     }
+
+    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
+    s/\\PERCENTAGE ?/\\%/g;
+    # Convert '\DOLLAR ' back into '\$'			       
+    s/\\DOLLAR /\\\$/g;
+
     # undo renaming of the \begin and \end,{,}  and dollars in comments 
     1 while s/(%.*)DOLLARDIF/$1\$/mg ;
 #   Convert \begin{SQUAREBRACKET} \end{SQUAREBRACKET} into \[ \]
@@ -2297,6 +2499,21 @@
     s/\\QLEFTBRACE /\\\{/sg;
     s/\\QRIGHTBRACE /\\\}/sg;
     s/\\AMPERSAND /\\&/sg;
+    # Highligh added inline verbatim commands if possible
+    if ( $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/ )   { 
+      # wrap added verb commands with color commands
+      s/\\DIFDIFadd((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*)/\{\\color{blue}$AUXCMD\n\\DIF$1%\n\}$AUXCMD\n/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD)/\{\\color{red}${AUXCMD}\n\\DIF$1\n\}${AUXCMD}/sg;
+    } else {
+      # currently if colour markup is not used just remove the added mark
+      s/\\DIFDIFadd(verb\*?|lstinline)/\\DIF$1/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD\n)//sg;
+    }
+    # expand \verb and friends inline arguments
+    s/\\DIF((?:DIFadd|DIFdel)?(?:verb\*?|lstinline(?:\[$brat0\])?))\{([-\d]*?)\}/"\\${1}". fromhash(\%verbhash,$2)/esg;
+    # add basicstyle color{blue} to added lstinline commands
+    # finally add the comment to the ones not having an optional argument before
+    ###s/\\DIFaddlstinline(?!\[)/\\lstinline\n[basicstyle=\\color{blue}]$AUXCMD\n/g;
 
   return;
   }
@@ -2307,7 +2524,7 @@
 sub restore_item_commands {
   my ($string)=@_ ;
   my ($itemarg, at itemargs);
-  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*?(?:\n|$DELCMDCLOSE))/
+  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*((?:${cmdoptseq}\s*?)*)(?:\n|$DELCMDCLOSE))/
      # if \item has an []argument, then mark up the argument as deleted)
      if (length($4)>0) {
        # use substr to exclude square brackets at end points
@@ -2316,7 +2533,8 @@
      } else {
        $itemarg="";
      }
-     "$1$2$3$itemarg$AUXCMD\n"/sge;
+     "$1$2$3$itemarg$AUXCMD\n";  ###.((length($5)>0) ? "%DIFDELCMD $5 $DELCMDCLOSE\n" : "")
+     /sge;
   return($string);
 }
 
@@ -2499,16 +2717,32 @@
 # init_regex_arr_data(\@array,"TOKEN INIT")
 # scans DATA file handel for line "%% TOKEN INIT" line
 # then appends each line not beginning with % into array (as a quoted regex)
+# This is used for command lists and configuration variables, but the processing is slightly 
+# different: 
+# For lists, the regular expression is extended to include beginning (^) and end ($) markers, to require full-string matching
+# For configuration variables (and all others), simply an unadorned list is copied
 sub init_regex_arr_data {
   my ($arr,$token)=@_;
-  my ($copy);
+  my $copy=0;
+  my ($mode);
+  if ($token =~ m/COMMANDS/ ) { 
+    $mode=0;
+  } else {
+    $mode=1;
+  }
+
   while (<DATA>) {
     if ( m/^%%BEGIN $token\s*$/ ) {
-      $copy=1; }
-    elsif ( m/^%%END $token\s*/ )  {
+      $copy=1; 
+      next;
+    } elsif ( m/^%%END $token\s*$/ )  {
       last; }
     chomp;
-    push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    if ( $mode==0 ) {
+      push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    } elsif ($mode==1) {
+      push (@$arr,"$_") if ( $copy && !/^%/ ) ;
+    }
   }
   seek DATA,0,0;    # rewind DATA handle to file begin
 }
@@ -2521,25 +2755,42 @@
 # Otherwise treat arg as a comma separated list of regular expressions
 sub init_regex_arr_ext {
   my ($arr,$arg)=@_;
-  my $regex;
-  if ( -f $ arg ) {
-    open(FILE,"$arg") or die ("Couldn't open $arg: $!");
-    while (<FILE>) {
-      chomp;
-      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
-      push (@$arr,qr/^$_$/);
-    }
-    close(FILE);
+  if ( -f $arg ) {
+    init_regex_arr_file($arr,$arg);
+  } else {
+    init_regex_arr_list($arr,$arg);
   }
-  else {
-    # assume it is a comma-separated list of reg-ex
-    foreach $regex (split(qr/(?<!\\),/,$arg)) {
-      $regex =~ s/\\,/,/g;
-      push (@$arr,qr/^$regex$/);
-    }
+}
+
+# init_regex_arr_file(\@array,$fname)
+# appends array with regular expressions.
+# Read in list of regular expressions from $fname
+# (one expression per line)
+sub init_regex_arr_file {
+  my ($arr,$fname)=@_;
+  open(FILE,"$fname") or die ("Couldn't open $fname: $!");
+  while (<FILE>) {
+    chomp;
+    next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+    push (@$arr,qr/^$_$/);
   }
+  close(FILE);
 }
 
+# init_regex_arr_list(\@array,$arg)
+# appends array with regular expressions.
+# read from comma separated list of regular expressions ($arg)
+sub init_regex_arr_list {
+  my ($arr,$arg)=@_;
+  my $regex;
+  ###    print STDERR "DEBUG init_regex_arr_list arg >$arg<\n" if $debug;
+  foreach $regex (split(qr/(?<!\\),/,$arg)) {
+    $regex =~ s/\\,/,/g;
+    push (@$arr,qr/^$regex$/);
+  }
+}
+
+
 #exetime() returns time since last execution of this command
 #exetime(1) resets this time
 my $lasttime=-1;   # global variable for persistence
@@ -2681,6 +2932,14 @@
                           SCALEDELGRAPHICS (Float)
                        This option can be repeated.
 
+--add-to-config  varenv1=pattern1,varenv2=pattern2
+                       For configuration variables containing a regular expression (essentially those ending
+                       in ENV, and COUNTERCMD) this provides an alternative way to modify the configuration 
+                       variables. Instead of setting the complete pattern, with this option it is possible to add an
+                       alternative pattern. varenv must be one of the variables listed above that take a regular
+                       expression as argument, and pattern is any regular expression (which might need to be 
+                       protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+                       to separate them, e.g. --add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote"
 
 --packages=pkg1,pkg2,..
                        Tell latexdiff that .tex file is processed with the packages in list 
@@ -3099,11 +3358,12 @@
 =item B<--replace-context1cmd=replace-file> 
 
 =item B<--append-context1cmd=append-file> or
+
 =item B<--append-context1cmd="cmd1,cmd2,...">
 
 Replace or append to the list of regex matching commands
 whose last argument is text but which require a particular
-context to work, e.g. \caption will only work within a figure
+context to work, e.g. C<\caption> will only work within a figure
 or table.  These commands behave like text commands, except when 
 they occur in a deleted section, when they are disabled, but their
 argument is shown as deleted text.
@@ -3111,6 +3371,7 @@
 =item B<--replace-context2cmd=replace-file> 
 
 =item B<--append-context2cmd=append-file> or
+
 =item B<--append-context2cmd="cmd1,cmd2,...">
 
 As corresponding commands for context1.  The only difference is that
@@ -3123,7 +3384,7 @@
 =item B<--append-mboxsafecmd=append-file> or B<--append-mboxsafecmd="cmd1,cmd2,...">
 
 Define safe commands, which additionally need to be protected by encapsulating
-in an \\mbox{..}. This is sometimes needed to get around incompatibilities 
+in an C<\mbox{..}>. This is sometimes needed to get around incompatibilities 
 between external packages and the ulem package, which is  used for highlighting
 in the default style UNDERLINE as well as CULINECHBAR CFONTSTRIKE
 
@@ -3163,6 +3424,16 @@
 
 C<SCALEDELGRAPHICS> (Float)
 
+=item B<--add-to-config varenv1=pattern1,varenv2=pattern2,...>
+
+For configuration variables, which are a regular expression (essentially those ending
+in ENV, and COUNTERCMD, see list above) this provides an alternative way to modify the configuration 
+variables. Instead of setting the complete pattern, with this option it is possible to add an
+alternative pattern. C<varenv> must be one of the variables listed above that take a regular
+expression as argument, and pattern is any regular expression (which might need to be 
+protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+to separate them, e.g. C<--add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote">
+
 =item B<--show-safecmd>
 
 Print list of RegEx matching and excluding safe commands.
@@ -3223,16 +3494,17 @@
 =item B<--graphics-markup=level>
 
  Change highlight style for graphics embedded with C<\includegraphics> commands.
-                      Possible values for level:
 
-                      C<none, C<off> or C<0>: no highlighting for figures
+Possible values for level:
 
-                      C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
+C<none>, C<off> or C<0>: no highlighting for figures
 
-                      C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
-                                scale, and crossed out with a red diagonal cross. Use configuration
-                                variable SCALEDELGRAPHICS to set size of deleted figures.
+C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
 
+C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
+scale, and crossed out with a red diagonal cross. Use configuration
+variable SCALEDELGRAPHICS to set size of deleted figures.
+
 Note that changes to the optional parameters will make the figure appear as changed 
 to latexdiff, and this figure will thus be highlighted
 
@@ -3288,7 +3560,7 @@
 
 Suppress inclusion of old and new file names as comment in output file
 
-=item B<--visble-label>
+=item B<--visible-label>
 
 Include old and new filenames (or labels set with C<--label> option) as 
 visible output.
@@ -3597,8 +3869,8 @@
 
 =head1 AUTHOR
 
-Version 1.2.0
-Copyright (C) 2004-2015 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2004-2017 Frederik Tilmann
 
 This program is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License Version 3
@@ -3880,9 +4152,68 @@
 institute
 %%END CONTEXT2 COMMANDS 
 
+%% CONFIGURATION variabe defaults
+%%BEGIN LISTENV CONFIG
+itemize
+description
+enumerate
+%%END LISTENV CONFIG
 
-%% TYPES (Commands for highlighting changed blocks)
+%%BEGIN FLOATENV CONFIG
+figure[\w\d*@]*
+table[\w\d*@]*
+plate[\w\d*@]*
+%%END FLOATENV CONFIG
 
+%%BEGIN PICTUREENV CONFIG
+picture[\w\d*@]*
+tikzpicture[\w\d*@]*
+DIFnomarkup
+%%END PICTUREENV CONFIG
+
+%%BEGIN MATHENV CONFIG
+equation[*]?
+displaymath
+DOLLARDOLLAR
+%%END MATHENV CONFIG
+
+%%BEGIN MATHARRENV CONFIG
+eqnarray[*]?
+align[*]?
+alignat[*]?
+gather[*]?
+multline[*]?
+flalign[*]?
+%%END MATHARRENV CONFIG
+
+%%BEGIN ARRENV CONFIG
+aligned
+array
+[pbvBV]?matrix
+smallmatrix
+cases
+split
+%%END ARRENV CONFIG
+
+%%BEGIN COUNTERCMD CONFIG
+footnote
+part
+chapter
+section
+subsection
+subsubsection
+paragraph
+subparagraph
+%%END COUNTERCMD CONFIG
+
+%%BEGIN VERBATIMENV CONFIG
+verbatim[*]?
+lstlisting
+comment
+%%END VERBATIMENV CONFIG
+
+%%% TYPES (Commands for highlighting changed blocks)
+
 %DIF UNDERLINE PREAMBLE
 \RequirePackage[normalem]{ulem}
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
@@ -3893,6 +4224,7 @@
 %DIF CTRADITIONAL PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} [..\footnote{removed: #1} ]}}
 %DIF END CTRADITIONAL PREAMBLE
@@ -3899,6 +4231,7 @@
 
 %DIF TRADITIONAL PREAMBLE
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{[..\footnote{removed: #1} ]}}
 %DIF END TRADITIONAL PREAMBLE
@@ -3905,6 +4238,7 @@
 
 %DIF CFONT PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} \scriptsize #1}}
 %DIF END CFONT PREAMBLE
@@ -3911,6 +4245,7 @@
 
 %DIF FONTSTRIKE PREAMBLE
 \RequirePackage[normalem]{ulem}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{\footnotesize \sout{#1}}}
 %DIF END FONTSTRIKE PREAMBLE
@@ -3949,6 +4284,7 @@
 %DIF END INVISIBLE PREAMBLE
 
 %DIF BOLD PREAMBLE
+\DeclareOldFontCommand{\bf}{\normalfont\bfseries}{\mathbf}
 \providecommand{\DIFadd}[1]{{\bf #1}}
 \providecommand{\DIFdel}[1]{}
 %DIF END BOLD PREAMBLE
@@ -3973,7 +4309,7 @@
 \providecommand{\DIFaddend}{\protect\marginpar{]}}
 \providecommand{\DIFdelbegin}{\protect\marginpar{d[}}
 \providecommand{\DIFdelend}{\protect\marginpar{]}}
-%DIF END BRACKET PREAMBLE
+%DIF END MARGIN PREAMBLE
 
 %DIF DVIPSCOL PREAMBLE
 %Note: only works with dvips converter
@@ -4166,4 +4502,18 @@
 %DIF HYPERREF PREAMBLE
 \providecommand{\DIFadd}[1]{\texorpdfstring{\DIFaddtex{#1}}{#1}}
 \providecommand{\DIFdel}[1]{\texorpdfstring{\DIFdeltex{#1}}{}}
-%DIF END HYPERREF PACKAGE
+%DIF END HYPERREF PREAMBLE
+
+%DIF LISTINGS PREAMBLE
+\lstdefinelanguage{codediff}{
+  moredelim=**[is][\color{red}]{*!----}{----!*},
+  moredelim=**[is][\color{blue}]{*!++++}{++++!*}
+}
+\lstdefinestyle{codediff}{
+	belowcaptionskip=.25\baselineskip,
+	language=codediff,
+	basicstyle=\ttfamily,
+	columns=fullflexible,
+	keepspaces=true,
+}
+%DIF END LISTINGS PREAMBLE

Modified: trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff-fast
===================================================================
--- trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff-fast	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/doc/support/latexdiff/latexdiff-fast	2017-06-26 21:22:49 UTC (rev 44704)
@@ -24,6 +24,28 @@
 #
 # ToDo:
 #
+# Version 1.2.1 (22 June 2017)
+#    - add "DeclareOldFontCommand" to styles using \bf or \sf old style font commands (fixies issue #92 )
+#    - improved markup: process lstinline commands in listings package correctly
+#      for styles using colour, \verb and \lstinline arguments are marked up with colour (blue for added, red for deleted)
+#    - bug fix: protecting inline math expressions for mbox did not work as intended (see stack exchange question: http://tex.stackexchange.com/questions/359412/compiling-the-latexdiff-when-adding-a-subscript-before-a-pmatrix-environment-cau)
+#    - bug fix: when deleted \item commands are followed immediately by unsafe commands, they were not restored properly
+#      (thanks to J. Protze for pull request) (pull request #89)
+#    - treat lstlisting and comment as equivalent to verbatim environment
+#      make environments that are treated like verbatim environments configurable (config variable VERBATIMENV)
+#      treat lstinlne as equivalent to verb command
+#      partially addresses issue #38
+#    - refactoring: set default configuration variables in a hash, and those that correspond to lists 
+#    - feature: option --add-to-config used to amend configuration variables, which are regex pattern lists
+#    - bug fix: deleted figures when endfloat package is activated
+#    - bug fix: alignat environment now always processed correctly (fix issues #65)
+#    - bug fix: avoid processing of commands as potential files in routine init_regex_arr (fix issue #70 )
+#    - minimal feature enhancement: treat '@' as allowed character in commands (strictly speaking requires prior \makeatletter statement, but always assuming it to be 
+#       @       a letter if it is part of a command name will usually lead to the correct behaviour (see http://tex.stackexchange.com/questions/346651/latexdiff-and-let)
+#    - new feature/bug fix: --flatten option \endinput in included files now respected but only if \endinput stands right at the beginning of the line (issue #77)
+#    - bug fix: flatten would incorrectly attempt to process commented out \include commands (from discussion in issue #77 )
+#    - introduce an invisible space (\hspace{0pt} after \mbox{..} auxiliary commands (not in math mode), to allow line breaks between added and deleted citations (change should not cause adverse behaviour otherwise)
+#
 # Version 1.2.0:
 #    - highlight new and deleted figures
 #    - bug fix in title mark-up. Previously deleted commands in title (such as \title, \author or \date) were marked up erroneously
@@ -701,34 +723,76 @@
 
 
 my ($versionstring)=<<EOF ;
-This is LATEXDIFF 1.2.0  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
-  (c) 2004-2016 F J Tilmann
+This is LATEXDIFF 1.2.1  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
+  (c) 2004-2017 F J Tilmann
 EOF
 
-# Configuration variables: these have to be visible from the subroutines
-my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
-                     # shorter identical blocks will be merged to the previous word
-my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
-                      # this is only used for --graphics-markup=BOTH option
-my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
-my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
-my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
-my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
-my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
-my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
-my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
-my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+# Hash with defaults for configuration variables. These marked undef have default values constructed from list defined in the DATA block
+# (under tag CONFIG)
+my %CONFIG=(
+   MINWORDSBLOCK => 3, # minimum number of tokens to form an independent block
+                        # shorter identical blocks will be merged to the previous word
+   SCALEDELGRAPHICS => 0.5, # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+                             # this is only used for --graphics-markup=BOTH option
+   FLOATENV => undef ,   # Environments in which FL variants of defined commands are used
+   PICTUREENV => undef ,   # Environments in which all change markup is removed
+   MATHENV => undef ,           # Environments turning on display math mode (code also knows about \[ and \])
+   MATHREPL => 'displaymath',  # Environment introducing deleted maths blocks
+   MATHARRENV => undef ,           # Environments turning on eqnarray math mode
+   MATHARRREPL => 'eqnarray*',  # Environment introducing deleted maths blocks
+   ARRENV => undef , # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+   COUNTERCMD => undef,
+                                        # COUNTERCMD textcmds which are associated with a counter
                                         # If any of these commands occur in a deleted block
                                         # they will be succeeded by an \addtocounter{...}{-1}
                                         # for the associated counter such that the overall numbers
                                         # should be the same as in the new file
+   LISTENV => undef ,  # list making environments - they will generally be kept
+   VERBATIMENV => undef,    # Environments whose content should be treated as verbatim text 
+   ITEMCMD => 'item'                    # command marking item in a list environment
+);
+# Configuration variables: these have to be visible from the subroutines
+my ($ARRENV,
+    $COUNTERCMD,
+    $FLOATENV,
+    $ITEMCMD,
+    $LISTENV,
+    $MATHARRENV,
+    $MATHARRREPL,
+    $MATHENV,
+    $MATHREPL,
+    $MINWORDSBLOCK,
+    $PICTUREENV,
+    $SCALEDELGRAPHICS,
+    $VERBATIMENV
+);
+
+# my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
+#                      # shorter identical blocks will be merged to the previous word
+# my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+#                       # this is only used for --graphics-markup=BOTH option
+# my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
+# my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
+# my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
+# my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
+# my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
+# my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
+# my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+# my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+#                                         # If any of these commands occur in a deleted block
+#                                         # they will be succeeded by an \addtocounter{...}{-1}
+#                                         # for the associated counter such that the overall numbers
+#                                         # should be the same as in the new file
+# my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
+# my $ITEMCMD='item';   # command marking item in a list environment
+
+
+
 my $LABELCMD='(?:label)';                # matching commands are disabled within deleted blocks - mostly useful for maths mode, as otherwise it would be fine to just not add those to SAFECMDLIST
 my @UNSAFEMATHCMD=('qedhere');           # Commands which are definitely unsafe for marking up in math mode (amsmath qedhere only tested to not work with UNDERLINE markup) (only affects WHOLE and COARSE math markup modes). Note that unlike text mode (or FINE math mode0 deleted unsafe commands are not deleted but simply taken outside \DIFdel
 my $MBOXINLINEMATH=0; # if set to 1 then surround marked-up inline maths expression with \mbox ( to get around compatibility
                       # problems between some maths packages and ulem package
 
-my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
-my $ITEMCMD='item';
 
 # Markup strings
 # If at all possible, do not change these as parts of the program
@@ -778,7 +842,8 @@
 
 my ($i,$j,$l);
 my ($old,$new);
-my ($line);
+my ($line,$key);
+my (@dumlist);
 my ($newpreamble,$oldpreamble);
 my (@newpreamble, at oldpreamble, at diffpreamble, at diffbody);
 my ($latexdiffpreamble);
@@ -813,7 +878,7 @@
 
 my ($mboxcmd);
 
-my (@configlist, at labels,
+my (@configlist, at addtoconfiglist, at labels,
     @appendsafelist, at excludesafelist,
     @appendmboxsafelist, at excludemboxsafelist,
     @appendtextlist, at excludetextlist,
@@ -855,6 +920,7 @@
 	   'subtype|s=s' => \$subtype, 
 	   'floattype|f=s' => \$floattype, 
 	   'config|c=s' => \@configlist,
+	   'add-to-config|c=s' => \@addtoconfiglist,
 	   'preamble|p=s' => \$preamblefile,
 	   'encoding|e=s' => \$encoding,
 	   'label|L=s' => \@labels,
@@ -1021,21 +1087,64 @@
 }
 foreach $assign ( @config ) {
   $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
-  if ( $1 eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $2; }
-  elsif ( $1 eq "FLOATENV" ) { $FLOATENV = $2 ; }
-  elsif ( $1 eq "ITEMCMD" ) { $ITEMCMD = $2 ; }
-  elsif ( $1 eq "LISTENV" ) { $LISTENV = $2 ; }
-  elsif ( $1 eq "PICTUREENV" ) { $PICTUREENV = $2 ; }
-  elsif ( $1 eq "MATHENV" ) { $MATHENV = $2 ; }
-  elsif ( $1 eq "MATHREPL" ) { $MATHREPL = $2 ; }
-  elsif ( $1 eq "MATHARRENV" ) { $MATHARRENV = $2 ; }
-  elsif ( $1 eq "MATHARRREPL" ) { $MATHARRREPL = $2 ; }
-  elsif ( $1 eq "ARRENV" ) { $ARRENV = $2 ; }
-  elsif ( $1 eq "COUNTERCMD" ) { $COUNTERCMD = $2 ; }
-  elsif ( $1 eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $2 ; }
-  else { die "Unknown variable $1 in assignment.";}
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}=$2;
 }
 
+my @addtoconfig=();
+foreach $config ( @addtoconfiglist ) {
+  if (-f $config || lc $config eq '/dev/null' ) {
+    open(FILE,$config) or die ("Couldn't open addd-to-config file $config: $!");
+    while (<FILE>) {
+      chomp;
+      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+      push (@addtoconfig,$_);
+    }
+    close(FILE);
+  }
+  else {
+#    foreach ( split(",",$config) ) {
+#      push @addtoconfig,$_;
+#    }
+     push @addtoconfig,split(",",$config)
+  }
+}
+
+# initialise default lists from DATA
+# for those configuration variables, which have not been set explicitly, initiate from list in document
+foreach $key ( keys(%CONFIG) ) {
+  if (!defined $CONFIG{$key}) {
+    @dumlist=();
+    init_regex_arr_data(\@dumlist,"$key CONFIG");
+    $CONFIG{$key}=join(";", at dumlist)
+  }
+}
+
+
+foreach $assign ( @addtoconfig ) {
+  $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}.=";$2";
+}
+
+# Map from hash to variables (we do this to have more concise code later, change from comma-separeted list)
+foreach  ( keys(%CONFIG) ) {
+  if ( $_ eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $CONFIG{$_}; }
+  elsif ( $_ eq "FLOATENV" ) { $FLOATENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "ITEMCMD" ) { $ITEMCMD = $CONFIG{$_} ; }
+  elsif ( $_ eq "LISTENV" ) { $LISTENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "PICTUREENV" ) { $PICTUREENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHENV" ) { $MATHENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHREPL" ) { $MATHREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "MATHARRENV" ) { $MATHARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHARRREPL" ) { $MATHARRREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "ARRENV" ) { $ARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "VERBATIMENV" ) { $VERBATIMENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "COUNTERCMD" ) { $COUNTERCMD = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $CONFIG{$_} ; }
+  else { die "Unknown configuration variable $_.";}
+}
+
 if ( $mathmarkup == COARSE || $mathmarkup == WHOLE ) {
   push(@MATHTEXTCMDLIST,qr/^MATHBLOCK(?:$MATHENV|$MATHARRENV|SQUAREBRACKET)$/);
 }
@@ -1048,46 +1157,11 @@
 
 
 
-if ($showpreamble) {
-  print "\nPreamble commands:\n";
-  print $latexdiffpreamble ;
+if ($showconfig || $showtext || $showsafe || $showpreamble) {
+  show_configuration();
+  exit 0; 
 }
 
-if ($showsafe) {
-  print "\nCommands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
-  print_regex_arr(@SAFECMDLIST);
-  print "\nCommands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
-  print_regex_arr(@SAFECMDEXCL);
-}
-
-if ($showtext) {
-  print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
-  print_regex_arr(@TEXTCMDLIST);
-  print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
-  print_regex_arr(@CONTEXT1CMDLIST);
-  print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
-  print_regex_arr(@CONTEXT2CMDLIST);  
-  print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
-  print_regex_arr(@TEXTCMDEXCL);
-}
-
-
-if ($showconfig) {
-  print "Configuration variables:\n";
-  print "ARRENV=$ARRENV\n";
-  print "COUNTERCMD=$COUNTERCMD\n";
-  print "FLOATENV=$FLOATENV\n";
-  print "ITEMCMD=$ITEMCMD\n";
-  print "LISTENV=$LISTENV\n";
-  print "MATHARRENV=$MATHARRENV\n";
-  print "MATHARRREPL=$MATHARRREPL\n";
-  print "MATHENV=$MATHENV\n";
-  print "MATHREPL=$MATHREPL\n";
-  print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
-  print "PICTUREENV=$PICTUREENV\n";
-}
-if ($showconfig || $showtext || $showsafe || $showpreamble) {
-  exit 0; }
 if ( @ARGV != 2 ) { 
   print STDERR "2 and only 2 non-option arguments required.  Write latexdiff -h to get help\n";
   exit(2);
@@ -1137,7 +1211,7 @@
   my $word_ja='\p{Han}+|\p{InHiragana}+|\p{InKatakana}+';
   my $word='(?:' . $word_ja . '|(?:(?:[-\w\d*]|\\\\[\"\'\`~^][A-Za-z\*])(?!(?:' . $word_ja . ')))+)';
   my $cmdleftright='\\\\(?:left|right|[Bb]igg?[lrm]?|middle)\s*(?:[<>()\[\]|\.]|\\\\(?:[|{}]|\w+))';
-  my $cmdoptseq='\\\\[\w\d\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
+  my $cmdoptseq='\\\\[\w\d@\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
   my $backslashnl='\\\\\n';
   my $oneletcmd='\\\\.\*?(?:\['.$brat0.'\]|\{'. $pat_n . '\})*';
   my $math='\$(?:[^$]|\\\$)*?\$|\\\\[(](?:.|\n)*?\\\\[)]';
@@ -1160,18 +1234,30 @@
 
 
 # set the labels to be included into the file
+# first find out which file name is longer for correct alignment
+my ($diff,$oldlabel_n_spaces,$newlabel_n_spaces);
+$oldlabel_n_spaces = 0;
+$newlabel_n_spaces = 0;
+$diff = length($newfile) - length($oldfile);
+if ($diff > 0) {
+  $oldlabel_n_spaces = $diff;
+}
+if ($diff < 0) {
+  $newlabel_n_spaces = abs($diff);
+}
+
 my ($oldtime,$newtime,$oldlabel,$newlabel);
 if (defined($labels[0])) {
   $oldlabel=$labels[0] ;
 } else {
   $oldtime=localtime((stat($oldfile))[9]); 
-  $oldlabel="$oldfile   " . " "x(length($newfile)-length($oldfile)) . $oldtime;
+  $oldlabel="$oldfile   " . " "x($oldlabel_n_spaces) . $oldtime;
 }
 if (defined($labels[1])) {
   $newlabel=$labels[1] ;
 } else {
   $newtime=localtime((stat($newfile))[9]);
-  $newlabel="$newfile   " . " "x(length($oldfile)-length($newfile)) . $newtime;
+  $newlabel="$newfile   " . " "x($newlabel_n_spaces) . $newtime;
 }
 
 $encoding=guess_encoding($newfile) unless defined($encoding);
@@ -1192,7 +1278,6 @@
 exetime(1);
 ($oldpreamble,$oldbody,$oldpost)=splitdoc($old,'\\\\begin\{document\}','\\\\end\{document\}');
 
-
 ($newpreamble,$newbody,$newpost)=splitdoc($new,'\\\\begin\{document\}','\\\\end\{document\}');
 
 
@@ -1271,22 +1356,28 @@
   # is not compatible with this.
   # (A more elegant solution would be to suppress the redefinitions of the \DIFaddbeginFL etc commands, but for this narrow use case
   #  I currently don't see this as an efficient use of time)
-  foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
-    @matches=( $latexdiffpreamble =~ m/command{\\DIFaddbeginFL}{($pat_n)}/sg ) ;
-    # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one elemen
-    if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
-      die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
-      exit 10;
-    }
+  ### The foreach loop does not make sense here. I don't know why I put this in -  (F Tilmann)
+  ###foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
+  @matches=( $latexdiffpreamble =~ m/command\{\\DIFaddbeginFL}\{($pat_n)}/sg ) ;
+  # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one element
+  if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
+    die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
+    exit 10;
   }
+  ###}
   $latexdiffpreamble .= join "\n",("\\newcommand{\\DIFscaledelfig}{$SCALEDELGRAPHICS}",extrapream("HIGHLIGHTGRAPHICS"),"");
 
   # only change required for highlighting both is to declare \includegraphics safe, as preamble already contains commands for deleted environment
   if ( $graphicsmarkup == BOTH ) {
-    init_regex_arr_ext(\@SAFECMDLIST,'includegraphics');
+    init_regex_arr_list(\@SAFECMDLIST,'includegraphics');
   }
 }
 
+# If listings is being used and latexdiffpreamble uses color markup
+if (defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/)) {
+  $latexdiffpreamble .= join "\n",(extrapream("LISTINGS"),"");
+}
+
 # adding begin and end marker lines to preamble
 $latexdiffpreamble = "%DIF PREAMBLE EXTENSION ADDED BY LATEXDIFF\n" . $ latexdiffpreamble . "%DIF END PREAMBLE EXTENSION ADDED BY LATEXDIFF\n";
 
@@ -1348,11 +1439,11 @@
   # this is done to get around an incompatibility between the ulem and siunitx package
   print STDERR "siunitx package detected.\n" if $verbose ;
   my $mboxcmds='SI,ang,numlist,numrange,SIlist,SIrange';
-  init_regex_arr_ext(\@SAFECMDLIST,'num,si');
+  init_regex_arr_list(\@SAFECMDLIST,'num,si');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -1362,9 +1453,9 @@
   print STDERR "cleveref package detected.\n" if $verbose ;
   my $mboxcmds='[Cc]ref(?:range)?\*?,labelcref,(?:lc)?name[cC]refs?' ;
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -1373,17 +1464,17 @@
   # this is done to get around an incompatibility between ulem and glossaries package 
   print STDERR "glossaries package detected.\n" if $verbose ;
   my $mboxcmds='[gG][lL][sS](?:|pl|disp|link|first|firstplural|desc|user[iv][iv]?[iv]?),[aA][cC][rR](?:long|longpl|full|fullpl),[aA][cC][lfp]?[lfp]?';
-  init_regex_arr_ext(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
+  init_regex_arr_list(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
 if (defined $packages{"chemformula"} or defined $packages{"chemmacros"} ) {
   print STDERR "chemformula package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ch');
+  init_regex_arr_list(\@SAFECMDLIST,'ch');
   push(@UNSAFEMATHCMD,'ch');
   # The next command would be needed to allow highlighting the interior of \ch commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -1392,7 +1483,7 @@
 
 if (defined $packages{"mhchem"} ) {
   print STDERR "mhchem package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ce');
+  init_regex_arr_list(\@SAFECMDLIST,'ce');
   push(@UNSAFEMATHCMD,'ce','cee');
   # The next command would be needed to allow highlighting the interior of \cee commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -1434,7 +1525,7 @@
 
 # add commands in MBOXCMDLIST to SAFECMDLIST
 foreach $mboxcmd ( @MBOXCMDLIST ) {
-  init_regex_arr_ext(\@SAFECMDLIST, $mboxcmd);
+  init_regex_arr_list(\@SAFECMDLIST, $mboxcmd);
 }
 
 # check if \label is in SAFECMDLIST, and if yes replace "label" in $LABELCMD by something that never matches (we hope!)
@@ -1480,7 +1571,71 @@
 print STDERR "(",exetime()," s)\n","Done.\n" if $verbose;
 
 
+# liststringtoregex(liststring)
+# expands string with semi-colon separated list into a regular expression corresponding 
+# matching any of the elements
+sub liststringtoregex {
+  my ($liststring)=@_;
+  my @elements=grep /\S/,split(";",$liststring);
+  if ( @elements) {
+    return('(?:(?:' . join(')|(?:', at elements) .'))');
+  } else {
+    return "";
+  }
+}
 
+# show_configuration
+# note that this is not encapsulated but uses variables from the main program 
+# It is provided for convenience because in the future it is planned to allow output
+# to be modified based on what packages are read etc - this works only if the input files are actually red
+# whether or not additional files are provided
+sub show_configuration {
+  if ($showpreamble) {
+    print "\nPreamble commands:\n";
+    print $latexdiffpreamble ;
+  }
+
+  if ($showsafe) {
+    print "\nsafecmd: Commands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
+    print_regex_arr(@SAFECMDLIST);
+    print "\nsafecmd-exlude: Commands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
+    print_regex_arr(@SAFECMDEXCL);
+    print "\nmboxsafecmd:  Commands safe only if they are surrounded by \\mbox command:\n";
+    print_regex_arr(@MBOXCMDLIST);
+    print "\nnmboxsafecmd: Commands not safe:\n";
+    print_regex_arr(@MBOXCMDEXCL);
+  }
+
+  if ($showtext) {
+    print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
+    print_regex_arr(@TEXTCMDLIST);
+    print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
+    print_regex_arr(@CONTEXT1CMDLIST);
+    print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
+    print_regex_arr(@CONTEXT2CMDLIST);  
+    print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
+    print_regex_arr(@TEXTCMDEXCL);
+  }
+
+
+  if ($showconfig) {
+    print "Configuration variables:\n";
+    print "ARRENV=$ARRENV\n";
+    print "COUNTERCMD=$COUNTERCMD\n";
+    print "FLOATENV=$FLOATENV\n";
+    print "ITEMCMD=$ITEMCMD\n";
+    print "LISTENV=$LISTENV\n";
+    print "MATHARRENV=$MATHARRENV\n";
+    print "MATHARRREPL=$MATHARRREPL\n";
+    print "MATHENV=$MATHENV\n";
+    print "MATHREPL=$MATHREPL\n";
+    print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
+    print "PICTUREENV=$PICTUREENV\n";
+  }
+}
+
+
+
 ## guess_encoding(filename)
 ## reads the first 20 lines of filename and looks for call of inputenc package
 ## if found, return the option of this package (encoding), otherwise return utf8
@@ -1584,46 +1739,44 @@
 # and adds the created commands which are clearly safe to @SAFECMDLIST
 sub add_safe_commands {
   my ($preamble)=@_;
-  my $added_command = 1;
 
+  # get rid of comments
   $preamble=~s/(?<!\\)%.*$//mg ;
 
-  while ($added_command) { 
-    $added_command  = 0;
-      # get rid of comments
-    my $to_test = "";
-      # test for \DeclareMathOperator{\foo}{myoperator}
-    while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
-      $to_test=$1;
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
-      }
+  my $to_test = "";
+  # test for \DeclareMathOperator{\foo}{myoperator}
+  while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
+    $to_test=$1;
+    if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+      # one should add $to_test to the list of safe commands.
+      init_regex_arr_list(\@SAFECMDLIST, $to_test);
+      print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
     }
+  }
 
-    while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
-      my $maybe_to_test  = $1;
-      my $should_be_safe = $2;
-      my $success = 0;
-      # test if all latex commands inside it are safe
-      $success = 1;
-      if ($should_be_safe =~ m/\\\\/) {
-	$success = 0;
-      } else {
-	while ($should_be_safe =~ m/\\(\w+)/g) {
-	  $success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
-	}
+  while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
+    my $maybe_to_test  = $1;
+    my $should_be_safe = $2;
+    print STDERR "DEBUG Checking new command: maybe_to_test, should_be_safe: $1 $2\n" if $debug;
+    my $success = 0;
+    # test if all latex commands inside it are safe
+    $success = 1;
+    if ($should_be_safe =~ m/\\\\/) {
+      $success = 0;
+    } else {
+      while ($should_be_safe =~ m/\\(\w+)/g) {
+	###	  print STDERR "DEBUG: Testing command $1 " if $debug;
+	$success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
+	###        print STDERR " success=$success\n" if $debug;
       }
-      ###      }
-      $to_test = $maybe_to_test if $success;
-      
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-#        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
+    }
+    ###      }
+    if ($success) {
+      $to_test = $maybe_to_test;
+      if (  not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+	#        # one should add $to_test to the list of safe commands.
+	init_regex_arr_list(\@SAFECMDLIST, $to_test);
+	print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
       }
     }
   }
@@ -1630,6 +1783,19 @@
 }
 
 
+# helper function for flatten
+# remove \endinput at beginning of line and everything
+# following it, # if \endinput is not at the beginning of
+# the line, nothing will be removed. It is assumed that
+# this case is most common when \endinput is part of a
+# conditional clause.  The file will only be processed
+# correctly if the conditional is always false,
+# i.e. \endinput # not actually reached
+sub remove_endinput {
+  # s/// operates on default input
+  $_[0] =~ s/^\\endinput.*\Z//ms ;
+  return($_[0]);
+}
 
 # flatten($text,$preamble,$filename,$encoding)
 # expands \input and \include commands within text
@@ -1657,7 +1823,7 @@
   print STDERR "DEBUG: includeonly $includeonly\n" if $debug;
 
   # recursively replace \\input and \\include files
-  1 while $text=~s/(^(?:[^%\n]|\\%)*)\\input{(.*?)}|\\include{(${includeonly}(?:\.tex)?)}/{ 
+  1 while $text=~s/(^(?:[^%\n]|\\%)*)(?:\\input\{(.*?)\}|\\include\{(${includeonly}(?:\.tex)?)\})/{ 
 	    $begline=(defined($1)? $1 : "") ;
 	    $fname = $2 if defined($2) ;
 	    $fname = $3 if defined($3) ;
@@ -1669,6 +1835,7 @@
             # content of file becomes replacement value (use recursion), add \newpage if the command was include
             ###$replacement=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
 	    $replacement=flatten(read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding), $preamble,$filename,$encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
+	    $replacement = remove_endinput($replacement); 
 	    # \include always starts a new page; use explicit \newpage command to simulate this
 	    $newpage=(defined($3)? " \\newpage " : "") ;
 	    "$begline$newpage$replacement$newpage";
@@ -1697,6 +1864,7 @@
            $subfile=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
            ($subpreamble,$subbody,$subpost)=splitdoc($subfile,'\\\\begin\{document\}','\\\\end\{document\}');
 	   $replacement=flatten($subbody, $preamble,$filename,$encoding);
+	   $replacement = remove_endinput($replacement); 
 	   "$begline$replacement";
   }/exgm;
 
@@ -2340,7 +2508,7 @@
       if ( $word =~ /^\\(?!\()(\\|[`'^"~=.]|[\w*@]+)(.*?)(\s*)$/s &&  iscmd($1,\@MBOXCMDLIST,\@MBOXCMDEXCL)) {
 	# $word is a safe command in MBOXCMDLIST
 	###print STDERR "DEBUG Mboxsafecmd detected:$word:\n" if $debug ;
-	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}$AUXCMD\n" );
+	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}\\hspace{0pt}$AUXCMD\n" );
       } else {
 	# $word is a normal word or a safe command (not in MBOXCMDLIST)
 	push (@$retval,$word);
@@ -2357,47 +2525,49 @@
 
 #used in preprocess
 sub take_comments_and_enter_from_frac() {
-      ###*************take the \n and % between frac and {}***********
-      ###notice all of the substitution are made none global
-      while( m/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/s ) {
-	### if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
-	if( $2 !~ m/\\end{$1}/s ) {
-	  ### take out % and \n from the next match only (none global)
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
+      #*************take the \n and % between frac and {}***********
+      #notice all of the substitution are made none global
+      while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/s ) {
+	# if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
+	### print STDERR "Match the following in take_comments and_enter_from_frac(1):\n****$&****\n" if $debug;
+	if( $2 !~ m/\\end\{$1}/s ) {
+	  # take out % and \n from the next match only (none global)
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
 	}
 	else{
-	  ###there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+	  #there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
 	}
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       ###*************take the \n and % between frac and {}***********
       
       ###**********take the \n and % between {} and {} of the frac***************
       while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/s ) {
-	### if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	# if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	###print STDERR "Match the following in take_comments and_enter_from_frac(2):\n****$&****\n" if $debug;
         if( $2 !~ m/\\end\{$1\}/s ) {
-          ### from now on CURRFRAC is the frac we are looking at
+          # from now on CURRFRAC is the frac we are looking at
           s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\\end\{$1\}/s;
           while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{(.*?)\\end\{\1\}/s ) {
-             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end{\1}/s ) {
+             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1}/s ) {
                    s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\}\{$6\\end\{$1\}/s;
                } 
               else { # there is no comment or \n between the two brackets {}{}
-              ### change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
-                   s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
+              # change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
+                   s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end\{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
               }
             }
         }
         else{
           ###there are no more frac in $2, we want to find the next one so we clear the begin-end from the pattern
-          s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+          s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
         }
         
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       s/FRACSTART/\\frac/g;
       ###***************take the \n and % between {} and {} of the frac*********************
 }
@@ -2410,8 +2580,8 @@
 # 2. mark all first empty line (in block of several) with \PAR tokens
 # 3. Convert all '\%' into '\PERCENTAGE ' and all '\$' into \DOLLAR to make parsing regular expressions easier
 # 4. Convert all \verb|some verbatim text| commands (where | can be an arbitrary character)
-#    into \verb{hash}
-# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}
+#    into \verb{hash}  (also lstinline)
+# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}  (not only verbatim, all patterns matching VERBATIMENV)
 # 6. Convert _n into \SUBSCRIPTNB{n} and _{nnn} into \SUBSCRIPT{nn}
 # 7. Convert ^n into \SUPERSCRIPTNB{n} and ^{nnn} into \SUPERSCRIPT{nn}
 # 8. a. Convert $$ $$ into \begin{DOLLARDOLLAR} \end{DOLLARDOLLAR}
@@ -2427,6 +2597,17 @@
 # Returns: leading white space removed in step 1
 sub preprocess {
   for (@_) { 
+
+
+    # change in \verb and similar commands - note that I introduce an extra space here so that the 
+    #       already hashed variants do not trigger again
+    # transform \lstinline{...}
+#    s/\\lstinline(\[$brat0\])?(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+#    s/\\lstinline(\[$brat0\])?((\S).*?\2)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)((\S).*?\3)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\(verb\*?|lstinline)(\S)(.*?)\2/"\\DIF${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
+
     #    Change \{ to \QLEFTBRACE, \} to \QRIGHTBRACE, and \& to \AMPERSAND
     s/(?<!\\)\\\{/\\QLEFTBRACE /sg;
     s/(?<!\\)\\\}/\\QRIGHTBRACE /sg;
@@ -2434,12 +2615,10 @@
 # replace {,} in comments with \\CLEFTBRACE,\\CRIGHTBRACE
     1 while s/((?<!\\)%.*)\{(.*)$/$1\\CLEFTBRACE $2/mg ;
     1 while s/((?<!\\)%.*)\}(.*)$/$1\\CRIGHTBRACE $2/mg ;
-
     s/\n(\s*?)\n((?:\s*\n)*)/\n$1\\PAR\n$2/g ;
     s/(?<!\\)\\%/\\PERCENTAGE /g ;  # (?<! is negative lookbehind assertion to prevent \\% from being converted
     s/(?<!\\)\\\$/\\DOLLAR /g ;  # (?<! is negative lookbehind assertion to prevent \\$ from being converted
-    s/(\\verb\*?)(\S)(.*?)\2/"${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
-    s/\\begin\{(verbatim\*?)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
+    s/\\begin\{($VERBATIMENV)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
     # Convert _n or _\cmd into \SUBSCRIPTNB{n} or \SUBSCRIPTNB{\cmd} and _{nnn} into \SUBSCRIPT{nn}
     1 while s/(?<!\\)_(\s*([^{\\\s]|\\\w+))/\\SUBSCRIPTNB{$1}/g ;
     1 while s/(?<!\\)_(\s*{($pat_n)})/\\SUBSCRIPT$1/g ;
@@ -2455,17 +2634,20 @@
     s/(?<!\\)\\\[/\\begin{SQUAREBRACKET}/sg;
     s/\\\]/\\end{SQUAREBRACKET}/sg;
     # Convert all picture environmentent (\begin{PICTUREENV} .. \end{PICTUREENV} \PICTUREBLOCKenv
-    s/\\begin{($PICTUREENV)}(.*?)\\end{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
+    s/\\begin\{($PICTUREENV)}(.*?)\\end\{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
     #    For --block-math-markup option -convert all \begin{MATH} .. \end{MATH}
     #    into \MATHBLOCKMATH{...} commands, where MATH is any valid math environment
     #    Also convert all array environments into ARRAYBLOCK environments
+
     if ( $mathmarkup != FINE ) {
-      s/\\begin{($ARRENV)}(.*?)\\end{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
+      s/\\begin\{($ARRENV)}(.*?)\\end\{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
       
       take_comments_and_enter_from_frac();
       
-      s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/\\MATHBLOCK$1\{$2\}/sg;
+
+      s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\end\{\1\}/\\MATHBLOCK$1\{$2\}/sg;
     }
+
     # add final token " STOP"
     $_ .= " STOP"
   }
@@ -2574,6 +2756,7 @@
     # Replace \RIGHTBRACE by }    
     s/\\RIGHTBRACE/}/g;
 
+
     # Check all deleted blocks: where a deleted block contains a matching \begin and
     #    \end environment (these will be disabled by a %DIFDELCMD statements), enable
     #    these commands again (such that for example displayed math in a deleted equation
@@ -2665,15 +2848,17 @@
 #    bb. disable active labels within deleted blocks (as these are not safe commands, this should normally only
 #        happen within deleted maths blocks
       $delblock=~ s/(?<!$DELCMDOPEN)(\\$LABELCMD(?:${extraspace})\{(?:[^{}])*\}[\t ]*)\n?/${DELCMDOPEN}$1${DELCMDCLOSE}/smg ;
+      # previous line causes trouble as by issue #90 I might need to modify this
 
 
 #     c. If in-line math mode contains array environment, enclose the whole environment in \mbox'es
       while ( $delblock =~ m/($math)(\s*)/sg ) {
+#	      print STDERR "DEBUG Delblock Match math $& at ",pos,"\n";
 	$cnt2=0;
 	$len2=length($&);
 	$begin2=pos($delblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/ );
 	substr($delblock,$begin2,$len2)=$mathblock;
 	pos($delblock) = $begin2 + length($mathblock);
       }
@@ -2682,11 +2867,22 @@
 	# note additional \newline after command is omitted from output if right at the end of deleted block (otherwise a spurious empty line is generated)
 	$delblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+      ###  #     change included verbatim environments
+      ###  $delblock =~ s/\\DIFverb\{/\\DIFDIFdelverb\{/g;
+      ###  $delblock =~ s/\\DIFlstinline/\\DIFDIFdellstinline/g;
+      ###}
+      # Mark deleted verbose commands
+      $delblock =~ s/(${DELCMDOPEN}\\DIF((?:verb\*?|lstinline(?:\[$brat0\])?)\{([-\d]*?)\}\s*).*)$/%\n\\DIFDIFdel$2${AUXCMD}\n$1/gm;
 
 #     splice in modified delblock
       substr($_,$begin,$len)=$delblock;
       pos = $begin + length($delblock);
     }
+
+    ### print STDERR "<<<$_>>>\n" if $debug;
+
+
     # make the array modification in added blocks
     while ( m/\\DIFaddbegin.*?\\DIFaddend/sg ) {
       $cnt=0;
@@ -2698,7 +2894,7 @@
 	$len2=length($&);
 	$begin2=pos($addblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/) ;
 	substr($addblock,$begin2,$len2)=$mathblock;
 	pos($addblock) = $begin2 + length($mathblock);
       }
@@ -2707,6 +2903,11 @@
 	##$addblock=~s/($math)/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
 	$addblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+	# mark added verbatim commands
+      $addblock =~ s/\\DIFverb/\\DIFDIFaddverb/g;
+      $addblock =~ s/\\DIFlstinline/\\DIFDIFaddlstinline/g;
+      ###}
 #     splice in modified addblock
       substr($_,$begin,$len)=$addblock;
       pos = $begin + length($addblock);
@@ -2750,8 +2951,8 @@
     # environments) are between the \begin{$MATHENV} and \end{MATHMODE} commands. This is necessary as the minimal matching 
     # is not globally minimal but only 'locally' (matching is beginning from the left side of the string)
     if ( $mathmarkup == FINE ) {
-      1 while s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin{MATHMODE})))*?)\\end{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
-      1 while s/\\begin{MATHMODE}((?:.(?!\\end{MATHMODE}))*?)\\end{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
+      1 while s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end\{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin\{MATHMODE})))*?)\\end\{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
+      1 while s/\\begin\{MATHMODE}((?:.(?!\\end\{MATHMODE}))*?)\\end\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
       # convert remaining \begin{MATHMODE} \end{MATHMODE} (and not containing & or \\ )into MATHREPL environments
       s/\\begin\{MATHMODE\}((?:(.(?!(?<!\\)\&|\\\\))*)?)\\end\{MATHMODE\}/\\begin{$MATHREPL}$1\\end{$MATHREPL}/sg;
       # others into MATHARRREPL
@@ -2758,7 +2959,7 @@
       s/\\begin\{MATHMODE\}(.*?)\\end\{MATHMODE\}/\\begin{$MATHARRREPL}$1\\end{$MATHARRREPL}/sg;
 
       # now look for AUXCMD math-mode pairs which have only comments (or empty lines between them), and remove the added commands
-      s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
+      s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
     } else {
       #   math modes OFF,WHOLE,COARSE: Convert \MATHBLOCKmath{..} commands back to environments
       s/\\MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{($pat_n)\}/\\begin{$1}$2\\end{$1}/sg;
@@ -2806,13 +3007,8 @@
     s/^\Q${DELCMDOPEN}\E\n//msg;
 
     # Expand hashes of verb and verbatim environments (note negative look behind assertion to not leak out of DIFDELCMD comments
-    s/(\\verb\*?)\{([-\d]*?)\}/"${1}". fromhash(\%verbhash,$2)/esg;
-    s/${DELCMDOPEN}\\(verbatim\*?)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
-    s/\\(verbatim\*?)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
-    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
-    s/\\PERCENTAGE ?/\\%/g;
-    # Convert '\DOLLAR ' back into '\$'			       
-    s/\\DOLLAR /\\\$/g;
+    s/${DELCMDOPEN}\\($VERBATIMENV)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
+    s/\\($VERBATIMENV)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
     # remove all \PAR tokens (taking care to properly keep commented out PAR's
     # from introducing uncommented newlines - next line)
     s/(%DIF < )([^\n]*?)\\PAR\n/$1$2\n$1\n/sg;
@@ -2828,11 +3024,17 @@
       print STDERR "endfloat package detected.\n" if $verbose ;
       # eliminate whitespace before and after
       s/^(\s*)(\\(?:begin|end)\{(?:figure|table)\})(\s*)$/$2/mg;
-      # split lines with remaining characters before float enviroment conmmand
-      s/^(.+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
-      # split lines with remaining characters after float enviroment conmmand
-      s/(\\(?:begin|end)\{(?:figure|table)\})(.+)$/$1\n$2/mg;
+      # split lines with remaining characters before float environment conmmand
+      s/^([^%]+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
+      # split lines with remaining characters after float environment conmmand
+      s/^((?:[^%]+)\\(?:begin|end)\{(?:figure|table)\}(?:\[[a-zA-Z]+\])?)(.+)((?:%.*)?)$/$1\n$2$3/mg;
     }
+
+    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
+    s/\\PERCENTAGE ?/\\%/g;
+    # Convert '\DOLLAR ' back into '\$'			       
+    s/\\DOLLAR /\\\$/g;
+
     # undo renaming of the \begin and \end,{,}  and dollars in comments 
     1 while s/(%.*)DOLLARDIF/$1\$/mg ;
 #   Convert \begin{SQUAREBRACKET} \end{SQUAREBRACKET} into \[ \]
@@ -2858,6 +3060,21 @@
     s/\\QLEFTBRACE /\\\{/sg;
     s/\\QRIGHTBRACE /\\\}/sg;
     s/\\AMPERSAND /\\&/sg;
+    # Highligh added inline verbatim commands if possible
+    if ( $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/ )   { 
+      # wrap added verb commands with color commands
+      s/\\DIFDIFadd((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*)/\{\\color{blue}$AUXCMD\n\\DIF$1%\n\}$AUXCMD\n/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD)/\{\\color{red}${AUXCMD}\n\\DIF$1\n\}${AUXCMD}/sg;
+    } else {
+      # currently if colour markup is not used just remove the added mark
+      s/\\DIFDIFadd(verb\*?|lstinline)/\\DIF$1/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD\n)//sg;
+    }
+    # expand \verb and friends inline arguments
+    s/\\DIF((?:DIFadd|DIFdel)?(?:verb\*?|lstinline(?:\[$brat0\])?))\{([-\d]*?)\}/"\\${1}". fromhash(\%verbhash,$2)/esg;
+    # add basicstyle color{blue} to added lstinline commands
+    # finally add the comment to the ones not having an optional argument before
+    ###s/\\DIFaddlstinline(?!\[)/\\lstinline\n[basicstyle=\\color{blue}]$AUXCMD\n/g;
 
   return;
   }
@@ -2868,7 +3085,7 @@
 sub restore_item_commands {
   my ($string)=@_ ;
   my ($itemarg, at itemargs);
-  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*?(?:\n|$DELCMDCLOSE))/
+  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*((?:${cmdoptseq}\s*?)*)(?:\n|$DELCMDCLOSE))/
      # if \item has an []argument, then mark up the argument as deleted)
      if (length($4)>0) {
        # use substr to exclude square brackets at end points
@@ -2877,7 +3094,8 @@
      } else {
        $itemarg="";
      }
-     "$1$2$3$itemarg$AUXCMD\n"/sge;
+     "$1$2$3$itemarg$AUXCMD\n";  ###.((length($5)>0) ? "%DIFDELCMD $5 $DELCMDCLOSE\n" : "")
+     /sge;
   return($string);
 }
 
@@ -3060,16 +3278,32 @@
 # init_regex_arr_data(\@array,"TOKEN INIT")
 # scans DATA file handel for line "%% TOKEN INIT" line
 # then appends each line not beginning with % into array (as a quoted regex)
+# This is used for command lists and configuration variables, but the processing is slightly 
+# different: 
+# For lists, the regular expression is extended to include beginning (^) and end ($) markers, to require full-string matching
+# For configuration variables (and all others), simply an unadorned list is copied
 sub init_regex_arr_data {
   my ($arr,$token)=@_;
-  my ($copy);
+  my $copy=0;
+  my ($mode);
+  if ($token =~ m/COMMANDS/ ) { 
+    $mode=0;
+  } else {
+    $mode=1;
+  }
+
   while (<DATA>) {
     if ( m/^%%BEGIN $token\s*$/ ) {
-      $copy=1; }
-    elsif ( m/^%%END $token\s*/ )  {
+      $copy=1; 
+      next;
+    } elsif ( m/^%%END $token\s*$/ )  {
       last; }
     chomp;
-    push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    if ( $mode==0 ) {
+      push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    } elsif ($mode==1) {
+      push (@$arr,"$_") if ( $copy && !/^%/ ) ;
+    }
   }
   seek DATA,0,0;    # rewind DATA handle to file begin
 }
@@ -3082,25 +3316,42 @@
 # Otherwise treat arg as a comma separated list of regular expressions
 sub init_regex_arr_ext {
   my ($arr,$arg)=@_;
-  my $regex;
-  if ( -f $ arg ) {
-    open(FILE,"$arg") or die ("Couldn't open $arg: $!");
-    while (<FILE>) {
-      chomp;
-      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
-      push (@$arr,qr/^$_$/);
-    }
-    close(FILE);
+  if ( -f $arg ) {
+    init_regex_arr_file($arr,$arg);
+  } else {
+    init_regex_arr_list($arr,$arg);
   }
-  else {
-    # assume it is a comma-separated list of reg-ex
-    foreach $regex (split(qr/(?<!\\),/,$arg)) {
-      $regex =~ s/\\,/,/g;
-      push (@$arr,qr/^$regex$/);
-    }
+}
+
+# init_regex_arr_file(\@array,$fname)
+# appends array with regular expressions.
+# Read in list of regular expressions from $fname
+# (one expression per line)
+sub init_regex_arr_file {
+  my ($arr,$fname)=@_;
+  open(FILE,"$fname") or die ("Couldn't open $fname: $!");
+  while (<FILE>) {
+    chomp;
+    next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+    push (@$arr,qr/^$_$/);
   }
+  close(FILE);
 }
 
+# init_regex_arr_list(\@array,$arg)
+# appends array with regular expressions.
+# read from comma separated list of regular expressions ($arg)
+sub init_regex_arr_list {
+  my ($arr,$arg)=@_;
+  my $regex;
+  ###    print STDERR "DEBUG init_regex_arr_list arg >$arg<\n" if $debug;
+  foreach $regex (split(qr/(?<!\\),/,$arg)) {
+    $regex =~ s/\\,/,/g;
+    push (@$arr,qr/^$regex$/);
+  }
+}
+
+
 #exetime() returns time since last execution of this command
 #exetime(1) resets this time
 my $lasttime=-1;   # global variable for persistence
@@ -3242,6 +3493,14 @@
                           SCALEDELGRAPHICS (Float)
                        This option can be repeated.
 
+--add-to-config  varenv1=pattern1,varenv2=pattern2
+                       For configuration variables containing a regular expression (essentially those ending
+                       in ENV, and COUNTERCMD) this provides an alternative way to modify the configuration 
+                       variables. Instead of setting the complete pattern, with this option it is possible to add an
+                       alternative pattern. varenv must be one of the variables listed above that take a regular
+                       expression as argument, and pattern is any regular expression (which might need to be 
+                       protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+                       to separate them, e.g. --add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote"
 
 --packages=pkg1,pkg2,..
                        Tell latexdiff that .tex file is processed with the packages in list 
@@ -3660,11 +3919,12 @@
 =item B<--replace-context1cmd=replace-file> 
 
 =item B<--append-context1cmd=append-file> or
+
 =item B<--append-context1cmd="cmd1,cmd2,...">
 
 Replace or append to the list of regex matching commands
 whose last argument is text but which require a particular
-context to work, e.g. \caption will only work within a figure
+context to work, e.g. C<\caption> will only work within a figure
 or table.  These commands behave like text commands, except when 
 they occur in a deleted section, when they are disabled, but their
 argument is shown as deleted text.
@@ -3672,6 +3932,7 @@
 =item B<--replace-context2cmd=replace-file> 
 
 =item B<--append-context2cmd=append-file> or
+
 =item B<--append-context2cmd="cmd1,cmd2,...">
 
 As corresponding commands for context1.  The only difference is that
@@ -3684,7 +3945,7 @@
 =item B<--append-mboxsafecmd=append-file> or B<--append-mboxsafecmd="cmd1,cmd2,...">
 
 Define safe commands, which additionally need to be protected by encapsulating
-in an \\mbox{..}. This is sometimes needed to get around incompatibilities 
+in an C<\mbox{..}>. This is sometimes needed to get around incompatibilities 
 between external packages and the ulem package, which is  used for highlighting
 in the default style UNDERLINE as well as CULINECHBAR CFONTSTRIKE
 
@@ -3724,6 +3985,16 @@
 
 C<SCALEDELGRAPHICS> (Float)
 
+=item B<--add-to-config varenv1=pattern1,varenv2=pattern2,...>
+
+For configuration variables, which are a regular expression (essentially those ending
+in ENV, and COUNTERCMD, see list above) this provides an alternative way to modify the configuration 
+variables. Instead of setting the complete pattern, with this option it is possible to add an
+alternative pattern. C<varenv> must be one of the variables listed above that take a regular
+expression as argument, and pattern is any regular expression (which might need to be 
+protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+to separate them, e.g. C<--add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote">
+
 =item B<--show-safecmd>
 
 Print list of RegEx matching and excluding safe commands.
@@ -3784,16 +4055,17 @@
 =item B<--graphics-markup=level>
 
  Change highlight style for graphics embedded with C<\includegraphics> commands.
-                      Possible values for level:
 
-                      C<none, C<off> or C<0>: no highlighting for figures
+Possible values for level:
 
-                      C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
+C<none>, C<off> or C<0>: no highlighting for figures
 
-                      C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
-                                scale, and crossed out with a red diagonal cross. Use configuration
-                                variable SCALEDELGRAPHICS to set size of deleted figures.
+C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
 
+C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
+scale, and crossed out with a red diagonal cross. Use configuration
+variable SCALEDELGRAPHICS to set size of deleted figures.
+
 Note that changes to the optional parameters will make the figure appear as changed 
 to latexdiff, and this figure will thus be highlighted
 
@@ -3849,7 +4121,7 @@
 
 Suppress inclusion of old and new file names as comment in output file
 
-=item B<--visble-label>
+=item B<--visible-label>
 
 Include old and new filenames (or labels set with C<--label> option) as 
 visible output.
@@ -4158,8 +4430,8 @@
 
 =head1 AUTHOR
 
-Version 1.2.0
-Copyright (C) 2004-2015 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2004-2017 Frederik Tilmann
 
 This program is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License Version 3
@@ -4441,9 +4713,68 @@
 institute
 %%END CONTEXT2 COMMANDS 
 
+%% CONFIGURATION variabe defaults
+%%BEGIN LISTENV CONFIG
+itemize
+description
+enumerate
+%%END LISTENV CONFIG
 
-%% TYPES (Commands for highlighting changed blocks)
+%%BEGIN FLOATENV CONFIG
+figure[\w\d*@]*
+table[\w\d*@]*
+plate[\w\d*@]*
+%%END FLOATENV CONFIG
 
+%%BEGIN PICTUREENV CONFIG
+picture[\w\d*@]*
+tikzpicture[\w\d*@]*
+DIFnomarkup
+%%END PICTUREENV CONFIG
+
+%%BEGIN MATHENV CONFIG
+equation[*]?
+displaymath
+DOLLARDOLLAR
+%%END MATHENV CONFIG
+
+%%BEGIN MATHARRENV CONFIG
+eqnarray[*]?
+align[*]?
+alignat[*]?
+gather[*]?
+multline[*]?
+flalign[*]?
+%%END MATHARRENV CONFIG
+
+%%BEGIN ARRENV CONFIG
+aligned
+array
+[pbvBV]?matrix
+smallmatrix
+cases
+split
+%%END ARRENV CONFIG
+
+%%BEGIN COUNTERCMD CONFIG
+footnote
+part
+chapter
+section
+subsection
+subsubsection
+paragraph
+subparagraph
+%%END COUNTERCMD CONFIG
+
+%%BEGIN VERBATIMENV CONFIG
+verbatim[*]?
+lstlisting
+comment
+%%END VERBATIMENV CONFIG
+
+%%% TYPES (Commands for highlighting changed blocks)
+
 %DIF UNDERLINE PREAMBLE
 \RequirePackage[normalem]{ulem}
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
@@ -4454,6 +4785,7 @@
 %DIF CTRADITIONAL PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} [..\footnote{removed: #1} ]}}
 %DIF END CTRADITIONAL PREAMBLE
@@ -4460,6 +4792,7 @@
 
 %DIF TRADITIONAL PREAMBLE
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{[..\footnote{removed: #1} ]}}
 %DIF END TRADITIONAL PREAMBLE
@@ -4466,6 +4799,7 @@
 
 %DIF CFONT PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} \scriptsize #1}}
 %DIF END CFONT PREAMBLE
@@ -4472,6 +4806,7 @@
 
 %DIF FONTSTRIKE PREAMBLE
 \RequirePackage[normalem]{ulem}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{\footnotesize \sout{#1}}}
 %DIF END FONTSTRIKE PREAMBLE
@@ -4510,6 +4845,7 @@
 %DIF END INVISIBLE PREAMBLE
 
 %DIF BOLD PREAMBLE
+\DeclareOldFontCommand{\bf}{\normalfont\bfseries}{\mathbf}
 \providecommand{\DIFadd}[1]{{\bf #1}}
 \providecommand{\DIFdel}[1]{}
 %DIF END BOLD PREAMBLE
@@ -4534,7 +4870,7 @@
 \providecommand{\DIFaddend}{\protect\marginpar{]}}
 \providecommand{\DIFdelbegin}{\protect\marginpar{d[}}
 \providecommand{\DIFdelend}{\protect\marginpar{]}}
-%DIF END BRACKET PREAMBLE
+%DIF END MARGIN PREAMBLE
 
 %DIF DVIPSCOL PREAMBLE
 %Note: only works with dvips converter
@@ -4727,4 +5063,18 @@
 %DIF HYPERREF PREAMBLE
 \providecommand{\DIFadd}[1]{\texorpdfstring{\DIFaddtex{#1}}{#1}}
 \providecommand{\DIFdel}[1]{\texorpdfstring{\DIFdeltex{#1}}{}}
-%DIF END HYPERREF PACKAGE
+%DIF END HYPERREF PREAMBLE
+
+%DIF LISTINGS PREAMBLE
+\lstdefinelanguage{codediff}{
+  moredelim=**[is][\color{red}]{*!----}{----!*},
+  moredelim=**[is][\color{blue}]{*!++++}{++++!*}
+}
+\lstdefinestyle{codediff}{
+	belowcaptionskip=.25\baselineskip,
+	language=codediff,
+	basicstyle=\ttfamily,
+	columns=fullflexible,
+	keepspaces=true,
+}
+%DIF END LISTINGS PREAMBLE

Modified: trunk/Master/texmf-dist/scripts/latexdiff/latexdiff-vc.pl
===================================================================
--- trunk/Master/texmf-dist/scripts/latexdiff/latexdiff-vc.pl	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/scripts/latexdiff/latexdiff-vc.pl	2017-06-26 21:22:49 UTC (rev 44704)
@@ -27,7 +27,12 @@
 #
 # TODO/IDEAS: - option to call external pre-processing codes
 #             - choose type of latex processor / bibtex (luatex, xelatex etc)
-# versio 1.2.0:
+# version 1.2.1 (22 June 2017)
+#    - bug fix: --hg option was not recognised (partially fixes github issue #93 )
+#    - wrap passed-through options to latexdiff in quotation marks (fix github issue #58 )
+#    - program names for latexdiff, latex, dvips, bibtex configurable (fixes issue #40)
+#
+# version 1.2.0:
 #    - depracation fix: left brace in RegEx now needs to be escaped
 #
 # version 1.1.1:
@@ -63,8 +68,8 @@
 use warnings ;
 
 my $versionstring=<<EOF ;
-This is LATEXDIFF-VC 1.2.0
-  (c) 2005-2016 F J Tilmann
+This is LATEXDIFF-VC 1.2.1
+  (c) 2005-2017 F J Tilmann
 EOF
 
 # output debug and intermediate files, set to 0 in final distribution
@@ -72,10 +77,14 @@
 
 
 # Option names
-my ($version,$help,$fast,$so,$postscript,$pdf,$onlychanges,$flatten,$force,$dir,$cvs,$rcs,$svn,$git,$hg,$diffcmd,$patchcmd, at revs);
+my ($version,$help,$fast,$showconfig,$so,$postscript,$pdf,$onlychanges,$flatten,$force,$dir,$cvs,$rcs,$svn,$git,$hg,
+    $run,$rundvi2,$diffcmd,$patchcmd, at revs, at configlist);
+# for passing through config options to latexdiff:
+my ($configlatexdiff, at config,$config,$assign);
+
 # Preset Variables
-my $latexdiff="latexdiff"; # Program for making the comparison
-my $latexcmd="latex"; # latex compiler if not further identified
+#my $latexdiff="latexdiff"; # Program for making the comparison
+#my $latexcmd="latex"; # latex compiler if not further identified
 my $vc="";
 my $tempdir=tempdir(CLEANUP => 1);   # generate a temp dir, which will automatically be deleted at program exit
 # Variables
@@ -82,6 +91,17 @@
 my ($file1,$file2,$diff,$diffbase,$rootdir,$answer,$options,$infile,$append,$dirname,$cwd);
 my (@files, at ldoptions, at tmpfiles, at ptmpfiles, at difffiles,$extracomp); # ,
 
+
+# keep commands to be used in config hash
+my (%CFG)=( 
+   LATEX     => "latex",
+   LATEXDIFF => "latexdiff",
+   DVI2      => "dvips",
+   BIBTEX    => "bibtex"
+   );
+
+
+
 Getopt::Long::Configure('pass_through','bundling');
 
 GetOptions('revision|r:s' => \@revs,
@@ -91,6 +111,8 @@
            'git' => \$git,
            'hg' => \$hg,
            'dir|d:s' => \$dir,
+	   'run' => \$run,
+	   'dvi' => \$rundvi2,
 	   'fast' => \$fast,
 	   'so' => \$so,
            'postscript|ps' => \$postscript,
@@ -98,6 +120,8 @@
            'force' => \$force,
 	   'only-changes' => \$onlychanges,
 	   'flatten:s' => \$flatten,
+	   'config|c=s' => \@configlist,
+	   'show-config' => \$showconfig,
            'version' => \$version,
 	   'help|h' => \$help,
 	   'debug!' => \$debug);
@@ -116,11 +140,11 @@
 }
 
 if ( $so ) {
-  $latexdiff='latexdiff-so';
+  $CFG{LATEXDIFF}='latexdiff-so';
 }
 if ( $fast ) { 
   die "Cannot specify more than one of --fast or --so " if ($so);
-  $latexdiff='latexdiff-fast';
+  $CFG{LATEXDIFF}='latexdiff-fast';
 }
 
 if ( $cvs ) {
@@ -138,8 +162,67 @@
   die "Cannot specify more than one of --cvs, --rcs, --svn --git or --hg." if ($vc);
   $vc="GIT";
 }
+if ( $hg ) {
+  die "Cannot specify more than one of --cvs, --rcs, --svn --git or --hg." if ($vc);
+  $vc="HG";
+}
 
+if ( $pdf ) {
+  $CFG{LATEX} = "pdflatex";
+  $run=1;
+} elsif ( $postscript ) {
+  $CFG{LATEX} = "latex";
+  $rundvi2=1;
+}
 
+if ( $rundvi2 ) {
+  $run=1;
+}
+
+# setting config variables from command line. Explicitly set configuration options take precedence over convenience options,
+# therefore this is executed last
+ at config=();
+foreach $config ( @configlist ) {
+  if (-f $config || lc $config eq '/dev/null' ) {
+    open(FILE,$config) or die ("Couldn't open configuration file $config: $!");
+    while (<FILE>) {
+      chomp;
+      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+      push (@config,$_);
+    }
+    close(FILE);
+  }
+  else {
+#    foreach ( split(",",$config) ) {
+#      push @config,$_;
+#    }
+     push @config,split(",",$config)
+  }
+}
+
+$configlatexdiff="";
+foreach $assign ( @config ) {
+  $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
+  if ( defined($CFG{$1})) {
+    # known latexdiff-vc option
+    $CFG{$1}=$2;
+  } else {
+    # unknown option, assume it is a latexdiff configuration variable
+    $configlatexdiff .= "$1=$2,";
+  }
+}
+# delete last comma inv $configlatexdiff if present
+$configlatexdiff =~ s/,$// ;
+
+if ($showconfig) {
+  print "# Configuration variables (latexdiff-vc):\n";
+  foreach ( keys %CFG ) {
+    print "$_=$CFG{$_}\n";
+  }
+  exit 0;
+}
+
+
 # check whether the first file name or first passed-through option for latexdiff got misinterpreted as an option to an empty -r option
 if ( @revs && ( -f $revs[$#revs] || $revs[$#revs] =~ /^-/ ) ) {
   push @ARGV,$revs[$#revs];
@@ -156,7 +239,7 @@
   $flatten="";
 }
 
-print "DEBUG: PDF $pdf latexdiff-vc command line: ", join(" ", at ARGV), "\n" if $debug; 
+print STDERR "DEBUG: PDF $pdf latexdiff-vc command line: ", join(" ", at ARGV), "\n" if $debug; 
 
 $file2=pop @ARGV;
 ( defined($file2) && $file2 =~ /\.(tex|bbl|flt)$/ ) or pod2usage("Must specify at least one tex, bbl or flt file");
@@ -217,7 +300,7 @@
     # $diffcmd  = "git diff ";
     # $patchcmd = "patch -R -p1";
   } elsif ( $vc eq "HG" ) {
-    $diffcmd  = "hg diff -r";
+    $diffcmd  = "hg diff --root . -r";
     $patchcmd = "patch -R -p1";
   } else {
     print STDERR "Unknown versioning system $vc \n";
@@ -241,6 +324,10 @@
   }
 }
 
+if ( length $configlatexdiff >0 ) {
+  push @ldoptions, "--config $configlatexdiff";
+}
+
 if ( defined($flatten) ) {
   push @ldoptions, "--flatten" ;
 }
@@ -252,15 +339,15 @@
 
 if ( scalar(@revs) == 0 ) {
   pod2usage("When -r option is not used, two .tex files (old and new) must be given on the command line") unless @files==2;
-  warn "Option -flatten should normally be combined with -r option (Results will probably not be as expected)" if $flatten;
+  warn "Option -flatten should normally be combined with -r option (Results will probably not be as expected)" if defined($flatten);
     #  compare two files
   $file1=shift @files ;
 } else {
   # revision control
-  if ( $flatten ) {
+  if ( defined($flatten) ) {
     pod2usage("Only one root file must be given  if --flatten option is used with version control") if  @files != 1;
     $tempdir='.' if ( $flatten eq "keep-intermediate" );
-    print "flatten tempdir |$tempdir|\n";
+    print STDERR "flatten tempdir |$tempdir|\n" if $debug;
     if ( $vc eq "SVN" ) {
       my (@infoout)=`svn info`;
       my (@urlline) = grep(/^URL:/, @infoout);
@@ -293,8 +380,14 @@
   length($revs[0]) > 0 or $revs[0]="HEAD";
 }
 
+if ($vc eq "HG" && scalar(@revs)) {
+  length($revs[$#revs]) > 0 or $revs[$#revs]="tip";
+  length($revs[0]) > 0 or $revs[0]="tip";
+}
 
+print STDERR "DEBUG LDOptions($#ldoptions): " . join(":", at ldoptions) . "\n" if $debug;
 
+
 # cycle through all files
 @difffiles=();
 
@@ -357,7 +450,11 @@
   system("mkdir -p $dirname") unless ( -e $dirname );
 
   # Remaining options are passed to latexdiff
-  $options = join(" ", at ldoptions);
+  if (scalar(@ldoptions) > 0 ) {
+    $options = "\'" . join("\' \'", at ldoptions) . "\'";
+  } else {
+    $options = "";
+  }
    
   if ( -e $diff && ! $force ) {
     print STDERR "OK to overwrite existing file $diff (y/n)? ";
@@ -367,13 +464,13 @@
       die "Abort ... " ;
     }
   }
-  print "Running: $latexdiff  $options \"$file1\" \"$file2\" > \"$diff\"\n";
-  unless ( system("$latexdiff $options \"$file1\" \"$file2\" > \"$diff\"") == 0 ) { 
-    print STDERR  "Something went wrong in $latexdiff. Deleting $diff and abort\n" ; unlink $diff ; exit(5) 
+  print STDERR "Running: $CFG{LATEXDIFF}  $options \"$file1\" \"$file2\" > \"$diff\"\n";
+  unless ( system("$CFG{LATEXDIFF} $options \"$file1\" \"$file2\" > \"$diff\"") == 0 ) { 
+    print STDERR  "Something went wrong in $CFG{LATEXDIFF}. Deleting $diff and abort\n" ; unlink $diff ; exit(5) 
   };
   print "Generated difference file $diff\n";
    
-  if ( ( $postscript or $pdf ) and !( scalar(@revs) && greptex( qr/\\document(?:class|style)/ , $diff ) ) ) {
+  if ( $run and !( scalar(@revs) && greptex( qr/\\document(?:class|style)/ , $diff ) ) ) {
     # save filename for later processing if postscript or pdf conversion is requested and either two-file mode was used (scalar(@revs)==0) or the diff file contains documentclass statement (ie. is a root document)
     push @difffiles, $diff ;
   }
@@ -393,16 +490,13 @@
   # adapt magically changebar styles to [pdftex] display driver if pdf output was selected
   if ( $pdf ) {
     system("sed \"s/Package\\[dvips\\]/Package[pdftex]/\" \"$diff\" > \"$diff.tmp$$\" ; \\mv \"$diff.tmp$$\" \"$diff\"");
-    $latexcmd = "pdflatex";
-  } elsif ( $postscript ) {
-    $latexcmd = "latex";
   }
 
-  if ( $pdf | $postscript ) {
+  if ( $run ) {
     print STDERR "PDF: $pdf Postscript: $postscript cwd $cwd\n" if $debug;
 
     if ( system("grep -q \'^[^%]*\\\\bibliography{\' \"$diff\"") == 0 ) { 
-      system("$latexcmd --interaction=batchmode \"$diff\"; bibtex \"$diffbase\";");
+      system("$CFG{LATEX} --interaction=batchmode \"$diff\"; $CFG{BIBTEX} \"$diffbase\";");
       push @ptmpfiles, "$diffbase.bbl","$diffbase.bbl" ; 
     }
 
@@ -409,30 +503,40 @@
     # if special needs, as CHANGEBAR
     if ( $extracomp ) {
       # print "Extracomp\n";
-      system("$latexcmd --interaction=batchmode \"$diff\";");
+      system("$CFG{LATEX} --interaction=batchmode \"$diff\";");
     }
 
     # final compilation
-    system("$latexcmd --interaction=batchmode \"$diff\";"); # needed if cross-refs
-    system("$latexcmd \"$diff\";"); # final, with possible error messages
+    system("$CFG{LATEX} --interaction=batchmode \"$diff\";"); # needed if cross-refs
+    system("$CFG{LATEX} \"$diff\";"); # final, with possible error messages
 
-    if ( $postscript ) {
+    if ( $rundvi2 ) {
       my $dvi="$diffbase.dvi";
-      my $ps="$diffbase.ps";
+      #my $ps="$diffbase.ps";
       my $ppoption="";
       
       if ( $onlychanges ) {
 	$ppoption="-pp ".join(",",findchangedpages("$diffbase.aux"));
       }
-      system("dvips $ppoption -o $ps $dvi");
+      #system("$CFG{DVI2} $ppoption -o $ps $dvi");
+      system("$CFG{DVI2} $ppoption $dvi");
       push @ptmpfiles, "$diffbase.aux","$diffbase.log",$dvi ;
-      print "Generated postscript file $ps\n";
-    } elsif ( $pdf ) {
+      #print STDERR "Generated postscript file $ps\n";
+    } elsif ( $run ) {
       if ( $onlychanges ) {
 	my @pages=findchangedpages("$diffbase.aux");
 	###      print ("Running pdftk \"$diffbase.pdf\" cat " . join(" ", at pages) . " output \"$diffbase-changedpage.pdf\"\n") or 
-	system ("pdftk \"$diffbase.pdf\" cat " . join(" ", at pages) . " output \"$diffbase-changedpage.pdf\"")==0 or 
-	  die ("Could not execute <pdftk $diffbase.pdf cat " . join(" ", at pages) . " output $diffbase-changedpage.pdf> . Return code: $?");
+        my $qpdf = `which qpdf`;
+        $qpdf =~ s/^\s+|\s+$//g;
+        my $pdftk = `which pdftk`;
+        $pdftk =~ s/^\s+|\s+$//g;
+        if (-x $qpdf) {
+          system("qpdf --linearize \"$diffbase.pdf\" --pages \"$diffbase.pdf\" " . join(",", @pages) . " -- \"$diffbase-changedpage.pdf\" ") == 0
+            or die("could not execute qpdf to strip pages. Return code: $?");
+        } elsif (-x $pdftk) {
+          system ("pdftk \"$diffbase.pdf\" cat " . join(" ", at pages) . " output \"$diffbase-changedpage.pdf\"")==0 or
+            die ("Could not execute <pdftk $diffbase.pdf cat " . join(" ", at pages) . " output $diffbase-changedpage.pdf> . Return code: $?");
+        }
 	move("$diffbase-changedpage.pdf","$diffbase.pdf");
       }
       push @ptmpfiles, "$diffbase.aux","$diffbase.log";
@@ -591,6 +695,51 @@
 
 With C<--flatten=keep-intermediate>, the intermediate revision snapshots are kept in the current directory (Default is to store them in a temporary directory and delete them after generating the diff file.)
 
+=item B<--config var1=val1,var2=val2,...> or B<-c var1=val1,..>
+
+
+
+=item B<--only-changes>
+
+Post-process the output such that only pages with changes on them are displayed. This requires the use of subtype ZLABEL 
+in latexdiff, which will be set automatically, but any manually set -s option will be overruled (also requires zref package to 
+be installed). (note that this option must be combined with --ps or --pdf to make sense)
+
+=item B<--force>
+
+Overwrite existing diff files without asking for confirmation. Default 
+behaviour is to ask for confirmation before overwriting an existing difference
+file.
+
+=item B<--run>
+
+run latex command on diff file after generation of diff file.
+
+=item B<--dvi>
+
+run latex and dvixxx commands after generation of diff file.
+
+=item B<-c configfile>
+
+Set configuration variables for latexdiff and latexdiff-vc.  The option can be repeated to set different
+variables (as an alternative to the comma-separated list).
+Available variables for latexdiff-vc:
+
+=over 8
+
+=item C<LATEXDIFF> latexdiff command (e.g. latexdiff-fast, latexdiff-so). This command should support the option C<--interaction=batchmode>
+
+=item C<LATEX> latex command (e.g. pdflatex, lualatex)
+
+=item C<DVI2>  Command for conversion of dvi file (e.g. dvips, dvipdf)
+
+=item C<BIBTEX> Command replacing bibtex
+
+=back
+
+All other config variables are passed to latexdiff. Explicity set configuration changes always override implicit
+changes by the following shortcut options B<--fast>, B<--so>, B<--ps> and B<--pdf>. 
+
 =item B<--fast> or B<--so>
 
 Use C<latexdiff-fast> or C<latexdiff-so>, respectively (instead of C<latexdiff>).
@@ -609,22 +758,16 @@
 Generate pdf output from difference file using C<pdflatex>. This will
 run the sequence C<pdflatex; pdflatex> on the difference file, or
 C<pdflatex; bibtex; pdflatex; pdflatex> for files requiring bibtex.
+Note that this is not just a shortcut for setting configuration variable but also triggers 
+some special behaviour.
 
-=item B<--only-changes>
 
-Post-process the output such that only pages with changes on them are displayed. This requires the use of subtype ZLABEL 
-in latexdiff, which will be set automatically, but any manually set -s option will be overruled (also requires zref package to 
-be installsed). (note that this option must be combined with --ps or --pdf to make sense)
+=item B<--show-config>
 
-=item B<--force>
+Show values of configuration variables.
 
-Overwrite existing diff files without asking for confirmation. Default 
-behaviour is to ask for confirmation before overwriting an existing difference
-file.
+=item B<--help> or B<-h>
 
-=item B<--help> or
-B<-h>
-
 Show help text
 
 =item B<--version>
@@ -654,8 +797,8 @@
 
 =head1 AUTHOR
 
-Version 1.2.0
-Copyright (C) 2005-2016 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2005-2017 Frederik Tilmann
 
 This program is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License Version 3

Modified: trunk/Master/texmf-dist/scripts/latexdiff/latexdiff.pl
===================================================================
--- trunk/Master/texmf-dist/scripts/latexdiff/latexdiff.pl	2017-06-26 21:22:32 UTC (rev 44703)
+++ trunk/Master/texmf-dist/scripts/latexdiff/latexdiff.pl	2017-06-26 21:22:49 UTC (rev 44704)
@@ -24,6 +24,28 @@
 #
 # ToDo:
 #
+# Version 1.2.1 (22 June 2017)
+#    - add "DeclareOldFontCommand" to styles using \bf or \sf old style font commands (fixies issue #92 )
+#    - improved markup: process lstinline commands in listings package correctly
+#      for styles using colour, \verb and \lstinline arguments are marked up with colour (blue for added, red for deleted)
+#    - bug fix: protecting inline math expressions for mbox did not work as intended (see stack exchange question: http://tex.stackexchange.com/questions/359412/compiling-the-latexdiff-when-adding-a-subscript-before-a-pmatrix-environment-cau)
+#    - bug fix: when deleted \item commands are followed immediately by unsafe commands, they were not restored properly
+#      (thanks to J. Protze for pull request) (pull request #89)
+#    - treat lstlisting and comment as equivalent to verbatim environment
+#      make environments that are treated like verbatim environments configurable (config variable VERBATIMENV)
+#      treat lstinlne as equivalent to verb command
+#      partially addresses issue #38
+#    - refactoring: set default configuration variables in a hash, and those that correspond to lists 
+#    - feature: option --add-to-config used to amend configuration variables, which are regex pattern lists
+#    - bug fix: deleted figures when endfloat package is activated
+#    - bug fix: alignat environment now always processed correctly (fix issues #65)
+#    - bug fix: avoid processing of commands as potential files in routine init_regex_arr (fix issue #70 )
+#    - minimal feature enhancement: treat '@' as allowed character in commands (strictly speaking requires prior \makeatletter statement, but always assuming it to be 
+#       @       a letter if it is part of a command name will usually lead to the correct behaviour (see http://tex.stackexchange.com/questions/346651/latexdiff-and-let)
+#    - new feature/bug fix: --flatten option \endinput in included files now respected but only if \endinput stands right at the beginning of the line (issue #77)
+#    - bug fix: flatten would incorrectly attempt to process commented out \include commands (from discussion in issue #77 )
+#    - introduce an invisible space (\hspace{0pt} after \mbox{..} auxiliary commands (not in math mode), to allow line breaks between added and deleted citations (change should not cause adverse behaviour otherwise)
+#
 # Version 1.2.0:
 #    - highlight new and deleted figures
 #    - bug fix in title mark-up. Previously deleted commands in title (such as \title, \author or \date) were marked up erroneously
@@ -597,34 +619,76 @@
 
 
 my ($versionstring)=<<EOF ;
-This is LATEXDIFF 1.2.0  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
-  (c) 2004-2016 F J Tilmann
+This is LATEXDIFF 1.2.1  (Algorithm::Diff $Algorithm::Diff::VERSION, Perl $^V)
+  (c) 2004-2017 F J Tilmann
 EOF
 
-# Configuration variables: these have to be visible from the subroutines
-my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
-                     # shorter identical blocks will be merged to the previous word
-my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
-                      # this is only used for --graphics-markup=BOTH option
-my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
-my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
-my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
-my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
-my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
-my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
-my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
-my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+# Hash with defaults for configuration variables. These marked undef have default values constructed from list defined in the DATA block
+# (under tag CONFIG)
+my %CONFIG=(
+   MINWORDSBLOCK => 3, # minimum number of tokens to form an independent block
+                        # shorter identical blocks will be merged to the previous word
+   SCALEDELGRAPHICS => 0.5, # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+                             # this is only used for --graphics-markup=BOTH option
+   FLOATENV => undef ,   # Environments in which FL variants of defined commands are used
+   PICTUREENV => undef ,   # Environments in which all change markup is removed
+   MATHENV => undef ,           # Environments turning on display math mode (code also knows about \[ and \])
+   MATHREPL => 'displaymath',  # Environment introducing deleted maths blocks
+   MATHARRENV => undef ,           # Environments turning on eqnarray math mode
+   MATHARRREPL => 'eqnarray*',  # Environment introducing deleted maths blocks
+   ARRENV => undef , # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+   COUNTERCMD => undef,
+                                        # COUNTERCMD textcmds which are associated with a counter
                                         # If any of these commands occur in a deleted block
                                         # they will be succeeded by an \addtocounter{...}{-1}
                                         # for the associated counter such that the overall numbers
                                         # should be the same as in the new file
+   LISTENV => undef ,  # list making environments - they will generally be kept
+   VERBATIMENV => undef,    # Environments whose content should be treated as verbatim text 
+   ITEMCMD => 'item'                    # command marking item in a list environment
+);
+# Configuration variables: these have to be visible from the subroutines
+my ($ARRENV,
+    $COUNTERCMD,
+    $FLOATENV,
+    $ITEMCMD,
+    $LISTENV,
+    $MATHARRENV,
+    $MATHARRREPL,
+    $MATHENV,
+    $MATHREPL,
+    $MINWORDSBLOCK,
+    $PICTUREENV,
+    $SCALEDELGRAPHICS,
+    $VERBATIMENV
+);
+
+# my $MINWORDSBLOCK=3; # minimum number of tokens to form an independent block
+#                      # shorter identical blocks will be merged to the previous word
+# my $SCALEDELGRAPHICS=0.5; # factor with which deleted figures will be scaled down (i.e. 0.5 implies they are shown at half linear size)
+#                       # this is only used for --graphics-markup=BOTH option
+# my $FLOATENV='(?:figure|table|plate)[\w\d*@]*' ;   # Environments in which FL variants of defined commands are used
+# my $PICTUREENV='(?:picture|tikzpicture|DIFnomarkup)[\w\d*@]*' ;   # Environments in which all change markup is removed
+# my $MATHENV='(?:equation[*]?|displaymath|DOLLARDOLLAR)[*]?' ;           # Environments turning on display math mode (code also knows about \[ and \])
+# my $MATHREPL='displaymath';  # Environment introducing deleted maths blocks
+# my $MATHARRENV='(?:eqnarray|align|alignat|gather|multline|flalign)[*]?' ;           # Environments turning on eqnarray math mode
+# my $MATHARRREPL='eqnarray*';  # Environment introducing deleted maths blocks
+# my $ARRENV='(?:aligned|array|[pbvBV]?matrix|smallmatrix|cases|split)'; # Environments making arrays in math mode.  The underlining style does not cope well with those - as a result in-text math environments are surrounded by \mbox{ } if any of these commands is used in an inline math block
+# my $COUNTERCMD='(?:footnote|part|chapter|section|subsection|subsubsection|paragraph|subparagraph)';  # textcmds which are associated with a counter
+#                                         # If any of these commands occur in a deleted block
+#                                         # they will be succeeded by an \addtocounter{...}{-1}
+#                                         # for the associated counter such that the overall numbers
+#                                         # should be the same as in the new file
+# my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
+# my $ITEMCMD='item';   # command marking item in a list environment
+
+
+
 my $LABELCMD='(?:label)';                # matching commands are disabled within deleted blocks - mostly useful for maths mode, as otherwise it would be fine to just not add those to SAFECMDLIST
 my @UNSAFEMATHCMD=('qedhere');           # Commands which are definitely unsafe for marking up in math mode (amsmath qedhere only tested to not work with UNDERLINE markup) (only affects WHOLE and COARSE math markup modes). Note that unlike text mode (or FINE math mode0 deleted unsafe commands are not deleted but simply taken outside \DIFdel
 my $MBOXINLINEMATH=0; # if set to 1 then surround marked-up inline maths expression with \mbox ( to get around compatibility
                       # problems between some maths packages and ulem package
 
-my $LISTENV='(?:itemize|description|enumerate)'; # list making environments - they will generally be kept
-my $ITEMCMD='item';
 
 # Markup strings
 # If at all possible, do not change these as parts of the program
@@ -674,7 +738,8 @@
 
 my ($i,$j,$l);
 my ($old,$new);
-my ($line);
+my ($line,$key);
+my (@dumlist);
 my ($newpreamble,$oldpreamble);
 my (@newpreamble, at oldpreamble, at diffpreamble, at diffbody);
 my ($latexdiffpreamble);
@@ -709,7 +774,7 @@
 
 my ($mboxcmd);
 
-my (@configlist, at labels,
+my (@configlist, at addtoconfiglist, at labels,
     @appendsafelist, at excludesafelist,
     @appendmboxsafelist, at excludemboxsafelist,
     @appendtextlist, at excludetextlist,
@@ -751,6 +816,7 @@
 	   'subtype|s=s' => \$subtype, 
 	   'floattype|f=s' => \$floattype, 
 	   'config|c=s' => \@configlist,
+	   'add-to-config|c=s' => \@addtoconfiglist,
 	   'preamble|p=s' => \$preamblefile,
 	   'encoding|e=s' => \$encoding,
 	   'label|L=s' => \@labels,
@@ -917,21 +983,64 @@
 }
 foreach $assign ( @config ) {
   $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
-  if ( $1 eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $2; }
-  elsif ( $1 eq "FLOATENV" ) { $FLOATENV = $2 ; }
-  elsif ( $1 eq "ITEMCMD" ) { $ITEMCMD = $2 ; }
-  elsif ( $1 eq "LISTENV" ) { $LISTENV = $2 ; }
-  elsif ( $1 eq "PICTUREENV" ) { $PICTUREENV = $2 ; }
-  elsif ( $1 eq "MATHENV" ) { $MATHENV = $2 ; }
-  elsif ( $1 eq "MATHREPL" ) { $MATHREPL = $2 ; }
-  elsif ( $1 eq "MATHARRENV" ) { $MATHARRENV = $2 ; }
-  elsif ( $1 eq "MATHARRREPL" ) { $MATHARRREPL = $2 ; }
-  elsif ( $1 eq "ARRENV" ) { $ARRENV = $2 ; }
-  elsif ( $1 eq "COUNTERCMD" ) { $COUNTERCMD = $2 ; }
-  elsif ( $1 eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $2 ; }
-  else { die "Unknown variable $1 in assignment.";}
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}=$2;
 }
 
+my @addtoconfig=();
+foreach $config ( @addtoconfiglist ) {
+  if (-f $config || lc $config eq '/dev/null' ) {
+    open(FILE,$config) or die ("Couldn't open addd-to-config file $config: $!");
+    while (<FILE>) {
+      chomp;
+      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+      push (@addtoconfig,$_);
+    }
+    close(FILE);
+  }
+  else {
+#    foreach ( split(",",$config) ) {
+#      push @addtoconfig,$_;
+#    }
+     push @addtoconfig,split(",",$config)
+  }
+}
+
+# initialise default lists from DATA
+# for those configuration variables, which have not been set explicitly, initiate from list in document
+foreach $key ( keys(%CONFIG) ) {
+  if (!defined $CONFIG{$key}) {
+    @dumlist=();
+    init_regex_arr_data(\@dumlist,"$key CONFIG");
+    $CONFIG{$key}=join(";", at dumlist)
+  }
+}
+
+
+foreach $assign ( @addtoconfig ) {
+  $assign=~ m/\s*(\w*)\s*=\s*(\S*)\s*$/ or die "Illegal assignment $assign in configuration list (must be variable=value)";  
+  exists $CONFIG{$1} or die "Unknown configuration variable $1.";
+  $CONFIG{$1}.=";$2";
+}
+
+# Map from hash to variables (we do this to have more concise code later, change from comma-separeted list)
+foreach  ( keys(%CONFIG) ) {
+  if ( $_ eq "MINWORDSBLOCK" ) { $MINWORDSBLOCK = $CONFIG{$_}; }
+  elsif ( $_ eq "FLOATENV" ) { $FLOATENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "ITEMCMD" ) { $ITEMCMD = $CONFIG{$_} ; }
+  elsif ( $_ eq "LISTENV" ) { $LISTENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "PICTUREENV" ) { $PICTUREENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHENV" ) { $MATHENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHREPL" ) { $MATHREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "MATHARRENV" ) { $MATHARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "MATHARRREPL" ) { $MATHARRREPL = $CONFIG{$_} ; }
+  elsif ( $_ eq "ARRENV" ) { $ARRENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "VERBATIMENV" ) { $VERBATIMENV = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "COUNTERCMD" ) { $COUNTERCMD = liststringtoregex($CONFIG{$_}) ; }
+  elsif ( $_ eq "SCALEDELGRAPHICS" ) { $SCALEDELGRAPHICS = $CONFIG{$_} ; }
+  else { die "Unknown configuration variable $_.";}
+}
+
 if ( $mathmarkup == COARSE || $mathmarkup == WHOLE ) {
   push(@MATHTEXTCMDLIST,qr/^MATHBLOCK(?:$MATHENV|$MATHARRENV|SQUAREBRACKET)$/);
 }
@@ -944,46 +1053,11 @@
 
 
 
-if ($showpreamble) {
-  print "\nPreamble commands:\n";
-  print $latexdiffpreamble ;
+if ($showconfig || $showtext || $showsafe || $showpreamble) {
+  show_configuration();
+  exit 0; 
 }
 
-if ($showsafe) {
-  print "\nCommands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
-  print_regex_arr(@SAFECMDLIST);
-  print "\nCommands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
-  print_regex_arr(@SAFECMDEXCL);
-}
-
-if ($showtext) {
-  print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
-  print_regex_arr(@TEXTCMDLIST);
-  print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
-  print_regex_arr(@CONTEXT1CMDLIST);
-  print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
-  print_regex_arr(@CONTEXT2CMDLIST);  
-  print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
-  print_regex_arr(@TEXTCMDEXCL);
-}
-
-
-if ($showconfig) {
-  print "Configuration variables:\n";
-  print "ARRENV=$ARRENV\n";
-  print "COUNTERCMD=$COUNTERCMD\n";
-  print "FLOATENV=$FLOATENV\n";
-  print "ITEMCMD=$ITEMCMD\n";
-  print "LISTENV=$LISTENV\n";
-  print "MATHARRENV=$MATHARRENV\n";
-  print "MATHARRREPL=$MATHARRREPL\n";
-  print "MATHENV=$MATHENV\n";
-  print "MATHREPL=$MATHREPL\n";
-  print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
-  print "PICTUREENV=$PICTUREENV\n";
-}
-if ($showconfig || $showtext || $showsafe || $showpreamble) {
-  exit 0; }
 if ( @ARGV != 2 ) { 
   print STDERR "2 and only 2 non-option arguments required.  Write latexdiff -h to get help\n";
   exit(2);
@@ -1033,7 +1107,7 @@
   my $word_ja='\p{Han}+|\p{InHiragana}+|\p{InKatakana}+';
   my $word='(?:' . $word_ja . '|(?:(?:[-\w\d*]|\\\\[\"\'\`~^][A-Za-z\*])(?!(?:' . $word_ja . ')))+)';
   my $cmdleftright='\\\\(?:left|right|[Bb]igg?[lrm]?|middle)\s*(?:[<>()\[\]|\.]|\\\\(?:[|{}]|\w+))';
-  my $cmdoptseq='\\\\[\w\d\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
+  my $cmdoptseq='\\\\[\w\d@\*]+'.$extraspace.'(?:(?:<'.$abrat0.'>|\['.$brat0.'\]|\{'. $pat_n . '\}|\(' . $coords .'\))'.$extraspace.')*';
   my $backslashnl='\\\\\n';
   my $oneletcmd='\\\\.\*?(?:\['.$brat0.'\]|\{'. $pat_n . '\})*';
   my $math='\$(?:[^$]|\\\$)*?\$|\\\\[(](?:.|\n)*?\\\\[)]';
@@ -1056,18 +1130,30 @@
 
 
 # set the labels to be included into the file
+# first find out which file name is longer for correct alignment
+my ($diff,$oldlabel_n_spaces,$newlabel_n_spaces);
+$oldlabel_n_spaces = 0;
+$newlabel_n_spaces = 0;
+$diff = length($newfile) - length($oldfile);
+if ($diff > 0) {
+  $oldlabel_n_spaces = $diff;
+}
+if ($diff < 0) {
+  $newlabel_n_spaces = abs($diff);
+}
+
 my ($oldtime,$newtime,$oldlabel,$newlabel);
 if (defined($labels[0])) {
   $oldlabel=$labels[0] ;
 } else {
   $oldtime=localtime((stat($oldfile))[9]); 
-  $oldlabel="$oldfile   " . " "x(length($newfile)-length($oldfile)) . $oldtime;
+  $oldlabel="$oldfile   " . " "x($oldlabel_n_spaces) . $oldtime;
 }
 if (defined($labels[1])) {
   $newlabel=$labels[1] ;
 } else {
   $newtime=localtime((stat($newfile))[9]);
-  $newlabel="$newfile   " . " "x(length($oldfile)-length($newfile)) . $newtime;
+  $newlabel="$newfile   " . " "x($newlabel_n_spaces) . $newtime;
 }
 
 $encoding=guess_encoding($newfile) unless defined($encoding);
@@ -1088,7 +1174,6 @@
 exetime(1);
 ($oldpreamble,$oldbody,$oldpost)=splitdoc($old,'\\\\begin\{document\}','\\\\end\{document\}');
 
-
 ($newpreamble,$newbody,$newpost)=splitdoc($new,'\\\\begin\{document\}','\\\\end\{document\}');
 
 
@@ -1167,22 +1252,28 @@
   # is not compatible with this.
   # (A more elegant solution would be to suppress the redefinitions of the \DIFaddbeginFL etc commands, but for this narrow use case
   #  I currently don't see this as an efficient use of time)
-  foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
-    @matches=( $latexdiffpreamble =~ m/command{\\DIFaddbeginFL}{($pat_n)}/sg ) ;
-    # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one elemen
-    if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
-      die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
-      exit 10;
-    }
+  ### The foreach loop does not make sense here. I don't know why I put this in -  (F Tilmann)
+  ###foreach my $cmd ( "DIFaddbegin","DIFaddend","DIFdelbegin","DIFdelend" ) {
+  @matches=( $latexdiffpreamble =~ m/command\{\\DIFaddbeginFL}\{($pat_n)}/sg ) ;
+  # we look at the last one of the list to take into account possible redefinition but almost always matches should have exactly one element
+  if ( $matches[$#matches] =~ m/\\DIFaddbegin/ ) {
+    die "Cannot combine graphics markup with float styles defining \\DIFaddbeginFL in terms of \\DIFaddbegin. Use --graphics-markup=none option or choose a different float style."; 
+    exit 10;
   }
+  ###}
   $latexdiffpreamble .= join "\n",("\\newcommand{\\DIFscaledelfig}{$SCALEDELGRAPHICS}",extrapream("HIGHLIGHTGRAPHICS"),"");
 
   # only change required for highlighting both is to declare \includegraphics safe, as preamble already contains commands for deleted environment
   if ( $graphicsmarkup == BOTH ) {
-    init_regex_arr_ext(\@SAFECMDLIST,'includegraphics');
+    init_regex_arr_list(\@SAFECMDLIST,'includegraphics');
   }
 }
 
+# If listings is being used and latexdiffpreamble uses color markup
+if (defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/)) {
+  $latexdiffpreamble .= join "\n",(extrapream("LISTINGS"),"");
+}
+
 # adding begin and end marker lines to preamble
 $latexdiffpreamble = "%DIF PREAMBLE EXTENSION ADDED BY LATEXDIFF\n" . $ latexdiffpreamble . "%DIF END PREAMBLE EXTENSION ADDED BY LATEXDIFF\n";
 
@@ -1244,11 +1335,11 @@
   # this is done to get around an incompatibility between the ulem and siunitx package
   print STDERR "siunitx package detected.\n" if $verbose ;
   my $mboxcmds='SI,ang,numlist,numrange,SIlist,SIrange';
-  init_regex_arr_ext(\@SAFECMDLIST,'num,si');
+  init_regex_arr_list(\@SAFECMDLIST,'num,si');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -1258,9 +1349,9 @@
   print STDERR "cleveref package detected.\n" if $verbose ;
   my $mboxcmds='[Cc]ref(?:range)?\*?,labelcref,(?:lc)?name[cC]refs?' ;
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
@@ -1269,17 +1360,17 @@
   # this is done to get around an incompatibility between ulem and glossaries package 
   print STDERR "glossaries package detected.\n" if $verbose ;
   my $mboxcmds='[gG][lL][sS](?:|pl|disp|link|first|firstplural|desc|user[iv][iv]?[iv]?),[aA][cC][rR](?:long|longpl|full|fullpl),[aA][cC][lfp]?[lfp]?';
-  init_regex_arr_ext(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
+  init_regex_arr_list(\@SAFECMDLIST,'[gG][lL][sS](?:(?:entry)?(?:text|plural|name|symbol)|displaynumberlist|entryfirst|entryfirstplural|entrydesc|entrydescplural|entrysymbolplural|entryuser[iv][iv]?[iv]?|entrynumberlist|entrydisplaynumberlist|entrylong|entrylongpl|entryshort|entryshortpl|entryfull|entryfullpl),[gG]lossentry(?:name|desc|symbol),[aA][cC][rR](?:short|shortpl),[aA]csp?');
   if ( $enablecitmark || ( $ulem  && ! $disablecitmark )) {
-    init_regex_arr_ext(\@MBOXCMDLIST,$mboxcmds);
+    init_regex_arr_list(\@MBOXCMDLIST,$mboxcmds);
   } else {
-    init_regex_arr_ext(\@SAFECMDLIST,$mboxcmds);
+    init_regex_arr_list(\@SAFECMDLIST,$mboxcmds);
   }
 }
 
 if (defined $packages{"chemformula"} or defined $packages{"chemmacros"} ) {
   print STDERR "chemformula package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ch');
+  init_regex_arr_list(\@SAFECMDLIST,'ch');
   push(@UNSAFEMATHCMD,'ch');
   # The next command would be needed to allow highlighting the interior of \ch commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -1288,7 +1379,7 @@
 
 if (defined $packages{"mhchem"} ) {
   print STDERR "mhchem package detected.\n" if $verbose ;
-  init_regex_arr_ext(\@SAFECMDLIST,'ce');
+  init_regex_arr_list(\@SAFECMDLIST,'ce');
   push(@UNSAFEMATHCMD,'ce','cee');
   # The next command would be needed to allow highlighting the interior of \cee commands in math environments
   # but the redefinitions in chemformula are too deep to make this viable
@@ -1330,7 +1421,7 @@
 
 # add commands in MBOXCMDLIST to SAFECMDLIST
 foreach $mboxcmd ( @MBOXCMDLIST ) {
-  init_regex_arr_ext(\@SAFECMDLIST, $mboxcmd);
+  init_regex_arr_list(\@SAFECMDLIST, $mboxcmd);
 }
 
 # check if \label is in SAFECMDLIST, and if yes replace "label" in $LABELCMD by something that never matches (we hope!)
@@ -1376,7 +1467,71 @@
 print STDERR "(",exetime()," s)\n","Done.\n" if $verbose;
 
 
+# liststringtoregex(liststring)
+# expands string with semi-colon separated list into a regular expression corresponding 
+# matching any of the elements
+sub liststringtoregex {
+  my ($liststring)=@_;
+  my @elements=grep /\S/,split(";",$liststring);
+  if ( @elements) {
+    return('(?:(?:' . join(')|(?:', at elements) .'))');
+  } else {
+    return "";
+  }
+}
 
+# show_configuration
+# note that this is not encapsulated but uses variables from the main program 
+# It is provided for convenience because in the future it is planned to allow output
+# to be modified based on what packages are read etc - this works only if the input files are actually red
+# whether or not additional files are provided
+sub show_configuration {
+  if ($showpreamble) {
+    print "\nPreamble commands:\n";
+    print $latexdiffpreamble ;
+  }
+
+  if ($showsafe) {
+    print "\nsafecmd: Commands safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE (unless excluded):\n";
+    print_regex_arr(@SAFECMDLIST);
+    print "\nsafecmd-exlude: Commands not safe within scope of $ADDOPEN $ADDCLOSE and $DELOPEN $DELCLOSE :\n";
+    print_regex_arr(@SAFECMDEXCL);
+    print "\nmboxsafecmd:  Commands safe only if they are surrounded by \\mbox command:\n";
+    print_regex_arr(@MBOXCMDLIST);
+    print "\nnmboxsafecmd: Commands not safe:\n";
+    print_regex_arr(@MBOXCMDEXCL);
+  }
+
+  if ($showtext) {
+    print "\nCommands with last argument textual (unless excluded) and safe in every context:\n";
+    print_regex_arr(@TEXTCMDLIST);
+    print "\nContext1 commands (last argument textual, command will be disabled in deleted passages, last argument will be shown as plain text):\n";
+    print_regex_arr(@CONTEXT1CMDLIST);
+    print "\nContext2 commands (last argument textual, command and its argument will be disabled in deleted passages):\n";
+    print_regex_arr(@CONTEXT2CMDLIST);  
+    print "\nExclude list of Commands with last argument not textual (overrides patterns above):\n";
+    print_regex_arr(@TEXTCMDEXCL);
+  }
+
+
+  if ($showconfig) {
+    print "Configuration variables:\n";
+    print "ARRENV=$ARRENV\n";
+    print "COUNTERCMD=$COUNTERCMD\n";
+    print "FLOATENV=$FLOATENV\n";
+    print "ITEMCMD=$ITEMCMD\n";
+    print "LISTENV=$LISTENV\n";
+    print "MATHARRENV=$MATHARRENV\n";
+    print "MATHARRREPL=$MATHARRREPL\n";
+    print "MATHENV=$MATHENV\n";
+    print "MATHREPL=$MATHREPL\n";
+    print "MINWORDSBLOCK=$MINWORDSBLOCK\n";
+    print "PICTUREENV=$PICTUREENV\n";
+  }
+}
+
+
+
 ## guess_encoding(filename)
 ## reads the first 20 lines of filename and looks for call of inputenc package
 ## if found, return the option of this package (encoding), otherwise return utf8
@@ -1480,46 +1635,44 @@
 # and adds the created commands which are clearly safe to @SAFECMDLIST
 sub add_safe_commands {
   my ($preamble)=@_;
-  my $added_command = 1;
 
+  # get rid of comments
   $preamble=~s/(?<!\\)%.*$//mg ;
 
-  while ($added_command) { 
-    $added_command  = 0;
-      # get rid of comments
-    my $to_test = "";
-      # test for \DeclareMathOperator{\foo}{myoperator}
-    while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
-      $to_test=$1;
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
-      }
+  my $to_test = "";
+  # test for \DeclareMathOperator{\foo}{myoperator}
+  while ( $preamble =~ m/\DeclareMathOperator\s*\*?\{\\(\w*?)\}/osg) {
+    $to_test=$1;
+    if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+      # one should add $to_test to the list of safe commands.
+      init_regex_arr_list(\@SAFECMDLIST, $to_test);
+      print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
     }
+  }
 
-    while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
-      my $maybe_to_test  = $1;
-      my $should_be_safe = $2;
-      my $success = 0;
-      # test if all latex commands inside it are safe
-      $success = 1;
-      if ($should_be_safe =~ m/\\\\/) {
-	$success = 0;
-      } else {
-	while ($should_be_safe =~ m/\\(\w+)/g) {
-	  $success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
-	}
+  while ( $preamble =~ m/\\(?:new|renew|provide)command\s*{\\(\w*)\}(?:|\[\d*\])\s*\{(${pat_n})\}/osg ) {
+    my $maybe_to_test  = $1;
+    my $should_be_safe = $2;
+    print STDERR "DEBUG Checking new command: maybe_to_test, should_be_safe: $1 $2\n" if $debug;
+    my $success = 0;
+    # test if all latex commands inside it are safe
+    $success = 1;
+    if ($should_be_safe =~ m/\\\\/) {
+      $success = 0;
+    } else {
+      while ($should_be_safe =~ m/\\(\w+)/g) {
+	###	  print STDERR "DEBUG: Testing command $1 " if $debug;
+	$success = 0 unless iscmd($1,\@SAFECMDLIST,\@SAFECMDEXCL); ### or $1 eq "";
+	###        print STDERR " success=$success\n" if $debug;
       }
-      ###      }
-      $to_test = $maybe_to_test if $success;
-      
-      if ($to_test ne "" and not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, []))  {
-#        # one should add $to_test to the list of safe commands.
-        init_regex_arr_ext(\@SAFECMDLIST, $to_test);
-        $added_command = 1;
-        print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
+    }
+    ###      }
+    if ($success) {
+      $to_test = $maybe_to_test;
+      if (  not iscmd($to_test,\@SAFECMDLIST,\@SAFECMDEXCL) and not iscmd($to_test, \@SAFECMDEXCL, [])) {
+	#        # one should add $to_test to the list of safe commands.
+	init_regex_arr_list(\@SAFECMDLIST, $to_test);
+	print STDERR "Adding $to_test to the list of safe commands\n" if $verbose;
       }
     }
   }
@@ -1526,6 +1679,19 @@
 }
 
 
+# helper function for flatten
+# remove \endinput at beginning of line and everything
+# following it, # if \endinput is not at the beginning of
+# the line, nothing will be removed. It is assumed that
+# this case is most common when \endinput is part of a
+# conditional clause.  The file will only be processed
+# correctly if the conditional is always false,
+# i.e. \endinput # not actually reached
+sub remove_endinput {
+  # s/// operates on default input
+  $_[0] =~ s/^\\endinput.*\Z//ms ;
+  return($_[0]);
+}
 
 # flatten($text,$preamble,$filename,$encoding)
 # expands \input and \include commands within text
@@ -1553,7 +1719,7 @@
   print STDERR "DEBUG: includeonly $includeonly\n" if $debug;
 
   # recursively replace \\input and \\include files
-  1 while $text=~s/(^(?:[^%\n]|\\%)*)\\input{(.*?)}|\\include{(${includeonly}(?:\.tex)?)}/{ 
+  1 while $text=~s/(^(?:[^%\n]|\\%)*)(?:\\input\{(.*?)\}|\\include\{(${includeonly}(?:\.tex)?)\})/{ 
 	    $begline=(defined($1)? $1 : "") ;
 	    $fname = $2 if defined($2) ;
 	    $fname = $3 if defined($3) ;
@@ -1565,6 +1731,7 @@
             # content of file becomes replacement value (use recursion), add \newpage if the command was include
             ###$replacement=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
 	    $replacement=flatten(read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding), $preamble,$filename,$encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
+	    $replacement = remove_endinput($replacement); 
 	    # \include always starts a new page; use explicit \newpage command to simulate this
 	    $newpage=(defined($3)? " \\newpage " : "") ;
 	    "$begline$newpage$replacement$newpage";
@@ -1593,6 +1760,7 @@
            $subfile=read_file_with_encoding(File::Spec->catfile($dirname,$fname), $encoding) or die "Couldn't find file ",File::Spec->catfile($dirname,$fname),": $!";
            ($subpreamble,$subbody,$subpost)=splitdoc($subfile,'\\\\begin\{document\}','\\\\end\{document\}');
 	   $replacement=flatten($subbody, $preamble,$filename,$encoding);
+	   $replacement = remove_endinput($replacement); 
 	   "$begline$replacement";
   }/exgm;
 
@@ -2236,7 +2404,7 @@
       if ( $word =~ /^\\(?!\()(\\|[`'^"~=.]|[\w*@]+)(.*?)(\s*)$/s &&  iscmd($1,\@MBOXCMDLIST,\@MBOXCMDEXCL)) {
 	# $word is a safe command in MBOXCMDLIST
 	###print STDERR "DEBUG Mboxsafecmd detected:$word:\n" if $debug ;
-	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}$AUXCMD\n" );
+	push(@$retval,"\\mbox{$AUXCMD\n\\" . $1 . $2 . $3 ."}\\hspace{0pt}$AUXCMD\n" );
       } else {
 	# $word is a normal word or a safe command (not in MBOXCMDLIST)
 	push (@$retval,$word);
@@ -2253,47 +2421,49 @@
 
 #used in preprocess
 sub take_comments_and_enter_from_frac() {
-      ###*************take the \n and % between frac and {}***********
-      ###notice all of the substitution are made none global
-      while( m/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/s ) {
-	### if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
-	if( $2 !~ m/\\end{$1}/s ) {
-	  ### take out % and \n from the next match only (none global)
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
+      #*************take the \n and % between frac and {}***********
+      #notice all of the substitution are made none global
+      while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/s ) {
+	# if there isn't any % or \n in the pattern $2 then there should be an \\end{...} in $2
+	### print STDERR "Match the following in take_comments and_enter_from_frac(1):\n****$&****\n" if $debug;
+	if( $2 !~ m/\\end\{$1}/s ) {
+	  # take out % and \n from the next match only (none global)
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\frac(([\s]*%[^\n]*?)*[\r\n|\r|\n])+\{(.*?)\\end\{\1}/\\begin{$1}$2\\frac{$5\\end{$1}/s;
 	}
 	else{
-	  ###there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
-		s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+	  #there are no more % and \n in $2, we want to find the next one so we clear the begin-end from the pattern
+		s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
 	}
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       ###*************take the \n and % between frac and {}***********
       
       ###**********take the \n and % between {} and {} of the frac***************
       while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/s ) {
-	### if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	# if there isn't any more //frac before the first //end in the pattern $2 then there should be an \\end{...} in $2
+	###print STDERR "Match the following in take_comments and_enter_from_frac(2):\n****$&****\n" if $debug;
         if( $2 !~ m/\\end\{$1\}/s ) {
-          ### from now on CURRFRAC is the frac we are looking at
+          # from now on CURRFRAC is the frac we are looking at
           s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\frac\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\\end\{$1\}/s;
           while( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{(.*?)\\end\{\1\}/s ) {
-             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end{\1}/s ) {
+             if( m/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1}/s ) {
                    s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)CURRFRAC\{($pat_n)\}([\s]*(%[^\n]*?)*[\r\n|\r|\n])+[\s]*\{(.*?)\\end\{\1\}/\\begin\{$1\}$2CURRFRAC\{$3\}\{$6\\end\{$1\}/s;
                } 
               else { # there is no comment or \n between the two brackets {}{}
-              ### change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
-                   s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
+              # change CURRFRAC to FRACSTART so we can change them all back to //frac{ when we finish
+                   s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)CURRFRAC\{(.*?)\\end\{\1}/\\begin{$1}$2FRACSTART\{$3\\end{$1}/s;
               }
             }
         }
         else{
           ###there are no more frac in $2, we want to find the next one so we clear the begin-end from the pattern
-          s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/MATHBLOCK$1$2MATHBLOCKEND/s;
+          s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end\{\1}/MATHBLOCK$1\{$2\}MATHBLOCKEND/s;
         }
         
       }
       ###cleaning up
-      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)(.*?)MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
+      while( s/MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{(.*?)\}MATHBLOCKEND/\\begin{$1}$2\\end{$1}/s ){}
       s/FRACSTART/\\frac/g;
       ###***************take the \n and % between {} and {} of the frac*********************
 }
@@ -2306,8 +2476,8 @@
 # 2. mark all first empty line (in block of several) with \PAR tokens
 # 3. Convert all '\%' into '\PERCENTAGE ' and all '\$' into \DOLLAR to make parsing regular expressions easier
 # 4. Convert all \verb|some verbatim text| commands (where | can be an arbitrary character)
-#    into \verb{hash}
-# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}
+#    into \verb{hash}  (also lstinline)
+# 5. Convert \begin{verbatim} some verbatim text \end{verbatim} into \verbatim{hash}  (not only verbatim, all patterns matching VERBATIMENV)
 # 6. Convert _n into \SUBSCRIPTNB{n} and _{nnn} into \SUBSCRIPT{nn}
 # 7. Convert ^n into \SUPERSCRIPTNB{n} and ^{nnn} into \SUPERSCRIPT{nn}
 # 8. a. Convert $$ $$ into \begin{DOLLARDOLLAR} \end{DOLLARDOLLAR}
@@ -2323,6 +2493,17 @@
 # Returns: leading white space removed in step 1
 sub preprocess {
   for (@_) { 
+
+
+    # change in \verb and similar commands - note that I introduce an extra space here so that the 
+    #       already hashed variants do not trigger again
+    # transform \lstinline{...}
+#    s/\\lstinline(\[$brat0\])?(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+#    s/\\lstinline(\[$brat0\])?((\S).*?\2)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)(\{(?:.*?)\})/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\lstinline((?:\[$brat0\])?)((\S).*?\3)/"\\DIFlstinline". $1 ."{". tohash(\%verbhash,"$2") ."}"/esg;
+    s/\\(verb\*?|lstinline)(\S)(.*?)\2/"\\DIF${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
+
     #    Change \{ to \QLEFTBRACE, \} to \QRIGHTBRACE, and \& to \AMPERSAND
     s/(?<!\\)\\\{/\\QLEFTBRACE /sg;
     s/(?<!\\)\\\}/\\QRIGHTBRACE /sg;
@@ -2330,12 +2511,10 @@
 # replace {,} in comments with \\CLEFTBRACE,\\CRIGHTBRACE
     1 while s/((?<!\\)%.*)\{(.*)$/$1\\CLEFTBRACE $2/mg ;
     1 while s/((?<!\\)%.*)\}(.*)$/$1\\CRIGHTBRACE $2/mg ;
-
     s/\n(\s*?)\n((?:\s*\n)*)/\n$1\\PAR\n$2/g ;
     s/(?<!\\)\\%/\\PERCENTAGE /g ;  # (?<! is negative lookbehind assertion to prevent \\% from being converted
     s/(?<!\\)\\\$/\\DOLLAR /g ;  # (?<! is negative lookbehind assertion to prevent \\$ from being converted
-    s/(\\verb\*?)(\S)(.*?)\2/"${1}{". tohash(\%verbhash,"${2}${3}${2}") ."}"/esg;
-    s/\\begin\{(verbatim\*?)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
+    s/\\begin\{($VERBATIMENV)\}(.*?)\\end\{\1\}/"\\${1}{". tohash(\%verbhash,"${2}") . "}"/esg;
     # Convert _n or _\cmd into \SUBSCRIPTNB{n} or \SUBSCRIPTNB{\cmd} and _{nnn} into \SUBSCRIPT{nn}
     1 while s/(?<!\\)_(\s*([^{\\\s]|\\\w+))/\\SUBSCRIPTNB{$1}/g ;
     1 while s/(?<!\\)_(\s*{($pat_n)})/\\SUBSCRIPT$1/g ;
@@ -2351,17 +2530,20 @@
     s/(?<!\\)\\\[/\\begin{SQUAREBRACKET}/sg;
     s/\\\]/\\end{SQUAREBRACKET}/sg;
     # Convert all picture environmentent (\begin{PICTUREENV} .. \end{PICTUREENV} \PICTUREBLOCKenv
-    s/\\begin{($PICTUREENV)}(.*?)\\end{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
+    s/\\begin\{($PICTUREENV)}(.*?)\\end\{\1}/\\PICTUREBLOCK$1\{$2\}/sg;
     #    For --block-math-markup option -convert all \begin{MATH} .. \end{MATH}
     #    into \MATHBLOCKMATH{...} commands, where MATH is any valid math environment
     #    Also convert all array environments into ARRAYBLOCK environments
+
     if ( $mathmarkup != FINE ) {
-      s/\\begin{($ARRENV)}(.*?)\\end{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
+      s/\\begin\{($ARRENV)}(.*?)\\end\{\1}/\\ARRAYBLOCK$1\{$2\}/sg;
       
       take_comments_and_enter_from_frac();
       
-      s/\\begin{($MATHENV|$MATHARRENV|SQUAREBRACKET)}(.*?)\\end{\1}/\\MATHBLOCK$1\{$2\}/sg;
+
+      s/\\begin\{($MATHENV|$MATHARRENV|SQUAREBRACKET)\}(.*?)\\end\{\1\}/\\MATHBLOCK$1\{$2\}/sg;
     }
+
     # add final token " STOP"
     $_ .= " STOP"
   }
@@ -2470,6 +2652,7 @@
     # Replace \RIGHTBRACE by }    
     s/\\RIGHTBRACE/}/g;
 
+
     # Check all deleted blocks: where a deleted block contains a matching \begin and
     #    \end environment (these will be disabled by a %DIFDELCMD statements), enable
     #    these commands again (such that for example displayed math in a deleted equation
@@ -2561,15 +2744,17 @@
 #    bb. disable active labels within deleted blocks (as these are not safe commands, this should normally only
 #        happen within deleted maths blocks
       $delblock=~ s/(?<!$DELCMDOPEN)(\\$LABELCMD(?:${extraspace})\{(?:[^{}])*\}[\t ]*)\n?/${DELCMDOPEN}$1${DELCMDCLOSE}/smg ;
+      # previous line causes trouble as by issue #90 I might need to modify this
 
 
 #     c. If in-line math mode contains array environment, enclose the whole environment in \mbox'es
       while ( $delblock =~ m/($math)(\s*)/sg ) {
+#	      print STDERR "DEBUG Delblock Match math $& at ",pos,"\n";
 	$cnt2=0;
 	$len2=length($&);
 	$begin2=pos($delblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/ );
 	substr($delblock,$begin2,$len2)=$mathblock;
 	pos($delblock) = $begin2 + length($mathblock);
       }
@@ -2578,11 +2763,22 @@
 	# note additional \newline after command is omitted from output if right at the end of deleted block (otherwise a spurious empty line is generated)
 	$delblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+      ###  #     change included verbatim environments
+      ###  $delblock =~ s/\\DIFverb\{/\\DIFDIFdelverb\{/g;
+      ###  $delblock =~ s/\\DIFlstinline/\\DIFDIFdellstinline/g;
+      ###}
+      # Mark deleted verbose commands
+      $delblock =~ s/(${DELCMDOPEN}\\DIF((?:verb\*?|lstinline(?:\[$brat0\])?)\{([-\d]*?)\}\s*).*)$/%\n\\DIFDIFdel$2${AUXCMD}\n$1/gm;
 
 #     splice in modified delblock
       substr($_,$begin,$len)=$delblock;
       pos = $begin + length($delblock);
     }
+
+    ### print STDERR "<<<$_>>>\n" if $debug;
+
+
     # make the array modification in added blocks
     while ( m/\\DIFaddbegin.*?\\DIFaddend/sg ) {
       $cnt=0;
@@ -2594,7 +2790,7 @@
 	$len2=length($&);
 	$begin2=pos($addblock) - $len2;
 	$mathblock="%\n\\mbox{$AUXCMD\n$1\n}$AUXCMD\n";
-        next unless $mathblock =~ m/\{$ARRENV\}/ ;
+        next unless ( $mathblock =~ /ARRAYBLOCK/ or $mathblock =~ m/\{$ARRENV\}/) ;
 	substr($addblock,$begin2,$len2)=$mathblock;
 	pos($addblock) = $begin2 + length($mathblock);
       }
@@ -2603,6 +2799,11 @@
 	##$addblock=~s/($math)/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
 	$addblock=~s/($math)(?:[\s\n]*)?/\\mbox{$AUXCMD\n$1\n}$AUXCMD\n/sg;
       }
+      ###if ( defined($packages{"listings"} and $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/))   { 
+	# mark added verbatim commands
+      $addblock =~ s/\\DIFverb/\\DIFDIFaddverb/g;
+      $addblock =~ s/\\DIFlstinline/\\DIFDIFaddlstinline/g;
+      ###}
 #     splice in modified addblock
       substr($_,$begin,$len)=$addblock;
       pos = $begin + length($addblock);
@@ -2646,8 +2847,8 @@
     # environments) are between the \begin{$MATHENV} and \end{MATHMODE} commands. This is necessary as the minimal matching 
     # is not globally minimal but only 'locally' (matching is beginning from the left side of the string)
     if ( $mathmarkup == FINE ) {
-      1 while s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin{MATHMODE})))*?)\\end{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
-      1 while s/\\begin{MATHMODE}((?:.(?!\\end{MATHMODE}))*?)\\end{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
+      1 while s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}((?:.(?!(?:\\end\{(?:(?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}|\\begin\{MATHMODE})))*?)\\end\{MATHMODE}/\\begin{$1}$2\\end{$1}/s;
+      1 while s/\\begin\{MATHMODE}((?:.(?!\\end\{MATHMODE}))*?)\\end\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}/\\begin{$2}$1\\end{$2}/s;
       # convert remaining \begin{MATHMODE} \end{MATHMODE} (and not containing & or \\ )into MATHREPL environments
       s/\\begin\{MATHMODE\}((?:(.(?!(?<!\\)\&|\\\\))*)?)\\end\{MATHMODE\}/\\begin{$MATHREPL}$1\\end{$MATHREPL}/sg;
       # others into MATHARRREPL
@@ -2654,7 +2855,7 @@
       s/\\begin\{MATHMODE\}(.*?)\\end\{MATHMODE\}/\\begin{$MATHARRREPL}$1\\end{$MATHARRREPL}/sg;
 
       # now look for AUXCMD math-mode pairs which have only comments (or empty lines between them), and remove the added commands
-      s/\\begin{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
+      s/\\begin\{((?:$MATHENV)|(?:$MATHARRENV)|SQUAREBRACKET)}$AUXCMD\n((?:\s*%.[^\n]*\n)*)\\end{\1}$AUXCMD\n/$2/sg;       
     } else {
       #   math modes OFF,WHOLE,COARSE: Convert \MATHBLOCKmath{..} commands back to environments
       s/\\MATHBLOCK($MATHENV|$MATHARRENV|SQUAREBRACKET)\{($pat_n)\}/\\begin{$1}$2\\end{$1}/sg;
@@ -2702,13 +2903,8 @@
     s/^\Q${DELCMDOPEN}\E\n//msg;
 
     # Expand hashes of verb and verbatim environments (note negative look behind assertion to not leak out of DIFDELCMD comments
-    s/(\\verb\*?)\{([-\d]*?)\}/"${1}". fromhash(\%verbhash,$2)/esg;
-    s/${DELCMDOPEN}\\(verbatim\*?)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
-    s/\\(verbatim\*?)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
-    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
-    s/\\PERCENTAGE ?/\\%/g;
-    # Convert '\DOLLAR ' back into '\$'			       
-    s/\\DOLLAR /\\\$/g;
+    s/${DELCMDOPEN}\\($VERBATIMENV)\{([-\d]*?)\}/"${DELCMDOPEN}\\begin{${1}}".fromhash(\%verbhash,$2,$DELCMDOPEN)."${DELCMDOPEN}\\end{${1}}"/esg;
+    s/\\($VERBATIMENV)\{([-\d]*?)\}/"\\begin{${1}}".fromhash(\%verbhash,$2)."\\end{${1}}"/esg;
     # remove all \PAR tokens (taking care to properly keep commented out PAR's
     # from introducing uncommented newlines - next line)
     s/(%DIF < )([^\n]*?)\\PAR\n/$1$2\n$1\n/sg;
@@ -2724,11 +2920,17 @@
       print STDERR "endfloat package detected.\n" if $verbose ;
       # eliminate whitespace before and after
       s/^(\s*)(\\(?:begin|end)\{(?:figure|table)\})(\s*)$/$2/mg;
-      # split lines with remaining characters before float enviroment conmmand
-      s/^(.+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
-      # split lines with remaining characters after float enviroment conmmand
-      s/(\\(?:begin|end)\{(?:figure|table)\})(.+)$/$1\n$2/mg;
+      # split lines with remaining characters before float environment conmmand
+      s/^([^%]+)(\\(?:begin|end)\{(?:figure|table)\})/$1\n$2/mg;
+      # split lines with remaining characters after float environment conmmand
+      s/^((?:[^%]+)\\(?:begin|end)\{(?:figure|table)\}(?:\[[a-zA-Z]+\])?)(.+)((?:%.*)?)$/$1\n$2$3/mg;
     }
+
+    # Convert '\PERCENTAGE ' back into '\%' (the final question mark catches a special situation where due to a latter pre-processing step the ' ' becomes separated	       
+    s/\\PERCENTAGE ?/\\%/g;
+    # Convert '\DOLLAR ' back into '\$'			       
+    s/\\DOLLAR /\\\$/g;
+
     # undo renaming of the \begin and \end,{,}  and dollars in comments 
     1 while s/(%.*)DOLLARDIF/$1\$/mg ;
 #   Convert \begin{SQUAREBRACKET} \end{SQUAREBRACKET} into \[ \]
@@ -2754,6 +2956,21 @@
     s/\\QLEFTBRACE /\\\{/sg;
     s/\\QRIGHTBRACE /\\\}/sg;
     s/\\AMPERSAND /\\&/sg;
+    # Highligh added inline verbatim commands if possible
+    if ( $latexdiffpreamble =~ /\\RequirePackage(?:\[$brat0\])?\{color\}/ )   { 
+      # wrap added verb commands with color commands
+      s/\\DIFDIFadd((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*)/\{\\color{blue}$AUXCMD\n\\DIF$1%\n\}$AUXCMD\n/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD)/\{\\color{red}${AUXCMD}\n\\DIF$1\n\}${AUXCMD}/sg;
+    } else {
+      # currently if colour markup is not used just remove the added mark
+      s/\\DIFDIFadd(verb\*?|lstinline)/\\DIF$1/sg;
+      s/\\DIFDIFdel((?:verb\*?|lstinline(?:\[$brat0\])?)\{[-\d]*?\}[\s\n]*$AUXCMD\n)//sg;
+    }
+    # expand \verb and friends inline arguments
+    s/\\DIF((?:DIFadd|DIFdel)?(?:verb\*?|lstinline(?:\[$brat0\])?))\{([-\d]*?)\}/"\\${1}". fromhash(\%verbhash,$2)/esg;
+    # add basicstyle color{blue} to added lstinline commands
+    # finally add the comment to the ones not having an optional argument before
+    ###s/\\DIFaddlstinline(?!\[)/\\lstinline\n[basicstyle=\\color{blue}]$AUXCMD\n/g;
 
   return;
   }
@@ -2764,7 +2981,7 @@
 sub restore_item_commands {
   my ($string)=@_ ;
   my ($itemarg, at itemargs);
-  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*?(?:\n|$DELCMDCLOSE))/
+  $string =~ s/(\%DIFDELCMD < \s*(\\$ITEMCMD)((?:<$abrat0>)?)((?:\[$brat0\])?)\s*((?:${cmdoptseq}\s*?)*)(?:\n|$DELCMDCLOSE))/
      # if \item has an []argument, then mark up the argument as deleted)
      if (length($4)>0) {
        # use substr to exclude square brackets at end points
@@ -2773,7 +2990,8 @@
      } else {
        $itemarg="";
      }
-     "$1$2$3$itemarg$AUXCMD\n"/sge;
+     "$1$2$3$itemarg$AUXCMD\n";  ###.((length($5)>0) ? "%DIFDELCMD $5 $DELCMDCLOSE\n" : "")
+     /sge;
   return($string);
 }
 
@@ -2956,16 +3174,32 @@
 # init_regex_arr_data(\@array,"TOKEN INIT")
 # scans DATA file handel for line "%% TOKEN INIT" line
 # then appends each line not beginning with % into array (as a quoted regex)
+# This is used for command lists and configuration variables, but the processing is slightly 
+# different: 
+# For lists, the regular expression is extended to include beginning (^) and end ($) markers, to require full-string matching
+# For configuration variables (and all others), simply an unadorned list is copied
 sub init_regex_arr_data {
   my ($arr,$token)=@_;
-  my ($copy);
+  my $copy=0;
+  my ($mode);
+  if ($token =~ m/COMMANDS/ ) { 
+    $mode=0;
+  } else {
+    $mode=1;
+  }
+
   while (<DATA>) {
     if ( m/^%%BEGIN $token\s*$/ ) {
-      $copy=1; }
-    elsif ( m/^%%END $token\s*/ )  {
+      $copy=1; 
+      next;
+    } elsif ( m/^%%END $token\s*$/ )  {
       last; }
     chomp;
-    push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    if ( $mode==0 ) {
+      push (@$arr,qr/^$_$/) if ( $copy && !/^%/ ) ;
+    } elsif ($mode==1) {
+      push (@$arr,"$_") if ( $copy && !/^%/ ) ;
+    }
   }
   seek DATA,0,0;    # rewind DATA handle to file begin
 }
@@ -2978,25 +3212,42 @@
 # Otherwise treat arg as a comma separated list of regular expressions
 sub init_regex_arr_ext {
   my ($arr,$arg)=@_;
-  my $regex;
-  if ( -f $ arg ) {
-    open(FILE,"$arg") or die ("Couldn't open $arg: $!");
-    while (<FILE>) {
-      chomp;
-      next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
-      push (@$arr,qr/^$_$/);
-    }
-    close(FILE);
+  if ( -f $arg ) {
+    init_regex_arr_file($arr,$arg);
+  } else {
+    init_regex_arr_list($arr,$arg);
   }
-  else {
-    # assume it is a comma-separated list of reg-ex
-    foreach $regex (split(qr/(?<!\\),/,$arg)) {
-      $regex =~ s/\\,/,/g;
-      push (@$arr,qr/^$regex$/);
-    }
+}
+
+# init_regex_arr_file(\@array,$fname)
+# appends array with regular expressions.
+# Read in list of regular expressions from $fname
+# (one expression per line)
+sub init_regex_arr_file {
+  my ($arr,$fname)=@_;
+  open(FILE,"$fname") or die ("Couldn't open $fname: $!");
+  while (<FILE>) {
+    chomp;
+    next if /^\s*#/ || /^\s*%/ || /^\s*$/ ;
+    push (@$arr,qr/^$_$/);
   }
+  close(FILE);
 }
 
+# init_regex_arr_list(\@array,$arg)
+# appends array with regular expressions.
+# read from comma separated list of regular expressions ($arg)
+sub init_regex_arr_list {
+  my ($arr,$arg)=@_;
+  my $regex;
+  ###    print STDERR "DEBUG init_regex_arr_list arg >$arg<\n" if $debug;
+  foreach $regex (split(qr/(?<!\\),/,$arg)) {
+    $regex =~ s/\\,/,/g;
+    push (@$arr,qr/^$regex$/);
+  }
+}
+
+
 #exetime() returns time since last execution of this command
 #exetime(1) resets this time
 my $lasttime=-1;   # global variable for persistence
@@ -3138,6 +3389,14 @@
                           SCALEDELGRAPHICS (Float)
                        This option can be repeated.
 
+--add-to-config  varenv1=pattern1,varenv2=pattern2
+                       For configuration variables containing a regular expression (essentially those ending
+                       in ENV, and COUNTERCMD) this provides an alternative way to modify the configuration 
+                       variables. Instead of setting the complete pattern, with this option it is possible to add an
+                       alternative pattern. varenv must be one of the variables listed above that take a regular
+                       expression as argument, and pattern is any regular expression (which might need to be 
+                       protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+                       to separate them, e.g. --add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote"
 
 --packages=pkg1,pkg2,..
                        Tell latexdiff that .tex file is processed with the packages in list 
@@ -3556,11 +3815,12 @@
 =item B<--replace-context1cmd=replace-file> 
 
 =item B<--append-context1cmd=append-file> or
+
 =item B<--append-context1cmd="cmd1,cmd2,...">
 
 Replace or append to the list of regex matching commands
 whose last argument is text but which require a particular
-context to work, e.g. \caption will only work within a figure
+context to work, e.g. C<\caption> will only work within a figure
 or table.  These commands behave like text commands, except when 
 they occur in a deleted section, when they are disabled, but their
 argument is shown as deleted text.
@@ -3568,6 +3828,7 @@
 =item B<--replace-context2cmd=replace-file> 
 
 =item B<--append-context2cmd=append-file> or
+
 =item B<--append-context2cmd="cmd1,cmd2,...">
 
 As corresponding commands for context1.  The only difference is that
@@ -3580,7 +3841,7 @@
 =item B<--append-mboxsafecmd=append-file> or B<--append-mboxsafecmd="cmd1,cmd2,...">
 
 Define safe commands, which additionally need to be protected by encapsulating
-in an \\mbox{..}. This is sometimes needed to get around incompatibilities 
+in an C<\mbox{..}>. This is sometimes needed to get around incompatibilities 
 between external packages and the ulem package, which is  used for highlighting
 in the default style UNDERLINE as well as CULINECHBAR CFONTSTRIKE
 
@@ -3620,6 +3881,16 @@
 
 C<SCALEDELGRAPHICS> (Float)
 
+=item B<--add-to-config varenv1=pattern1,varenv2=pattern2,...>
+
+For configuration variables, which are a regular expression (essentially those ending
+in ENV, and COUNTERCMD, see list above) this provides an alternative way to modify the configuration 
+variables. Instead of setting the complete pattern, with this option it is possible to add an
+alternative pattern. C<varenv> must be one of the variables listed above that take a regular
+expression as argument, and pattern is any regular expression (which might need to be 
+protected from the shell by quotation). Several patterns can be added at once by using semi-colons
+to separate them, e.g. C<--add-to-config "LISTENV=myitemize;myenumerate,COUNTERCMD=endnote">
+
 =item B<--show-safecmd>
 
 Print list of RegEx matching and excluding safe commands.
@@ -3680,16 +3951,17 @@
 =item B<--graphics-markup=level>
 
  Change highlight style for graphics embedded with C<\includegraphics> commands.
-                      Possible values for level:
 
-                      C<none, C<off> or C<0>: no highlighting for figures
+Possible values for level:
 
-                      C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
+C<none>, C<off> or C<0>: no highlighting for figures
 
-                      C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
-                                scale, and crossed out with a red diagonal cross. Use configuration
-                                variable SCALEDELGRAPHICS to set size of deleted figures.
+C<new-only> or C<1>: surround newly added or changed figures with a blue frame [Default if graphicx package loaded]
 
+C<both> or C<2>:     highlight new figures with a blue frame and show deleted figures at reduced 
+scale, and crossed out with a red diagonal cross. Use configuration
+variable SCALEDELGRAPHICS to set size of deleted figures.
+
 Note that changes to the optional parameters will make the figure appear as changed 
 to latexdiff, and this figure will thus be highlighted
 
@@ -3745,7 +4017,7 @@
 
 Suppress inclusion of old and new file names as comment in output file
 
-=item B<--visble-label>
+=item B<--visible-label>
 
 Include old and new filenames (or labels set with C<--label> option) as 
 visible output.
@@ -4054,8 +4326,8 @@
 
 =head1 AUTHOR
 
-Version 1.2.0
-Copyright (C) 2004-2015 Frederik Tilmann
+Version 1.2.1
+Copyright (C) 2004-2017 Frederik Tilmann
 
 This program is free software; you can redistribute it and/or modify
 it under the terms of the GNU General Public License Version 3
@@ -4337,9 +4609,68 @@
 institute
 %%END CONTEXT2 COMMANDS 
 
+%% CONFIGURATION variabe defaults
+%%BEGIN LISTENV CONFIG
+itemize
+description
+enumerate
+%%END LISTENV CONFIG
 
-%% TYPES (Commands for highlighting changed blocks)
+%%BEGIN FLOATENV CONFIG
+figure[\w\d*@]*
+table[\w\d*@]*
+plate[\w\d*@]*
+%%END FLOATENV CONFIG
 
+%%BEGIN PICTUREENV CONFIG
+picture[\w\d*@]*
+tikzpicture[\w\d*@]*
+DIFnomarkup
+%%END PICTUREENV CONFIG
+
+%%BEGIN MATHENV CONFIG
+equation[*]?
+displaymath
+DOLLARDOLLAR
+%%END MATHENV CONFIG
+
+%%BEGIN MATHARRENV CONFIG
+eqnarray[*]?
+align[*]?
+alignat[*]?
+gather[*]?
+multline[*]?
+flalign[*]?
+%%END MATHARRENV CONFIG
+
+%%BEGIN ARRENV CONFIG
+aligned
+array
+[pbvBV]?matrix
+smallmatrix
+cases
+split
+%%END ARRENV CONFIG
+
+%%BEGIN COUNTERCMD CONFIG
+footnote
+part
+chapter
+section
+subsection
+subsubsection
+paragraph
+subparagraph
+%%END COUNTERCMD CONFIG
+
+%%BEGIN VERBATIMENV CONFIG
+verbatim[*]?
+lstlisting
+comment
+%%END VERBATIMENV CONFIG
+
+%%% TYPES (Commands for highlighting changed blocks)
+
 %DIF UNDERLINE PREAMBLE
 \RequirePackage[normalem]{ulem}
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
@@ -4350,6 +4681,7 @@
 %DIF CTRADITIONAL PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} [..\footnote{removed: #1} ]}}
 %DIF END CTRADITIONAL PREAMBLE
@@ -4356,6 +4688,7 @@
 
 %DIF TRADITIONAL PREAMBLE
 \RequirePackage[stable]{footmisc}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{[..\footnote{removed: #1} ]}}
 %DIF END TRADITIONAL PREAMBLE
@@ -4362,6 +4695,7 @@
 
 %DIF CFONT PREAMBLE
 \RequirePackage{color}\definecolor{RED}{rgb}{1,0,0}\definecolor{BLUE}{rgb}{0,0,1}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\protect\color{blue} \sf #1}}
 \providecommand{\DIFdel}[1]{{\protect\color{red} \scriptsize #1}}
 %DIF END CFONT PREAMBLE
@@ -4368,6 +4702,7 @@
 
 %DIF FONTSTRIKE PREAMBLE
 \RequirePackage[normalem]{ulem}
+\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf}
 \providecommand{\DIFadd}[1]{{\sf #1}}
 \providecommand{\DIFdel}[1]{{\footnotesize \sout{#1}}}
 %DIF END FONTSTRIKE PREAMBLE
@@ -4406,6 +4741,7 @@
 %DIF END INVISIBLE PREAMBLE
 
 %DIF BOLD PREAMBLE
+\DeclareOldFontCommand{\bf}{\normalfont\bfseries}{\mathbf}
 \providecommand{\DIFadd}[1]{{\bf #1}}
 \providecommand{\DIFdel}[1]{}
 %DIF END BOLD PREAMBLE
@@ -4430,7 +4766,7 @@
 \providecommand{\DIFaddend}{\protect\marginpar{]}}
 \providecommand{\DIFdelbegin}{\protect\marginpar{d[}}
 \providecommand{\DIFdelend}{\protect\marginpar{]}}
-%DIF END BRACKET PREAMBLE
+%DIF END MARGIN PREAMBLE
 
 %DIF DVIPSCOL PREAMBLE
 %Note: only works with dvips converter
@@ -4623,4 +4959,18 @@
 %DIF HYPERREF PREAMBLE
 \providecommand{\DIFadd}[1]{\texorpdfstring{\DIFaddtex{#1}}{#1}}
 \providecommand{\DIFdel}[1]{\texorpdfstring{\DIFdeltex{#1}}{}}
-%DIF END HYPERREF PACKAGE
+%DIF END HYPERREF PREAMBLE
+
+%DIF LISTINGS PREAMBLE
+\lstdefinelanguage{codediff}{
+  moredelim=**[is][\color{red}]{*!----}{----!*},
+  moredelim=**[is][\color{blue}]{*!++++}{++++!*}
+}
+\lstdefinestyle{codediff}{
+	belowcaptionskip=.25\baselineskip,
+	language=codediff,
+	basicstyle=\ttfamily,
+	columns=fullflexible,
+	keepspaces=true,
+}
+%DIF END LISTINGS PREAMBLE



More information about the tex-live-commits mailing list