#ifndef UTIL_H
#define UTIL_H

#include <algorithm>
#include <cstring>
#include <memory>
#include <string>
#include <SWI-Prolog.h>
#include <windows.h>

/* Buf is a span-like structure of a buffer and its size. */
template <typename T>
struct Buf
{
	T* data;
	size_t size;
	Buf(T* data, size_t size) : data(data), size(size) {}
	Buf(std::basic_string<T>& s) : data(s.data()), size(s.capacity()) {}
	template <size_t N> Buf(T (&data)[N]) : data(data), size(N) {}
	operator T*() { return data; }
	T& operator *() { return *data; }
	T& operator [](size_t i) { return data[i]; }
};

/* Format wide string. */
template<typename... T>
inline int Swprintf(Buf<wchar_t> buf, const wchar_t* const fmt, T... xs)
{
	return _snwprintf_s(buf, buf.size, _TRUNCATE, fmt, xs...);
}

/* Format static narrow string. */
template<typename... T>
inline int Sprintf(Buf<char> buf, const char* const fmt, T... xs)
{
	return _snprintf_s(buf, buf.size, _TRUNCATE, fmt, xs...);
}

/* Copy to static wide string buffer. */
inline wchar_t* Wcscpy(Buf<wchar_t> dst, const wchar_t* const src)
{
	const size_t len = std::min(dst.size, wcslen(src)+1);
	memcpy(dst, src, len*sizeof(wchar_t));
	dst[len-1] = 0;
	return dst;
}

/* Copy to static narrow string buffer. */
inline char* Strcpy(Buf<char> dst, const char* const src)
{
	const size_t len = std::min(dst.size, strlen(src)+1);
	memcpy(dst, src, len);
	dst[len-1] = 0;
	return dst;
}

inline int Cmp(const int a, const int b)
{
	if (a == b) return 0;
	if (a > b) return 1;
	return -1;
}

#define CAT(a, b) a##b
#define APPLY(a, ...) a(__VA_ARGS__)
#define UNUSED APPLY(CAT, unused_, __COUNTER__)

#endif