Ich habe eine CSV-Datei mit Tags und (Seiten-)Nummern wie folgt:
tag,nr
t1,1
t2,5
t3,9
Die Nr.-Spalte bezieht sich auf Seitenzahlen in einer PDF-Datei. Jede Seite hat ein entsprechendes Tag. Bei einem gegebenen Tag möchte ich die entsprechende Seite in einer Abbildung in meinem Latex-Dokument anzeigen.
Mit Datatools kann ich die CSV-Datei laden und die Seitennummer eines bestimmten Tags abrufen. Für das Tag t2
erhalte ich beispielsweise die Seitennummer 5
.
Aber wenn ich jetzt diese Seitennummer als page=...
Argument für includegraphics eingebe, erhalte ich die Fehlermeldung
"! Fehlende Zahl, wird als Null behandelt."`
Welchen zusätzlichen Schritt muss ich unternehmen, um die Latex- 5
Zeichenfolge in eine Seitennummer umzuwandeln, die von Includegraphics verarbeitet werden kann?
Hier ist eine vereinfachte Version (mit Zählern anstelle von Includegraphics), die das Problem auslöst:
\documentclass{article}
\usepackage{datatool}
\begin{document}
\DTLloaddb{tags}{prestags.csv}
\newcommand{\pagenr}{\DTLfetch{tags}{tag}{t2}{nr}}
\newcounter{snr}
\addtocounter{snr}{\pagenr}
\end{document}
% Output:
! Missing number, treated as zero.
<to be read again>
\let
l.10 \addtocounter{snr}{\pagenr}
Was kann ich tun, um das Problem zu beheben?
Antwort1
Das Problem ist, dass das \DTLfetch
Makro nicht erweiterbar ist. Sie können die hier beschriebene Technik verwenden:
um dies zu überwinden. Hier ist ein ausgearbeitetes Beispiel:
\documentclass{article}
\begin{filecontents}{\jobname.csv}
tag,nr
t1,1
t2,5
t3,9
\end{filecontents}
\usepackage{datatool}
\newcommand{\getvar}[1]{\DTLfetchsave{\myvar}{tags}{tag}{#1}{nr}}
\newcommand{\DTLfetchsave}[5]{%
\edtlgetrowforvalue{#2}{\dtlcolumnindex{#2}{#3}}{#4}%
\dtlgetentryfromcurrentrow{\dtlcurrentvalue}{\dtlcolumnindex{#2}{#5}}%
\let#1\dtlcurrentvalue
}
\newcounter{snr}
\begin{document}
\DTLloaddb{tags}{prestags.csv}
\setcounter{snr}{3}
The counter value is \thesnr
\getvar{t2}
\addtocounter{snr}{\myvar}
The counter value is: \thesnr
\end{document}
Antwort2
Hier verwende ich das readarray
Paket, um die CSV-Datei zu verarbeiten und ein listofitems
Lese-Objekt, um die Daten in eine Liste zu platzieren. Das Makro \getvar
ist erweiterbar.
\begin{filecontents*}[overwrite]{prestags.csv}
tag,nr
t1,1
t2,5
t3,9
\end{filecontents*}
\documentclass{article}
\usepackage{readarray,graphicx}
\begin{document}
\readarraysepchar{\\}
\readdef{prestags.csv}\mycsvdata
\setsepchar{\\/,}
\readlist*{\mycsvlist}{\mycsvdata}
\foreachitem\z\in\mycsvlist[]{%
\expandafter\gdef\csname mytag\mycsvlist[\zcnt,1]\expandafter
\endcsname\expandafter{\expandafter
\mycsvlist\expandafter[\zcnt,2]}%
}
\newcommand\getvar[1]{\csname mytag#1\endcsname}
The t2 tag corresonds with page \getvar{t2}
\includegraphics[page=\getvar{t3},scale=.6]{readarray.pdf}
\end{document}
Antwort3
Wenn es Ihnen nichts ausmacht, ein Scratch-Makro global zu definieren, können Sie wahrscheinlich Folgendes verwenden \DTLassignfirstmatch
:
Beim Kompilieren des folgenden Beispiels erhalten Sie zwei Fehlermeldungen.
Dies ist beabsichtigt, damit Sie sehen können, was beim Versuch passiert, nicht vorhandene Werte abzurufen.
\begin{filecontents*}{prestags.csv}
tag,nr
t1,1
t2,5
t3,9
\end{filecontents*}
\makeatletter
\newcommand\MyStuffDTLfetchsavePackageError[7]{%
\PackageError{#1}{#7}{%
In order to retrieve from database `#2`\MessageBreak
the value of the key `#5` in the first row\MessageBreak
where the key `#3` has value `#4`,\MessageBreak
database `#2` must exist and have keys\MessageBreak
`#3` and `#5` and a row where the key `#3`\MessageBreak
has the value `#4`.%
}%
\global\let#6=\dtlnovalue
}%
\DeclareRobustCommand{\MyStuffDTLfetchsave}[5]{%
%#1 = macro to store as result
%#2 = data base name
%#3 = key 1
%#4 = value of key 1
%#5 = key 2
\@bsphack
\DTLifdbexists{#2}{%
\DTLifhaskey{#2}{#3}{%
\DTLifhaskey{#2}{#5}{%
\global\let#1=\dtlnovalue
\DTLassignfirstmatch{#2}{#3}{#4}{#1=#5}%
\ifx#1\dtlnovalue\expandafter\@firstofone\else\expandafter\@gobble\fi
{%
\MyStuffDTLfetchsavePackageError{My Stuff}{#2}{#3}{#4}{#5}{#1}%
{In data base `#2' no row's key `#3' holds the value `#4'}%
}%
}{%
\MyStuffDTLfetchsavePackageError{My Stuff}{#2}{#3}{#4}{#5}{#1}%
{There is no key `#5' in data base `#2'}%
}%
}{%
\MyStuffDTLfetchsavePackageError{My Stuff}{#2}{#3}{#4}{#5}{#1}%
{There is no key `#3' in data base `#2'}%
}%
}{%
\MyStuffDTLfetchsavePackageError{My Stuff}{#2}{#3}{#4}{#5}{#1}%
{Data base `#2' doesn't exist}%
}%
\@esphack
}%
\makeatother
\documentclass{article}
\usepackage{datatool}
\newcounter{snr}
\begin{document}
\DTLloaddb{tags}{prestags.csv}
\setcounter{snr}{3}%
\noindent The counter value is: \thesnr
\MyStuffDTLfetchsave{\scratchy}{tags}{tag}{t1}{nr}%
\ifx\scratchy\dtlnovalue
\noindent The value of the key `nr` in the row where the key `tag` has the value `t1` could not be retrieved.
\noindent Calculations where this value would have been used were not performed.
\else
\noindent The value of the key `nr` in the row where the key `tag` has the value `t1` is: \scratchy
\addtocounter{snr}{\scratchy}%
\noindent After adding that value to the counter, the counter value is: \thesnr
\fi
\MyStuffDTLfetchsave{\scratchy}{tags}{tag}{t2}{nr}%
\ifx\scratchy\dtlnovalue
\noindent The value of the key `nr` in the row where the key `tag` has the value `t2` could not be retrieved.
\noindent Calculations where this value would have been used were not performed.
\else
\noindent The value of the key `nr` in the row where the key `tag` has the value `t2` is: \scratchy
\addtocounter{snr}{\scratchy}%
\noindent After adding that value to the counter, the counter value is: \thesnr
\fi
\MyStuffDTLfetchsave{\scratchy}{tags}{tag}{t3}{nr}%
\ifx\scratchy\dtlnovalue
\noindent The value of the key `nr` in the row where the key `tag` has the value `t3` could not be retrieved.
\noindent Calculations where this value would have been used were not performed.
\else
\noindent The value of the key `nr` in the row where the key `tag` has the value `t3` is: \scratchy
\addtocounter{snr}{\scratchy}%
\noindent After adding that value to the counter, the counter value is: \thesnr
\fi
\MyStuffDTLfetchsave{\scratchy}{tags}{tag}{t4}{nr}%
\ifx\scratchy\dtlnovalue
\noindent The value of the key `nr` in the row where the key `tag` has the value `t4` could not be retrieved.
\noindent Calculations where this value would have been used were not performed.
\else
\noindent The value of the key `nr` in the row where the key `tag` has the value `t4` is: \scratchy
\addtocounter{snr}{\scratchy}%
\noindent After adding that value to the counter, the counter value is: \thesnr
\fi
\end{document}
Antwort4
Wie vorgeschlagen von@cabohahkann die DTLfetch
Zeile in der Frage durch die folgenden beiden Zeilen ersetzt werden:
\DTLgetlocation{\rownr}{\colnr}{tags}{t2}
\DTLgetvalue{\pagenr}{tags}{\rownr}{\dtlcolumnindex{tags}{nr}}
Da es sich um eine einfache Änderung handelte, habe ich letztendlich Folgendes verwendet.