¿Cómo puedo hacer una macro similar a \averagecharwidth
ConTeX en LaTeX, que calcule el ancho promedio de un carácter en función de la frecuencia de ese carácter en mi documento? esa macro se muestra enestecorreo.
Respuesta1
He aquí un enfoque ingenuo.
- Almacene el documento completo en una lista de tokens
- Cuente el número de apariciones de cada carácter alfabético (en su mayoría)
- Divida el recuento de cada carácter por el número total de caracteres alfabéticos para obtener la frecuencia relativa de ese carácter.
- Multiplique esa proporción por el ancho del carácter y sume para obtener el ancho promedio del carácter.
Algunas notas:
- los grupos de llaves se cuentan como un solo token, por lo que cosas como
\begin{environment}
y\par
no coincidirán con ningún carácter alfabético, esto es una ventaja. - Al mismo tiempo, las palabras que contiene
\text{some text}
no se cuentan, lo cual es una desventaja. - Se pueden tener en cuenta las mayúsculas pero es lento.
- No creo que me haya perdido nada importante, pero nunca se sabe.
- Editar:Los espacios ahora se incluyen en el cálculo y el efecto de la macro es acumulativo. Al tratar con espacios, supuse que el alargamiento y la contracción se anulan mutuamente a largo plazo y que el ancho promedio de un espacio es simplemente el ancho normal de un espacio. Alguien, por favor, hágamelo saber si hay una mejor manera de lidiar con eso.
- Editar:Compile dos veces para ajustar automáticamente el ancho del texto al valor deseado.
De todos modos, para texto sencillo, esto proporciona un ancho de carácter promedio exacto. El resultado se vuelve menos preciso si se oculta más texto impreso entre grupos de llaves.
\documentclass{article}
\usepackage{xparse}
\usepackage{siunitx}
\usepackage{booktabs}
\usepackage{environ}
\ExplSyntaxOn
\bool_new:N \g_has_run_bool
\tl_new:N \l_aw_text_tl
\int_new:N \l_aw_tot_int
\int_new:N \g_aw_tot_alph_int
\int_new:N \g_wid_space_int
\int_new:N \g_space_int
\fp_new:N \g_rat_space_int
\fp_new:N \g_aw_avg_width_fp
\dim_new:N \myalphabetwidth
\dim_new:N \mytextwidth
\input{testing.aux}
\tl_const:Nx \c_aw_the_alphabet_tl {abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ,.;?()!' \token_to_str:N :}
% this can be changed to an evironment or renamed or whatever
\NewDocumentCommand {\avgwidthstart} {}
{
\aw_avg_width:w
}
\NewDocumentCommand {\avgwidthend}{}{}
% Here is the environment version, using just "text" as a name is probably a bad idea.
\NewEnviron{awtext}
{
\expandafter\avgwidthstart\BODY\avgwidthend
}
\makeatletter
\cs_new:Npn \aw_avg_width:w #1 \avgwidthend
{
% if first run, then generate variables to be used
\bool_if:NF \g_has_run_bool
{
\tl_map_inline:Nn \c_aw_the_alphabet_tl
{
\int_new:c {g_##1_int}
\fp_new:c {g_rat_##1_fp}
\fp_new:c {g_wid_##1_fp}
}
}
\tl_set:Nn \l_aw_text_tl {#1}
% this can be used rather than the preceding line to take capital
% letters into account, but is Slooooooow
%\tl_set:Nx \l_aw_text_tl {\tl_expandable_lowercase:n {#1}}
\int_set:Nn \l_aw_tot_int {\tl_count:N \l_aw_text_tl}
\tl_map_function:NN \c_aw_the_alphabet_tl \aw_get_counts:n
\deal_with_spaces:n {#1}
\tl_map_function:NN \c_aw_the_alphabet_tl \aw_calc_ratios:n
\tl_map_function:NN \c_aw_the_alphabet_tl \aw_calc_avg_width:n
\fp_gset_eq:NN \g_aw_avg_width_fp \l_tmpa_fp
\fp_zero:N \l_tmpa_fp
% the dimension \myalphabetwidth gives the width of the alphabet based on your character freq,
% can be accessed by \the\myalphabetwidth
\dim_gset:Nn \myalphabetwidth {\fp_to_dim:n {\fp_eval:n {61*\g_aw_avg_width_fp}}}
% the dimension \mytextwidth gives the recommended \textwidth based on 66 chars per line.
% can be accessed by \the\mytextwidth
\dim_gset:Nn \mytextwidth {\fp_to_dim:n {\fp_eval:n {66*\g_aw_avg_width_fp}}}
\protected@write\@mainaux{}{\mytextwidth=\the\mytextwidth}
\bool_gset_true:N \g_has_run_bool
% and lastly print the content
#1
}
\makeatother
\cs_new:Npn \aw_get_counts:n #1
{
% make a temporary token list from the document body
\tl_set_eq:NN \l_tmpb_tl \l_aw_text_tl
% remove all occurrences of the character
\tl_remove_all:Nn \l_tmpb_tl {#1}
% add to appropriate int the number of occurrences of that character in current block
\int_set:Nn \l_tmpa_int {\int_eval:n{\l_aw_tot_int -\tl_count:N \l_tmpb_tl}}
% add to appropriate int the number of occurrences of that character in current block
\int_gadd:cn {g_#1_int} {\l_tmpa_int}
% add this to the total
\int_gadd:Nn \g_aw_tot_alph_int {\l_tmpa_int}
}
\cs_new:Npn \deal_with_spaces:n #1
{
\tl_set:Nn \l_tmpa_tl {#1}
% rescan body with spaces as characters
\tl_set_rescan:Nnn \l_tmpb_tl {\char_set_catcode_letter:N \ }{#1}
% find number of new characters introduced. add to number of spaces and alph chars
\int_set:Nn \l_tmpa_int {\tl_count:N \l_tmpb_tl -\tl_count:N \l_tmpa_tl}
\int_gadd:Nn \g_space_int {\l_tmpa_int}
\int_gadd:Nn \g_aw_tot_alph_int {\l_tmpa_int}
% since this comes after the rest of chars are dealt with, tot_alph is final total
\fp_set:Nn \g_rat_space_fp {\g_space_int/\g_aw_tot_alph_int}
% get width of space and use it. obviously space is stretchable, so i'll assume
% that the expansions and contractions cancel one another over large text. is this
% a terrible assumption???
\hbox_set:Nn \l_tmpa_box {\ }
\fp_gset:Nn \g_wid_space_fp {\dim_to_fp:n {\box_wd:N \l_tmpa_box}}
\fp_add:Nn \l_tmpa_fp {\g_wid_space_fp*\g_rat_space_fp}
}
\cs_new:Npn \aw_calc_ratios:n #1
{
% divide number of occurrences of char by total alphabetic chars
\fp_gset:cn {g_rat_#1_fp}{{\int_use:c {g_#1_int}}/\g_aw_tot_alph_int}
}
\cs_new:Npn \aw_calc_avg_width:n #1
{
% only need to find char widths once
\bool_if:NF \g_has_run_bool
{
% find width of char box
\hbox_set:Nn \l_tmpa_box {#1}
\fp_gset:cn {g_wid_#1_fp}{\dim_to_fp:n {\box_wd:N \l_tmpa_box}}
}
% multiply it by char frequency and add to avg width
\fp_add:Nn \l_tmpa_fp {{\fp_use:c {g_wid_#1_fp}}*{\fp_use:c {g_rat_#1_fp}}}
}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% This part is just for fun. Delete it and the showtable command from the document if
% it isn't wanted
\tl_new:N \l_aw_tab_rows_tl
\seq_new:N \g_aw_the_alphabet_seq
\NewDocumentCommand {\showtable}{}
{
\clearpage
\aw_make_table:
}
\cs_generate_variant:Nn \seq_set_split:Nnn {NnV}
\cs_new:Npn \aw_make_table:
{
\thispagestyle{empty}
\seq_set_split:NnV \g_aw_the_alphabet_seq {} \c_aw_the_alphabet_tl
\seq_map_function:NN \g_aw_the_alphabet_seq \aw_generate_row:n
\begin{table}
\centering
\sisetup{round-mode = places,round-precision = 5,output-decimal-marker={,},table-format = 3.5}
\begin{tabular}{lll}
\toprule
{Average\,text\,width}&{Average\,character\,width}&{Average\,alphabet\,width}\\
\midrule
\the\mytextwidth&\fp_eval:n {round(\g_aw_avg_width_fp,5)}pt&\the\myalphabetwidth\\
\bottomrule
\end{tabular}\par
\end{table}
\vfil
\centering
\sisetup{round-mode = places,round-precision = 5,output-decimal-marker={,},table-format = 3.5}
\begin{longtable}{cS}
\toprule
{Letter}&{Actual}\\
\midrule
spaces&\fp_eval:n {\g_rat_space_fp*100}\%\\
\tl_use:N \l_aw_tab_rows_tl
\bottomrule
\end{longtable}\par
}
\cs_new:Npn \aw_generate_row:n #1
{
\tl_put_right:Nn \l_aw_tab_rows_tl {#1&}
\tl_put_right:Nx \l_aw_tab_rows_tl {\fp_eval:n {100*{\fp_use:c {g_rat_#1_fp}}}\%}
\tl_put_right:Nn \l_aw_tab_rows_tl {\\}
}
\ExplSyntaxOff
\begin{document}
\avgwidthstart
My audit group's Group Manager and his wife have an infant I can describe only as fierce.
Its expression is fierce; its demeanor is fierce; its gaze over bottle or pacifier or finger-fierce,
intimidating, aggressive. I have never heard it cry. When it feeds or sleeps, its pale face reddens,
which makes it look all the fiercer.
\avgwidthend
\avgwidthstart
On those workdays when our Group Manager, Mr. Yeagle, brought it in to the District office, hanging papoose-style in a nylon device on his back, the infant appeared to
be riding him as a mahout does an elephant. It hung there, radiating authority. Its back lay directly
against Mr. Yeagle's, its large head resting in the hollow of its father's neck and forcing our Group
Manager's head out and down into a posture of classic oppression. They made a creature with two faces,
one of which was calm and blandly adult and the other unformed and yet emphatically fierce. The infant
never wiggled or fussed in the device. Its gaze around the corridor at the rest of us gathered waiting
for the morning elevator was level and unblinking and (it seemed) almost accusing. The infant's face, as
I experienced it, was mostly eyes and lower lip, its nose a mere pinch, its forehead milky and domed,
its pale red hair wispy, no eyebrows or lashes or even eyelids I could see. I never saw it blink. Its
features seemed suggestions only. It had roughly as much face as a whale does. I did not like it at all.\par\noindent
http://harpers.org/media/pdf/dfw/HarpersMagazine-2008-02-0081893.pdf
\avgwidthend
\begin{awtext}
Here is some more text in an environment this time. This text is included in the calculation of the average width.
\end{awtext}
\showtable{}
\end{document}
Explicación La esencia que obtengo de esto del "ancho promedio de un carácter" es la siguiente.
- La gente ha decidido que tener ~66 caracteres por línea mejora la legibilidad del texto.
- Dado que el ancho de línea es fijo, el número real de caracteres por línea depende de qué caracteres se escriben, por ejemplo, una línea de todos
m
contendrá menos caracteres que una línea de todos,i
ya que anm
es más ancha que ani
. - Por lo tanto, para establecer un ancho de línea razonable para aproximadamente 66 caracteres por línea, necesitamos conocer las frecuencias relativas de los caracteres que se utilizan en el documento. Si la mayoría de los caracteres son anchos, entonces necesitamos líneas más anchas. Si la mayoría de los caracteres son estrechos entonces necesitamos líneas correspondientemente más estrechas.
- Por lo tanto, calculamos el ancho promedio de los caracteres que se utilizan y lo utilizamos para determinar el ancho de línea. Por ejemplo, si nuestro documento consta de
m
' yi
' en la misma proporción (50/50), entonces "el carácter promedio" tiene un ancho entre el ancho de anm
y el de ani
. Específicamente, el carácter promedio tiene anchox=(wd(m)+wd(i))/2
y debemos establecerlo\textwidth
en66*x
. Extrapolando a un documento arbitrario calculamos el promedio ponderado de los anchos de los caracteres utilizados de acuerdo con sus frecuencias relativas dentro del documento, y lo multiplicamos por 66 (o lo usamos de cualquier manera) para obtener el\textwidth
que mejor se adapte a los 66 caracteres por línea. criterios.
Respuesta2
Las macros son TeX de nivel bastante bajo, por lo que es fácil usarlas en LaTeX agregando algunas definiciones faltantes. Con estas definiciones implementadas, simplemente puede importaridioma-frq.mkii,
idioma-frd.mkiiy el archivo auxiliarsupp-mis.mkii(en la página de destino, haga clic raw
para descargar) y use ConTeXt \averagecharwidth
directamente.
% Copy definition of \emptybox from supp-box.mkii
\ifx\voidbox\undefined \newbox\voidbox \fi
\def\emptybox{\box\voidbox}
% Copy definition of \startnointerference from syst-new.mkii
\newbox\nointerferencebox
\def\startnointerference
{\setbox\nointerferencebox\vbox
\bgroup}
\def\stopnointerference
{\egroup
\setbox\nointerferencebox\emptybox}
% Load a trimmed down version of ConTeXt macros
\input supp-mis.mkii
\input lang-frq.mkii
\input lang-frd.mkii
% Set the main language. (I don't know what the LateX equivalent of
% \currentmainlanguage)
\def\currentmainlanguage{en}
\documentclass{article}
\begin{document}
The average character width is \the\averagecharwidth
\end{document}
NOTA: Línea de comentario 116 de lang-frd.mkii
(la que dice \startcharactertable[en] 100 x \stopcharactertable % kind of default
).