// -*- coding: utf-8 -*- // Copyright (C) 2012, 2013 Laboratoire de Recherche et Développement // de l'Epita (LRDE). // // This file is part of Spot, a model checking library. // // Spot is free software; you can redistribute it and/or modify it // under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 3 of the License, or // (at your option) any later version. // // Spot is distributed in the hope that it will be useful, but WITHOUT // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public // License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . #ifndef SPOT_MISC_FORMATER_HH #define SPOT_MISC_FORMATER_HH #include "common.hh" #include #include #include namespace spot { class printable { public: virtual ~printable() { } virtual void print(std::ostream&, const char*) const = 0; }; template class printable_value: public printable { protected: T val_; public: const T& val() const { return val_; } T& val() { return val_; } operator const T&() const { return val(); } operator T&() { return val(); } printable_value& operator=(const T& new_val) { val_ = new_val; return *this; } virtual void print(std::ostream& os, const char*) const { os << val_; } }; /// The default callback simply writes "%c". class printable_id: public printable { public: virtual void print(std::ostream& os, const char* x) const { os << '%' << *x; } }; /// Called by default for "%%" and "%\0". class printable_percent: public printable { public: virtual void print(std::ostream& os, const char*) const { os << '%'; } }; class SPOT_API formater { printable_id id; printable_percent percent; public: formater() : has_(256), call_(256, &id) { call_['%'] = call_[0] = &percent; } virtual ~formater() { } /// \brief Scan the %-sequences occuring in \a fmt. /// /// Set has['c'] for each %c in \a fmt. \a has must /// be 256 wide. /// @{ void scan(const char* fmt, std::vector& has) const; void scan(const std::string& fmt, std::vector& has) const { scan(fmt.c_str(), has); } /// @} /// Collect the %-sequences occurring in \a fmt. /// @{ void prime(const char* fmt); void prime(const std::string& fmt) { prime(fmt.c_str()); } /// @} /// Whether %c occurred in the primed formats. bool has(char c) const { return has_[c]; } /// Declare a callback function for %c. void declare(char c, const printable* f) { call_[c] = f; } /// Remember where to output any string. void set_output(std::ostream& output) { output_ = &output; } /// Expand the %-sequences in \a fmt, write the result on \a output_. std::ostream& format(const char* fmt); /// Expand the %-sequences in \a fmt, write the result on \a output. std::ostream& format(std::ostream& output, const char* fmt) { set_output(output); return format(fmt); } /// Expand the %-sequences in \a fmt, write the result on \a output_. std::ostream& format(const std::string& fmt) { return format(fmt.c_str()); } /// Expand the %-sequences in \a fmt, write the result on \a output. std::ostream& format(std::ostream& output, const std::string& fmt) { return format(output, fmt.c_str()); } private: std::vector has_; std::vector call_; protected: std::ostream* output_; }; } #endif // SPOT_MISC_FORMATER_HH