#ifndef INCLUDED_BOBCAT_PATTERN_
#define INCLUDED_BOBCAT_PATTERN_

#include <string>
#include <utility>              // for pair<>
#include <regex.h>
#include <iostream>

#include <bobcat/exception>

//  match() throws an Errno when either the construction (i.e.,
//  compilation of the pattern) or the match failed.
//
//  The 0-index for position() or operator[] indicates the matched text,
//  other indices indicate subsequent subexpressions
//
//  Patterns may use:
//      \b - indicating a word-boundary
//      \d - indicating a digit
//      \D - indicating no digit
//      \s - indicating a white-space ([:space:]) char
//      \S - indicating no white-space ([:space:]) char
//
//  Pattern strings:
//
//      ------------------------------------------------------------
//      Required pattern        Provide Pattern()   Use as argument:
//                              internally with:
//      ------------------------------------------------------------
//      \\                      \\\\                \\\\\\\\       |
//      \d                      \d                  \\d            |
//      ------------------------------------------------------------

namespace FBB
{

class Pattern
{
    using conversion = std::pair<char const *, char const *>;

    static conversion       s_patmod[];
    static size_t           s_sizeofPatmod;

    struct Regex
    {
        size_t              d_referenceCount;
        regex_t             d_regex;

        static std::string  s_converted;

        Regex(std::string pattern, int options);
        ~Regex();                                       // destructor.f

        private:
            Regex(Regex const &other) = delete;
            Regex &operator=(Regex const &other) = delete;
    };

    Regex                  *d_regex = 0;
    regmatch_t             *d_subExpression = 0;
    size_t                  d_nSub;
    size_t                  d_beyondLast;
    std::string             d_text;
    int                     d_matchOptions;

    public: 
        using Position = 
                std::pair<std::string::size_type, std::string::size_type>;

        //  define a pattern using a case-flag and number of
        //  ()-subexpressions. Compilation flags:
        //
        //  REG_EXTENDED
        //      Use POSIX Extended Regular Expression  syntax  when
        //      interpreting  regex.  If not set, POSIX Basic Regu-
        //      lar Expression syntax is used.
        //
        //  REG_NOSUB
        //      Support for substring addressing of matches is  not
        //      required.   The  nmatch  and  pmatch  parameters to
        //      regexec are ignored if the pattern buffer  supplied
        //      was compiled with this flag set.
        //
        //  REG_NEWLINE
        //      Match-any-character  operators  don't  match a newline.
        //
        //      A non-matching list ([^...])  not containing a newline
        //      does not match a newline.
        //
        //      Match-beginning-of-line  operator  (^)  matches the
        //      empty string immediately after a  newline,  regardless
        //      of  whether  eflags,  the  execution flags of
        //      regexec, contains REG_NOTBOL.
        //
        //      Match-end-of-line operator ($)  matches  the  empty
        //      string  immediately before a newline, regardless of
        //      whether eflags contains REG_NOTEOL.

        Pattern();                                      // 1

        Pattern(Pattern const &other);                  // 2.f

        explicit Pattern(std::string const &pattern,    // 3.cc
                        bool caseSensitive = true,
                        size_t nSub = 10,
                        int options = REG_EXTENDED | REG_NEWLINE);
        Pattern(Pattern &&tmp);                         // 4
        ~Pattern();

        Pattern &operator=(Pattern const &other);
        Pattern &operator=(Pattern &&tmp);

        Pattern &operator<<(int matchOption);
        bool operator<<(std::string const &text);

        void setPattern(std::string const &pattern,
                        bool caseSensitive = true,
                        size_t nSub = 10,
                        int options = REG_EXTENDED | REG_NEWLINE);

        //  match a string with a pattern. true: string matched
        //  options could be:
        //
        //  REG_NOTBOL
        //      The  match-beginning-of-line  operator always fails
        //      to match (but see the compilation flag  REG_NEWLINE
        //      above)  This  flag  may be used when different portions
        //      of a string are passed  to  regexec  and  the
        //      beginning  of  the string should not be interpreted
        //      as the beginning of the line.
        //
        //  REG_NOTEOL
        //      The  match-end-of-line  operator  always  fails  to
        //      match  (but  see  the  compilation flag REG_NEWLINE)

        void match(std::string const &text, int options = 0);

        std::string before() const;  // text before the matched text
        std::string matched() const;   // the matched text              .f
        std::string beyond() const;  // text beyond the matched text

            // (0) is complete matching part. Remaining are subexpressions.
            // (npos, npos) is returned if index exceeds available indices
            // (which may be 0)

            // position of subexpression
        Position position(size_t index) const;

            // subexpression itself
        std::string operator[](size_t index) const;
        size_t end() const;           // index beyond the last available    .f

        std::string const &pattern() const; // compiled pattern

        void swap(Pattern &other);

    private:
        void newRegex(std::string const &pattern, int options);         // .f
        void destroy();
        void copy(Pattern const &other);
};

#include "pattern2.f"
#include "matched.f"
#include "pattern.f"
#include "end.f"

} // FBB

#endif
