src/format.h

Sun, 30 Mar 2014 21:53:20 +0300

author
Teemu Piippo <crimsondusk64@gmail.com>
date
Sun, 30 Mar 2014 21:53:20 +0300
changeset 120
5ea0faefa82a
parent 119
bdf8d46c145f
child 124
a7b769a0e537
permissions
-rw-r--r--

- updated .gitignore

/*
	Copyright 2012-2014 Santeri Piippo
	All rights reserved.

	Redistribution and use in source and binary forms, with or without
	modification, are permitted provided that the following conditions
	are met:

	1. Redistributions of source code must retain the above copyright
	   notice, this list of conditions and the following disclaimer.
	2. Redistributions in binary form must reproduce the above copyright
	   notice, this list of conditions and the following disclaimer in the
	   documentation and/or other materials provided with the distribution.
	3. The name of the author may not be used to endorse or promote products
	   derived from this software without specific prior written permission.

	THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
	IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
	OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
	IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
	INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
	NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
	DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
	THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
	(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
	THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/

#ifndef BOTC_FORMAT_H
#define BOTC_FORMAT_H

#include "string.h"
#include "list.h"

class FormatArgument
{
	public:
		FormatArgument (const String& a) : m_text (a) {}
		FormatArgument (char a) : m_text (a) {}
		FormatArgument (int a) : m_text (String::fromNumber (a)) {}
		FormatArgument (long a) : m_text (String::fromNumber (a)) {}
		FormatArgument (const char* a) : m_text (a) {}

		FormatArgument (void* a)
		{
			m_text.sprintf ("%p", a);
		}

		FormatArgument (const void* a)
		{
			m_text.sprintf ("%p", a);
		}

		template<class T> FormatArgument (const List<T>& list)
		{
			if (list.isEmpty())
			{
				m_text = "{}";
				return;
			}

			m_text = "{ ";

			for (const T& a : list)
			{
				if (&a != &list[0])
					m_text += ", ";

				m_text += FormatArgument (a).text();
			}

			m_text += " }";
		}

		inline const String& text() const
		{
			return m_text;
		}

	private:
		String m_text;
};

#ifndef IN_IDE_PARSER
# ifdef DEBUG
#  define devf(...) PrintTo (stderr, __VA_ARGS__)
#  define dvalof( A ) PrintTo (stderr, "value of '%1' = %2\n", #A, A)
# else
#  define devf(...)
#  define dvalof( A )
# endif // DEBUG
#else
// print something in debug builds
void devf (void, ...);

// print the value of @a
void dvalof (void a);
#endif // IN_IDE_PARSER


/**
 * Formats the given string with the given args.
 *
 * @param fmtstr Formatter string to process.
 * @param args Args to format with the string.
 * @see format()
 */
String formatArgs (const String& fmtstr, const std::vector<String>& args);

/**
 * Expands the given arguments into a vector of strings.
 *
 * @param data Where to insert the strings.
 * @param arg First argument to process
 * @param rest... Rest of the arguments.
 */
template<typename T, typename... RestTypes>
void expandFormatArguments (std::vector<String>& data, const T& arg, const RestTypes& ... rest)
{
	data.push_back (FormatArgument (arg).text());
	expandFormatArguments (data, rest...);
}

/**
 * This is an overload of @c ExpandFormatArguments for end-of-args support.
 */
static void expandFormatArguments (std::vector<String>& data) __attribute__ ( (unused));
static void expandFormatArguments (std::vector<String>& data)
{
	(void) data;
}

/**
 * Formats the given formatter string and args and returns the string.
 * This is essentially a modernized sprintf.
 *
 * Args in the format string take the form %n where n is a digit. The argument
 * will be expanded to the nth argument passed. This is essentially Qt's
 * QString::arg() syntax. Note: %0 is invalid.
 *
 * Arguments can be passed a modifier which takes the form of a character
 * just before the digit. Currently supported modifiers are s, d and x.
 *
 * - s: The argument will expand into "s" if it would've not expanded into "1"
 *      otherwise. If it would have expanded into "1" it will expand into an
 *      empty string.
 *
 * - d: The argument expands into the numeric form of the first character of
 *      its previous expansion. Use this to get numeric forms of @c char
 *      arguments.
 *
 * - x: The numeric argument will be represented in hexadecimal notation. This
 *      will work if the argument is a string representing a number. If the
 *      argument did not expand into a number in the first place, 0 is used
 *      (and 0x0 is printed).
 *
 * @param fmtstr Formatter string to process
 * @param raw_args Arguments for the formatter string.
 * @return the formatted string.
 * @see Print
 * @see PrintTo
 */
template<typename... argtypes>
String format (const String& fmtstr, const argtypes&... raw_args)
{
	std::vector<String> args;
	expandFormatArguments (args, raw_args...);
	assert (args.size() == sizeof... (raw_args));
	return formatArgs (fmtstr, args);
}

/**
 * This is an overload of @c format where no arguments are supplied.
 * @return the formatter string as-is.
 */
static String format (const String& fmtstr) __attribute__ ( (unused));
static String format (const String& fmtstr)
{
	return fmtstr;
}

/**
 * Processes the given formatter string using @c format and prints it to the
 * specified file pointer.
 *
 * @param fp File pointer to print the formatted string to
 * @param fmtstr Formatter string for @c format
 * @param args Arguments for @c fmtstr
 */
template<typename... argtypes>
void printTo (FILE* fp, const String& fmtstr, const argtypes&... args)
{
	fprintf (fp, "%s", format (fmtstr, args...).c_str());
}

/**
 * Processes the given formatter string using @c format and prints the result to
 * @c stdout.
 *
 * @param fmtstr Formatter string for @c format
 * @param args Arguments for @c fmtstr
 */
template<typename... argtypes>
void print (const String& fmtstr, const argtypes&... args)
{
	printTo (stdout, fmtstr, args...);
}

/**
 * Throws an std::runtime_error with the processed formatted string. The program
 * execution terminates after a call to this function as the exception is first
 * caught in @c main which prints the error to stderr and then exits.
 *
 * @param fmtstr The formatter string of the error.
 * @param args The args to the formatter string.
 * @see Format
 */
template<typename... argtypes>
void error (const String& fmtstr, const argtypes&... args)
{
	error (format (fmtstr, args...));
}

/**
 * An overload of @c Error with no string formatting in between.
 *
 * @param msg The error message.
 */
void error (String msg);

#endif // BOTC_FORMAT_H

mercurial