/*
 *	@file 	ejsString.c
 *	@brief 	EJScript string class
 */
/********************************* Copyright **********************************/
/*
 *	@copy	default
 *	
 *	Copyright (c) Mbedthis Software LLC, 2003-2006. All Rights Reserved.
 *	Copyright (c) Michael O'Brien, 1994-1995. All Rights Reserved.
 *	
 *	This software is distributed under commercial and open source licenses.
 *	You may use the GPL open source license described below or you may acquire 
 *	a commercial license from Mbedthis Software. You agree to be fully bound 
 *	by the terms of either license. Consult the LICENSE.TXT distributed with 
 *	this software for full details.
 *	
 *	This software is open source; 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 2 of the License, or (at your 
 *	option) any later version. See the GNU General Public License for more 
 *	details at: http://www.mbedthis.com/downloads/gplLicense.html
 *	
 *	This program is distributed WITHOUT ANY WARRANTY; without even the 
 *	implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. 
 *	
 *	This GPL license does NOT permit incorporating this software into 
 *	proprietary programs. If you are unable to comply with the GPL, you must
 *	acquire a commercial license to use this software. Commercial licenses 
 *	for this software and support services are available from Mbedthis 
 *	Software at http://www.mbedthis.com 
 *	
 *	@end
 */
/********************************** Includes **********************************/

#include	"ejs.h"

#if BLD_FEATURE_EJS
/******************************************************************************/
/*********************************** Constructors *****************************/
/******************************************************************************/
/*
 *	String constructor. 
 */

int ejsStringConstructor(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	char	*str;

	if (argc == 0) {
		ejsSetReturnValueToString(ejs, "");

	} else if (argc == 1) {
		/* MOB -- rc */
		str = ejsVarToString(ejs, argv[0]);
		ejsSetReturnValueToString(ejs, str);

	} else {
		ejsArgError(ejs, "usage: String([var])");
		return -1;
	}

	return 0;
}

/******************************************************************************/
/******************************** Visible Methods *****************************/
/******************************************************************************/
/*
 *	Return a string containing the character at a given index
 *
 *	String string.charAt(Number)
 */

static int charAt(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	EjsNum		num;
	char		buf[2];

	if (argc != 1) {
		ejsArgError(ejs, "usage: charAt(integer)");
		return -1;
	}

	num = ejsVarToNumber(argv[0]);
	if (num < 0 || num >= thisObj->length) {
		ejsError(ejs, EJS_RANGE_ERROR, "Bad index");
		return -1;
	}

	mprAssert(ejsVarIsString(thisObj));

	buf[0] = argv[0]->string[num];
	buf[1] = '\0';
	ejsSetReturnValueToString(ejs, buf);

	return 0;
}

/******************************************************************************/
/*
 *	Return an integer containing the character at a given index
 *
 *	Number string.charCodeAt(Number)
 */

static EjsNum charCodeAt(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	EjsNum			num;

	if (argc != 1) {
		ejsArgError(ejs, "usage: charCodeAt(integer)");
		return -1;
	}

	num = ejsVarToNumber(argv[0]);
	if (num < 0 || num >= thisObj->length) {
		ejsError(ejs, EJS_RANGE_ERROR, "Bad index");
		return -1;
	}
	ejsSetReturnValueToNumber(ejs, (EjsNum) argv[0]->string[num]);

	return 0;
}

/******************************************************************************/
/*
 *	Catenate
 *
 *	String string.catenate(var, ...)
 */

static int concat(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	int				i;

	if (argc == 0) {
		ejsArgError(ejs, "usage: concat(String, ...)");
		return -1;
	}

	mprAssert(ejsVarIsString(thisObj));

	for (i = 0; i < argc; i++) {
		if (ejsStrcat(ejs, thisObj, argv[i]) < 0) {
			ejsMemoryError(ejs);
			return -1;
		}
	}
	ejsSetReturnValue(ejs, thisObj);
	return 0;
}

/******************************************************************************/
/*
 *	Return the position of the first occurance of a substring
 *
 *	Number string.indexOf(String subString [, Number start])
 */

static int indexOf(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	char	*pat, *s1, *s2, *origin;
	int		start, i;

	if (argc == 0 || argc > 2) {
		ejsArgError(ejs, "usage: indexOf(String [, Number])");
		return -1;
	}

	pat = ejsVarToString(ejs, argv[0]);

	if (argc == 2) {
		start = ejsVarToNumber(argv[1]);
		if (start > thisObj->length) {
			start = thisObj->length;
		}
	} else {
		start = 0;
	}

	i = start;
	for (origin = &thisObj->string[i]; i < thisObj->length; i++, origin++) {
		s1 = origin;
		for (s2 = pat; *s1 && *s2; s1++, s2++) {
			if (*s1 != *s2) {
				break;
			}
		}
		if (*s2 == '\0') {
			ejsSetReturnValueToNumber(ejs, (EjsNum) (origin - thisObj->string));
		}
	}

	ejsSetReturnValueToNumber(ejs, (EjsNum) -1);
	return 0;
}

/******************************************************************************/
/*
 *	Return the position of the last occurance of a substring
 *
 *	Number string.lastIndexOf(String subString [, Number start])
 */

static int lastIndexOf(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	char	*pat, *s1, *s2, *origin;
	int		start;

	if (argc == 0 || argc > 2) {
		ejsArgError(ejs, "usage: indexOf(String [, Number])");
		return -1;
	}

	pat = ejsVarToString(ejs, argv[0]);

	if (argc == 2) {
		start = ejsVarToNumber(argv[1]);
		if (start > thisObj->length) {
			start = thisObj->length;
		}
	} else {
		start = 0;
	}

	origin = &thisObj->string[thisObj->length - 1];
	for (; origin >= &thisObj->string[start]; origin--) {

		s1 = origin;
		for (s2 = pat; *s1 && *s2; s1++, s2++) {
			if (*s1 != *s2) {
				break;
			}
		}
		if (*s2 == '\0') {
			ejsSetReturnValueToNumber(ejs, (EjsNum) (origin - thisObj->string));
		}
	}

	ejsSetReturnValueToNumber(ejs, (EjsNum) -1);
	return 0;
}

/******************************************************************************/
/*
 *	Return a substring
 *
 *	Number string.slice(Number start, Number end)
 */

static int slice(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	EjsNum		start, end;

	if (argc != 2) {
		ejsArgError(ejs, "usage: slice(Number, Number)");
		return -1;
	}

	start = ejsVarToNumber(argv[0]);
	end = ejsVarToNumber(argv[1]);
	if (start < 0 || start >= thisObj->length) {
		ejsError(ejs, EJS_RANGE_ERROR, "Bad start index");
		return-1;
	}
	if (end < 0 || end >= thisObj->length) {
		ejsError(ejs, EJS_RANGE_ERROR, "Bad end index");
		return -1;
	}

	mprAssert(ejsVarIsString(thisObj));

	ejsSetReturnValueToBinaryString(ejs, (uchar*) &thisObj->string[start],
		end - start);

	return 0;
}

/******************************************************************************/
/*
 *	Split a string
 *
 *	Number string.split(String delimiter [, Number limit])
 */

static int split(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
{
	EjsVar		*array, *vp;
	char		*delim, *last, *cp;
	int			len, limit, alloc;

	if (argc == 0 || argc > 2) {
		ejsArgError(ejs, "usage: split(String [, Number])");
		return -1;
	}

	delim = ejsVarToStringEx(ejs, argv[0], &alloc);

	limit = ejsVarToNumber(argv[1]);

	array = ejsCreateArray(ejs, 0);

	len = strlen(delim);

	last = thisObj->string;
	for (cp = last; *cp; cp++) {
		if (*cp == *delim && strncmp(cp, delim, len) == 0) {
			if (cp > last) {
				vp = ejsCreateBinaryStringVar(ejs, (uchar*) last, (cp - last));
				ejsAddArrayElt(ejs, array, vp, EJS_SHALLOW_COPY);
				ejsFreeVar(ejs, vp);
			}
		}
	}

	ejsSetReturnValue(ejs, array);
	ejsFreeVar(ejs, array);

	if (alloc) {
		mprFree(delim);
	}

	return 0;
}

/******************************************************************************/
/*
 *	Create the object class
 */

int ejsDefineStringClass(Ejs *ejs)
{
	EjsVar		*sc;

	sc = ejsDefineClass(ejs, "String", "Object", ejsStringConstructor);
	if (sc == 0) {
		return MPR_ERR_CANT_INITIALIZE;
	}

	ejsDefineCMethod(ejs, sc, "charAt", charAt, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "charCodeAt", charCodeAt, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "concat", concat, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "indexOf", indexOf, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "lastIndexOf", lastIndexOf, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "slice", slice, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "split", split, EJS_NO_LOCAL);
#if UNUSED
	ejsDefineCMethod(ejs, sc, "match", match, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "replace", replace, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "search", search, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "substring", substring, EJS_NO_LOCAL);
	//	MOB bad name
	ejsDefineCMethod(ejs, sc, "substr", substr, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "toLowerCase", toLowerCase, EJS_NO_LOCAL);
	ejsDefineCMethod(ejs, sc, "toUpperCase", toUpperCase, EJS_NO_LOCAL);

	//	Static method
	ejsDefineCMethod(ejs, sc, "fromCharCode", fromCharCode, 0, EJS_NO_LOCAL);
#endif

	if (ejsObjHasErrors(sc)) {
		ejsFreeVar(ejs, sc);
		return MPR_ERR_CANT_CREATE;
	}
	return 0;
}

/******************************************************************************/
#endif /* BLD_FEATURE_EJS */

/*
 * Local variables:
 * tab-width: 4
 * c-basic-offset: 4
 * End:
 * vim:tw=78
 * vim600: sw=4 ts=4 fdm=marker
 * vim<600: sw=4 ts=4
 */