
% We need to know the vertical position on the page where the figures
% are located. Because this position can't be known until the page is
% fully composed ("shipout"), the value is only accurate after the
% second run, just like the \ref command. The value from the previous
% run is available in the .aux file.
\RequirePackage{zref-savepos}

% The zref package also provides the page number. It gives the
% absolute page number, not the logical page number. Things like the
% table of contents and preface are part of the count. This value is
% available from abspage. This number is zero-based, so that the
% "first" page is numbered as zero. Just as positions within a page
% are not really known until a page is shipped out, the page number
% isn't known either. Generally, if you're near the middle or the end
% of the page, then \thepage or abspage is correct, but it is often
% wrong near the top of the page. The only way around this seems to be
% to place a label on the page, in a manner similar to what is done
% for positions within the page.
\RequirePackage{zref-thepage}
\RequirePackage{zref-abspage}
\RequirePackage{zref-user}
\RequirePackage{zref-pagelayout}
\RequirePackage{xsim}
\RequirePackage{tikz}

% This is needed to help with the process of replacing the body of
% the environment and (optionally) copying it to an external file.
\RequirePackage{verbatim}


\ExplSyntaxOn

% The fig.aux file is open throughout the run.
\iow_new:N \g_figurefile_iow
\iow_open:Nn \g_figurefile_iow { \jobname.fig.aux }

% The inner and outer margin widths
\dim_new:N \l_figput_innermargin_dim
\dim_new:N \l_figput_outermargin_dim

% By default, set these margins to have zero width. This has no effect
% on how the text is laid out by latex. It's used on the browser end
% to set the origin for drawing. In turn, that affects where the
% figures end up when the tikz is loaded.
%
% BUG: There should be a way (?) to determine where the left margin of
% the text is from latex instead of inputing your best approximation.
% At the same time, the user might want to adjust this regardless.
\dim_set:Nn \l_figput_innermargin_dim { 0 pt }
\dim_set:Nn \l_figput_outermargin_dim { 0 pt }

% Two little commands to set these values.
\NewDocumentCommand\SetInnerMargin { m }
{
  \dim_set:Nn \l_figput_innermargin_dim { #1 }
}

\NewDocumentCommand\SetOuterMargin { m }
{
  \dim_set:Nn \l_figput_outermargin_dim { #1 }
}

% Use this to turn off any of the optional "skip" arguments to figput.
% If you call \NeverSkip, then these "skip" arguments are ignored.
\bool_new:N \l_figput_skipoff_bool
\bool_set_false:N \l_figput_skipoff_bool

\NewDocumentCommand\NeverSkip{} {
  \bool_set_true:N \l_figput_skipoff_bool
}

\NewDocumentCommand\AllowSkip{} {
  \bool_set_false:N \l_figput_skipoff_bool
}

% Use this to request that a specific file be loaded by the browser.
% This simply writes a line to the .aux file.
\NewDocumentCommand\LoadFigureCode{ m }
{
  \group_begin:
  
  \iow_now:Nx \g_figurefile_iow { load~ #1 }
  
  \group_end:
}


% The variables used below:
%
% Manditory name for the figure, as given by the user. Using string,
% although maybe token list would be better?
\str_new:N \l_figput_figname_str

% The manditory height given by the user. This is a dim, as opposed to
% merely a floating-point value. 
\dim_new:N \l_figput_height_dim

% The optional height adjustments given by the user. Again, dims.
\dim_new:N \l_figput_htAbove_dim
\dim_new:N \l_figput_htBelow_dim

% Scratch values used for calculation of figure position.
\tl_new:N \l_figput_tempa_tl
\tl_new:N \l_figput_tempb_tl
\tl_new:N \l_figput_tempc_tl

% And several booleans.
% BUG: In theory, it would be possible to get rid of nostatic.
% If the figure height is set to zero, then that should implicitly
% imply nostatic.
\bool_new:N \l_figput_hasintht_bool
\bool_new:N \l_figput_nostatic_bool
\bool_new:N \l_figput_done_bool
\bool_new:N \l_figput_skip_bool

% This is for the figures file: the position of the figure on the page.
\dim_new:N \l_figput_pagepos_dim

% BUG: I don't understand the whole concept of these variants of
% functions. Somehow it deals with various cases of whether arguments 
% are expanded or not. There's another variant defintion below as well.
\cs_generate_variant:Nn \file_if_exist:nTF {V}
\cs_generate_variant:Nn \file_input:n {V}



% This command works in (almost) exactly the same way as the figput 
% environment. The differenence is that, as a command instead of
% an environment, there is no body to deal with. Most of the comments
% have been removed. See the definition of the environment for *much*
% more detailed comments.
%
% BUG: possible to define a function and combine these?
\NewDocumentCommand\FigPut { > { \SplitArgument { 1 } { , } } m!o }
{
  \group_begin:

  % The manditory arguments come in in #1
  \figput_manargs #1

  % Now the optional arguments.
  \bool_set_false:N \l_figput_hasintht_bool
  \bool_set_false:N \l_figput_nostatic_bool
  \bool_set_false:N \l_figput_done_bool
  \bool_set_false:N \l_figput_skip_bool
  
  \tl_if_blank:nTF { #2 } {} {
    \clist_set:Nn \l_figput_optclist_clist { #2 }
    \figput_optargs { } 
  }

  % We have everything needed to know what to do.
  \str_set:Nx \l_figput_filespec_str { 
    \use:c { zref@extractdefault } { \l_figput_figname_str -pageno} { abspage } { 0 }
  }

  % The inner and outer margins.
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_innermargin_dim }
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_outermargin_dim }

  % And the text width.
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \textwidth }

  % Massive fooling around to do arithmetic on page position.
  \tl_set:Nn \l_figput_tempa_tl { \zposy { \l_figput_figname_str } }
  % Convert the figure height to sp.
  \tl_set:Nn \l_figput_tempb_tl { \dim_to_decimal_in_sp:n{ \l_figput_height_dim } }
  % Add 
  \tl_set:Nn \l_figput_tempc_tl { \int_eval:n { \l_figput_tempa_tl + \l_figput_tempb_tl } }
  % Convert to a dim
  \dim_set:Nn \l_figput_pagepos_dim { \l_figput_tempc_tl sp }
  % And (whew!) write it out, in bp.
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_pagepos_dim }

  % The latex height of the figure (in bp)
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_height_dim }

  % The interactive height paddings, which are zero by default.
  \bool_if:NTF \l_figput_hasintht_bool
  { 
    \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_htAbove_dim }
    \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_htBelow_dim }
  }
  {
    \str_put_right:Nx \l_figput_filespec_str { ~0~0 }
  }

  % The figure name -- used for the JS function to call.
  \str_put_right:Nx \l_figput_filespec_str { ~ \l_figput_figname_str }

  % And a boolean for whether the tikz is 'done.'
  \bool_if:NTF \l_figput_done_bool
  {
    \str_put_right:Nx \l_figput_filespec_str { ~ true }
  }
  {
    \str_put_right:Nx \l_figput_filespec_str { ~ false }
  }
  
  % And indicate that this function must appear in an external file.
  % It is not (false) to be loaded from a seperate file.
  \str_put_right:Nx \l_figput_filespec_str { ~ false }

  % String is ready. Write out the line.
  \iow_now:Nx \g_figurefile_iow { \l_figput_filespec_str }
  
  % That's it for the figures.aux file. Next is the tikz.
  % Above is identical to the environment code, but the following differs
  % slightly since there is no body.
  \bool_set_true:N \l_figput_readtikz_bool
  
  \bool_if:NTF \l_figput_skip_bool
  {
    \bool_set_false:N \l_figput_readtikz_bool
  }
  {}

  % In the environment definition, this was part of the "post body"
  % set of commands.
  \bool_if:NTF \l_figput_nostatic_bool
  {
    % nostatic is set, so do nothing at all -- no figure in the static version.
  }
  {
    % nostatic is not set, so there is a figure.
    \str_set_eq:NN \l_figput_tikzfile_str \l_figput_figname_str
    \str_put_right:Nn \l_figput_tikzfile_str { .tikz }

    % Whether to read the tikz file or display a "not available" message
    % depends on whether the tikz file exists and whether skip was set.
    \file_if_exist:VTF \l_figput_tikzfile_str
    {
      % Yes, tikz exists, although we may not read it if skip is set.
    }
    {
      % Tikz file doesn't exist, so we can't read it in any case.
      \bool_set_false:N \l_figput_readtikz_bool
    }
    
    \bool_if:NTF \l_figput_readtikz_bool
    {
      % Load the tikz file here.
      \file_input:V \l_figput_tikzfile_str
    } 
    {
      % Not loading the tikz file. Display a big empty box.
      \begin{tikzpicture}
        \useasboundingbox (0pt,0pt) rectangle (\textwidth,\l_figput_height_dim);
        \draw[dashed] (0pt,0pt) rectangle ( \textwidth,\l_figput_height_dim);
        \node at (\textwidth / 2,\l_figput_height_dim / 2) {The\ drawing\ is\ not\ available\ to\ load.};
        \draw (5pt,5pt) rectangle ( \textwidth - 5,\l_figput_height_dim - 5);
      \end{tikzpicture}
    }

    % End of figure inclusion -- nostatic is not set.
  }
  
  % Note the figures location as of after the figure.
  \par\zsaveposy { \l_figput_figname_str }
  \zlabel{ \l_figput_figname_str -pageno}

  \group_end:  
}



% See xparse for the top-level definition, \NewDocumentEnvironment and
% its use of \SplitArgument. The gist is that there will be two
% arguments, one for the manditory items (m), given in {}, and one for
% the optional arguments (o), given in [].
%
% In theory, it is possible to take the body of the environment as an
% argument, but that won't work if you want to receive the body
% verbatim. Somehow, using xsim allows the body to be taken verbatim.
%
% Note also the ! that appears before the o. This is explained in the
% xparse documentation. Without the !, if the user does not provide
% any optional arguments, then any leading space in the body will be
% eaten and disappear.

\NewDocumentEnvironment{figput} { > { \SplitArgument { 1 } { , } } m!o }
{
  % I don't understand latex's concept of local variable, but
  % this makes what follows local in some sense.
  % BUG: Does this even matter? It looks like all my variables are in
  % the figput module.
  \group_begin:

  % The manditory arguments come in in #1, but they have been
  % broken up so that #1 "reads as" {first arg}{second arg}. By
  % passing this to another function, it automatically breaks this
  % into two distinct arguments. This seems a silly way, but it works.
  % This sets \l_figput_figname_str and \l_figput_height_dim.
  \figput_manargs #1

  % Now the optional arguments. Start off with some default values.
  % It looks like the htAbove and htBelow values are blank by default,
  % which is what I want.
  \bool_set_false:N \l_figput_hasintht_bool
  \bool_set_false:N \l_figput_nostatic_bool
  \bool_set_false:N \l_figput_done_bool
  \bool_set_false:N \l_figput_skip_bool
  
  % Only try to set these values if there *were* some optional arguments,
  % which come in as the #2 argument to figput.
  % I set the clist here and "pass" it as global variable since I can't
  % figure out how to pass the #2 argument directly. This works, but it
  % is ugly. So this does nothing if the #2 argument is empty, leaving
  % the boolean values as set above. Otherwise, it reads in the
  % height adjustments and sets the boolean values above.
  \tl_if_blank:nTF { #2 } {} {
    \clist_set:Nn \l_figput_optclist_clist { #2 }
    \figput_optargs { } 
  }

  % We have everything needed to know what to do. Prepare to write the
  % data to the figures file. It seems that the easiest way to do this
  % is to build up the string, then write it out. Getting spaces right
  % is hard when I tried to write each bit directly to the file.
  % Unravelling what's actually written and its meaning is hard to do
  % here. It's easier to understand the js code that reads this output.
  
  % Need the position on the page. This saves it (for the next run).
  % Just as for the page number (see below), it turns out that this
  % shouldn't be done here; if it is, the figures at the bottom of the page are
  % improperly positioned. This is done *after* the environment closes.
  %\zsaveposy { \l_figput_figname_str }
  
  % And this gets the value from the previous run.
  % This value is relative to the bottom of the page (normal axes)
  % See below. The possibility of a figure at the bottom of the page
  % requires a bit of trickery.
  %\dim_set:Nn \l_figput_pagepos_dim { \zposy { \l_figput_figname_str } sp}

  % Getting the page number correct is fiddly. We need to place a
  % label at this location, and this label is updated at shipout.
  % Because I am using the figure name as the label for the vertical
  % position, I add a suffix ("-pageno") to distinguish this label.
  % Originally, I had this line here, but that doesn't work because
  % the page number is that at which the environment "opens" rather
  % than when it "closes." If the figure is at the bottom of the page,
  % then it seems that you get the page prior to the one you want.
  %\zlabel{ \l_figput_figname_str -pageno}

  % I would have thought that saying
  % \zref[abspage]{ \l_figput_figname_str }
  % would work, but it's not properly expanded. It *does* work to say
  % the above and have it appear as a number on the page. I don't fully
  % understand what the bit below does, but it works.
  \str_set:Nx \l_figput_filespec_str { 
    \use:c { zref@extractdefault } { \l_figput_figname_str -pageno} { abspage } { 0 }
  }

  % The inner and outer margins.
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_innermargin_dim }
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_outermargin_dim }

  % And the text width.
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \textwidth }

  % Use tilde (~) to get a space in the string, and convert to bp (big
  % points). Fortunately, this eliminates the "bp" suffix too.
  % There is a problem here, related to whether the figure is at the
  % bottom of the page and gets bumped to the next page.
  % I want to know the location of the top of the figure relative to
  % the page on which it appears. The way to get the position on the
  % page is with \zsaveposy. The problem is that if I want this
  % location *before* the figure, then you get the bottom of the text
  % immediately before the figure; if you ask for this location
  % *after* the figure (or from within the figure), then you get the
  % location of the bottom edge of the figure.
  %
  % The solution is to get the location of the bottom edge of the
  % figure, then subtract the known figure height -- actually we *add*
  % because we are in "normal" coordinates. We want to move up on the
  % page.
  %
  % So, get the value from the previous run, in sp (small or scaled points).
  \tl_set:Nn \l_figput_tempa_tl { \zposy { \l_figput_figname_str } }
  % Convert the figure height to sp.
  \tl_set:Nn \l_figput_tempb_tl { \dim_to_decimal_in_sp:n{ \l_figput_height_dim } }
  % Add 
  \tl_set:Nn \l_figput_tempc_tl { \int_eval:n { \l_figput_tempa_tl + \l_figput_tempb_tl } }
  % Convert to a dim
  \dim_set:Nn \l_figput_pagepos_dim { \l_figput_tempc_tl sp }
  % And (whew!) write it out, in bp.
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_pagepos_dim }

  % The latex height of the figure (in bp)
  \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_height_dim }

  % The interactive height paddings, which are zero by default.
  \bool_if:NTF \l_figput_hasintht_bool
  { 
    \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_htAbove_dim }
    \str_put_right:Nx \l_figput_filespec_str { ~ \dim_to_decimal_in_bp:n \l_figput_htBelow_dim }
  }
  {
    \str_put_right:Nx \l_figput_filespec_str { ~0~0 }
  }

  % The figure name -- used for the JS function to call.
  \str_put_right:Nx \l_figput_filespec_str { ~ \l_figput_figname_str }

  % And a boolean
  \bool_if:NTF \l_figput_done_bool
  {
    \str_put_right:Nx \l_figput_filespec_str { ~ true }
  }
  {
    \str_put_right:Nx \l_figput_filespec_str { ~ false }
  }

  % We're making a seperate file for this particular function.
  % Tell the browser (true) to load it.
  \str_put_right:Nx \l_figput_filespec_str { ~ true }
  
  % String is ready. Write out the line.
  \iow_now:Nx \g_figurefile_iow { \l_figput_filespec_str }
  
  % That's it for the fig.aux file. Next is the heavy lifting,
  % making the environment shift blocks of text around. Exactly what
  % to do depends on the optional flags.
  %
  % Ultimately, there are only two things we might do (plus nothing):
  % write the body of the environment to a file, and/or read a tikz
  % file in to replace the current body (which may be blank). The
  % easiest way to manage these options is to set a couple of booleans
  % to indicate whether we should do these two things.
  \bool_set_true:N \l_figput_writebody_bool
  \bool_set_true:N \l_figput_readtikz_bool

  % Note that there is no need to check nostatic here.
  
  \bool_if:NTF \l_figput_skip_bool
  {
    % skip == true, so don't write body or read tikz.
    \bool_set_false:N \l_figput_writebody_bool
    \bool_set_false:N \l_figput_readtikz_bool
  }
  {}

  \bool_if:NTF \l_figput_writebody_bool
  {
    % Yes, we are to write the body to an external file.
    
    % The output file name needs an extra ".fjs" on the end.
    % I don't like using this suffix, but it's seems the least bad.
    \str_set_eq:NN \l_figput_jinfile_str \l_figput_figname_str
    \str_put_right:Nn \l_figput_jinfile_str { .fjs }
  
    % Whether to add an additional EOL to the output file depends on
    % whether there were any optional arguments. That's what the boolean
    % argument does here. It tells the writer whether to start off with
    % an "extra" EOL.
    \IfValueTF {#2}
      { \xsim_file_write_start:nn { \c_true_bool } }
      { \xsim_file_write_start:nn { \c_false_bool } }
      { \l_figput_jinfile_str }
  }
  {
    % Do not write the body to an external file, but it still needs
    % to be "eaten" and thrown away; otherwise, it passes through
    % and latex tries to make the JS code into part of the document.
    % This uses the verbatim package.
    \comment
  }
}{
  % Post environment commands. Remember that the "execution" of an
  % environment is divided into two parts.
  \bool_if:NTF \l_figput_writebody_bool
  {
    % Stop writing the body since it's done.
    \xsim_file_write_stop:
  }
  {
    \endcomment
  }

  \bool_if:NTF \l_figput_nostatic_bool
  {
    % nostatic is set, so do nothing at all -- no figure in the static version.
  }
  {
    % nostatic is not set, so there is a figure.

    % And replace the current body with an external tikz file.
    \str_set_eq:NN \l_figput_tikzfile_str \l_figput_figname_str
    \str_put_right:Nn \l_figput_tikzfile_str { .tikz }

    % Whether to read the tikz file or display a "not available" message
    % depends on whether the tikz file exists and whether skip was set.
    \file_if_exist:VTF \l_figput_tikzfile_str
    {
      % Yes, tikz exists, although we may not read it if skip is set.
    }
    {
      % Tikz file doesn't exist, so we can't read it in any case.
      \bool_set_false:N \l_figput_readtikz_bool
    }
    
    \bool_if:NTF \l_figput_readtikz_bool
    {
      % Load the tikz file here.
      \file_input:V \l_figput_tikzfile_str
    } 
    {
      % Not loading the tikz file. Display a big empty box.
      % The problem here is that I need to specify the size of the
      % drawing, and tizk doesn't make that easy. It prefers to look at
      % the drawing and determine the size based on what was actually
      % drawn. It seems as though it is possible to *scale* a drawing to
      % take a particular size, although simply specifying the size is
      % hard. What does seem to work is one of the two following approaches.
      % (1) If you want to "trick" latex into allocating a given area
      % for the drawing, even if the drawing ends up going "outside the
      % lines," then you can say (within the tikzpicture environment)
      % \useasboundingbox (0pt,0pt) rectangle (100pt,200pt);
      % or whatever the dimensions are. Use this on the first line.
      % (2) This idea doesn't so much trick latex into using a particular
      % vbox/hbox for the drawing as limit the drawing to the rectangle
      % you want by clipping it. Here, as the first line of the drawing
      % you say
      % \clip (0,0) rectangle (100pt,200pt);
      % and the drawing will be limited to that region. I suspect (?)
      % that if the actual drawing is smaller than the clip region, the
      % latex will reduce the size of the figure accordiningly. So (1)
      % is probably the better solution.
      
      \begin{tikzpicture}
        % The opposite corner is given by (width,height), and the width
        % doesn't really matter since I assume that the figure spans the page.
        % BUG: Maybe I should change this to bp?
        % BUG: The right thing to do here is (?) to make the box match
        % the settings for \l_figput_innermargin_dim and outermargin, but
        % I think this requires that I know the page width.
        \useasboundingbox (0pt,0pt) rectangle (\textwidth,\l_figput_height_dim);
        
        % I don't understand exactly how the figure is placed on the
        % page, but this seems to work.
        \draw[dashed] (0pt,0pt) rectangle ( \textwidth,\l_figput_height_dim);
        
        % It looks like the text is drawn so that the bounding box is
        % centered at the given coordinates. Note how tikz lets you
        % divide with a simple "/2". Nice.
        %
        % Note also that, because this is being done with \ExplSyntaxOn,
        % the spaces in the text need to be \-spaces. Also, if I say
        % \node[draw], then a box is drawn around the text.
        \node at (\textwidth / 2,\l_figput_height_dim / 2) {The\ drawing\ is\ not\ available\ to\ load.};
      
        % For visual appeal, an extra enclosing rectangle.
        % BUG: I might want to check how tall the figure is and adjust
        % this box accordingly, or not show it at all if the box is very small.
        \draw (5pt,5pt) rectangle ( \textwidth - 5,\l_figput_height_dim - 5);

      \end{tikzpicture}
    }

    % End of figure inclusion -- nostatic is not set.
  }
  
  % See above. I originally had these lines near the start of the
  % process, but I had to put it here, as the very last thing, for the
  % values to be correct. Otherwise figures at the very end of a
  % page are assigned the page number prior to what it should be, and
  % they end up on the bottom of that page.
  %
  % Wow, I really don't understand this, but somehow, adding the \par
  % here serves to "quiet" another \par that would be inserted and
  % that I do not want. Without this, it seems like there is an extra
  % \par following the .tikz file when one is loaded. This solution
  % was provided as the answer to my question 643125 on tex.stackexchagne.
  % Apparently, it has something to do with \zsaveposy being a
  % "whatsit" command. I haven't unravelled this fully, but it somehow
  % explains what is going on.
  \par\zsaveposy { \l_figput_figname_str }
  \zlabel{ \l_figput_figname_str -pageno}

  % Note that group_end occurs here. It can't appear above, in the
  % pre-environment stuff because we still need access to some of the
  % variables here, post-environment.
  \group_end:  
}

% I wanted to define this with \cs_new, but lost patience with it.
% This works, but it's probably not "the right way."
\NewDocumentCommand{\figput_manargs}{ m m }
{
  % You need braces around #1 or you only get the first character.
  \str_set:Nn \l_figput_figname_str { #1 }
  \dim_set:Nn \l_figput_height_dim { #2 }
}

% This is to handle the optional arguments. It merely sets the
% relevant variables. 
\NewDocumentCommand{\figput_optargs} { }
{
  % Certain things are hard to do using a clist, so it's held as a seq
  % too. Again, no doubt there is a  better way, but this works.
  \seq_set_from_clist:NN \l_figput_optseq_seq { \l_figput_optclist_clist }
  
  % The first of the optional arguments might be a length. If it is, then
  % we have two lengths (dims), for the additional space above/below when
  % the figure is viewed interactively.
  % 
  % Note that one could make the call directly from the entire set of
  % optional arguments with
  %\figput_if_length:eTF { \clist_item:nn { #2 } { 1 } } {TRUE}{FALSE} \\
  % However (!) this only works if the function if_length is defined
  % with \prg_new_conditional:Nnn rather than
  % \prg_new_protected_conditional. I have no idea what the difference
  % is.
  %
  % Pull out the first of the optional arguments to make it easier to
  % work with. Note that by using a copy of the data in a seq variable,
  % I can pop things off without worrying about side-effects.
  \seq_pop_left:NN \l_figput_optseq_seq \l_figput_firstvalue_tl

  % If the first value parses out to be a dimension, then note that
  % value as the height above and get the height below too.
  \figput_if_length:VTF \l_figput_firstvalue_tl 
  {
    \dim_set:Nn \l_figput_htAbove_dim \l_figput_firstvalue_tl
    \bool_set_true:N \l_figput_hasintht_bool
    
    % Get the second value (reusing the "firstvalue" variable).
    \seq_pop_left:NN \l_figput_optseq_seq \l_figput_firstvalue_tl
    \dim_set:Nn \l_figput_htBelow_dim \l_figput_firstvalue_tl
  }
  {
    % If it's not a dimension, then ignore it for now.
  }

  % Check whether each possible boolean flag has been set.
  % BUG: I should check whether the user passed in something
  % spurious, and he might accidentally indicate the interactive
  % heights out of order. As things stand now, the user could pass in
  % "gibberish" as an option and it would be ignored. No harm done,
  % but it could confuse the user.
  \clist_if_in:NnTF \l_figput_optclist_clist { nostatic } 
  {
    \bool_set_true:N \l_figput_nostatic_bool
  }{}
  \clist_if_in:NnTF \l_figput_optclist_clist { done } 
  {
    \bool_set_true:N \l_figput_done_bool
  }{}
  \clist_if_in:NnTF \l_figput_optclist_clist { skip } 
  {
    % Note the test for whether \NeverSkip was called.
    \bool_if:NTF \l_figput_skipoff_bool
    {}{
      \bool_set_true:N \l_figput_skip_bool
    }
  }{}
}

% This was a "top-level" command, defined with
%\NewDocumentCommand{\iflengthTF}{mmm}
% but that's really not the way to do this.
%
% Instead, we define a function "in the module" (figput), plus a
% "variant." See p. 32 of the interface3 document.  For the initial
% function (not the variant), see p. 62. I can't say that I really
% understand what's going on here, other than a bit of copy and paste
% and the help from stack overflow does it.
\prg_new_protected_conditional:Nnn \figput_if_length:n { T, F, TF }
 {
  \regex_match:nnTF
  % Note that I only allow positive values.
   { \A [+]? ((\d+(\.\d*)?)|(\.\d+)) \s* (pt|pc|in|bp|cm|mm|dd|cc|sp|ex|em) \Z} 
   { #1 } % test string
   { \prg_return_true: }
   { \prg_return_false: }
 }
 
\prg_generate_conditional_variant:Nnn \figput_if_length:n { V } { T, F, TF }


\ExplSyntaxOff
