/* -*-c++-*- OpenSceneGraph - Copyright (C) 2008 Robert Osfield 
 *
 * This library is open source and may be redistributed and/or modified under  
 * the terms of the OpenSceneGraph Public License (OSGPL) version 0.0 or 
 * (at your option) any later version.  The full license is in LICENSE file
 * included with this distribution, and on the openscenegraph.org website.
 * 
 * This library 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 
 * OpenSceneGraph Public License for more details.
*/

#include <osgDB/ConvertUTF>
#include <osg/Notify>

#include <string.h>
#include <wchar.h>

#if defined(WIN32) && !defined(__CYGWIN__)
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#endif

namespace osgDB
{

std::string convertUTF16toUTF8(const std::wstring& s){return convertUTF16toUTF8(s.c_str(), s.length());}
std::string convertUTF16toUTF8(const wchar_t* s){return convertUTF16toUTF8(s, wcslen(s));}

std::wstring convertUTF8toUTF16(const std::string& s){return convertUTF8toUTF16(s.c_str(), s.length());}
std::wstring convertUTF8toUTF16(const char* s){return convertUTF8toUTF16(s, strlen(s));}

std::string convertMultiByteToUTF8(const std::string& s){return convertMultiByteToUTF8(s.c_str(), s.length());}
std::string convertMultiByteToUTF8(const char* s){return convertMultiByteToUTF8(s, strlen(s));}

std::string convertUTF8toMultiByte(const std::string& s){return convertUTF8toMultiByte(s.c_str(), s.length());}
std::string convertUTF8toMultiByte(const char* s){return convertUTF8toMultiByte(s, strlen(s));}

std::string convertUTF16toUTF8(const wchar_t* source, unsigned sourceLength)
{
#if defined(WIN32) && !defined(__CYGWIN__)
    if (sourceLength == 0)
    {
        return std::string();
    }

    int destLen = WideCharToMultiByte(CP_UTF8, 0, source, sourceLength, 0, 0, 0, 0);
    if (destLen <= 0)
    {
        osg::notify(osg::WARN) << "Cannot convert UTF-16 string to UTF-8." << std::endl;
        return std::string();
    }

    std::string sDest(destLen, '\0');
    destLen = WideCharToMultiByte(CP_UTF8, 0, source, sourceLength, &sDest[0], destLen, 0, 0);

    if (destLen <= 0)
    {
        osg::notify(osg::WARN) << "Cannot convert UTF-16 string to UTF-8." << std::endl;
        return std::string();
    }

    return sDest;
#else
    //TODO: Implement for other platforms
    osg::notify(osg::WARN) << "ConvertUTF16toUTF8 not implemented." << std::endl;
    return std::string();
#endif
}

std::wstring convertUTF8toUTF16(const char* source, unsigned sourceLength)
{
#if defined(WIN32) && !defined(__CYGWIN__)
    if (sourceLength == 0)
    {
        return std::wstring();
    }

    int destLen = MultiByteToWideChar(CP_UTF8, 0, source, sourceLength, 0, 0);
    if (destLen <= 0)
    {
        osg::notify(osg::WARN) << "Cannot convert UTF-8 string to UTF-16." << std::endl;
        return std::wstring();
    }

    std::wstring sDest(destLen, L'\0');
    destLen = MultiByteToWideChar(CP_UTF8, 0, source, sourceLength, &sDest[0], destLen);

    if (destLen <= 0)
    {
        osg::notify(osg::WARN) << "Cannot convert UTF-8 string to UTF-16." << std::endl;
        return std::wstring();
    }

    return sDest;
#else
    //TODO: Implement for other platforms
    osg::notify(osg::WARN) << "ConvertUTF8toUTF16 not implemented." << std::endl;
    return std::wstring();
#endif
}

std::string convertMultiByteToUTF8(const char* source, unsigned sourceLength)
{
#if defined(WIN32) && !defined(__CYGWIN__)
	if (sourceLength == 0)
	{
		return std::string();
	}

	int utf16Length = MultiByteToWideChar(CP_ACP, 0, source, sourceLength, 0, 0);
	if (utf16Length <= 0)
	{
		osg::notify(osg::WARN) << "Cannot convert multi-byte string to UTF-8." << std::endl;
		return std::string();
	}

	std::wstring sUTF16(utf16Length, L'\0');
	utf16Length = MultiByteToWideChar(CP_ACP, 0, source, sourceLength, &sUTF16[0], utf16Length);
	if (utf16Length <= 0)
	{
		osg::notify(osg::WARN) << "Cannot convert multi-byte string to UTF-8." << std::endl;
		return std::string();
	}

	return convertUTF16toUTF8(sUTF16);
#else
	return source;
#endif
}

std::string convertUTF8toMultiByte(const char* source, unsigned sourceLength)
{
#if defined(WIN32) && !defined(__CYGWIN__)
	if (sourceLength == 0)
	{
		return std::string();
	}

	std::wstring utf16 = convertUTF8toUTF16(source, sourceLength);
	sourceLength = utf16.length();

	int destLen = WideCharToMultiByte(CP_ACP, 0, utf16.c_str(), sourceLength, 0, 0, 0, 0);
	if (destLen <= 0)
	{
		osg::notify(osg::WARN) << "Cannot convert multi-byte string to UTF-8." << std::endl;
		return std::string();
	}

	std::string sDest(destLen, '\0');
	destLen = WideCharToMultiByte(CP_ACP, 0, utf16.c_str(), sourceLength, 0, 0, 0, 0);
	if (destLen <= 0)
	{
		osg::notify(osg::WARN) << "Cannot convert multi-byte string to UTF-8." << std::endl;
		return std::string();
	}

	return sDest;
#else
	return source;
#endif
}

}

