/* Copyright (C) 2003-2015 LiveCode Ltd. This file is part of LiveCode. LiveCode is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License v3 as published by the Free Software Foundation. LiveCode 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 LiveCode. If not see . */ #include "prefix.h" #include "globdefs.h" #include "objdefs.h" #include "parsedef.h" #include "filedefs.h" #include "globals.h" #include "handler.h" #include "scriptpt.h" #include "newobj.h" #include "chunk.h" #include "param.h" #include "mcerror.h" #include "property.h" #include "object.h" #include "button.h" #include "field.h" #include "image.h" #include "card.h" #include "eps.h" #include "graphic.h" #include "group.h" #include "scrolbar.h" #include "player.h" #include "aclip.h" #include "vclip.h" #include "stack.h" #include "dispatch.h" #include "stacklst.h" #include "sellst.h" #include "undolst.h" #include "util.h" #include "date.h" #include "printer.h" #include "debug.h" #include "cmds.h" #include "mode.h" #include "osspec.h" #include "hndlrlst.h" #include "securemode.h" #include "stacksecurity.h" #include "license.h" #ifdef _SERVER #include "srvscript.h" #endif #include "exec.h" MCChoose::~MCChoose() { delete etool; } Parse_stat MCChoose::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); sp.skip_token(SP_FACTOR, TT_THE); if (sp.next(type) != PS_NORMAL) { MCperror->add(PE_CHOOSE_NOTOKEN, sp); return PS_ERROR; } if (sp.lookup(SP_TOOL, te) != PS_NORMAL) { sp.backup(); if (sp.parseexp(False, True, &etool) != PS_NORMAL) { MCperror->add(PE_CHOOSE_BADEXP, sp); return PS_ERROR; } sp.skip_token(SP_TOOL, TT_END); return PS_NORMAL; } else { littool = (Tool)te->which; while (sp.skip_token(SP_TOOL, TT_TOOL) == PS_NORMAL) ; sp.skip_token(SP_TOOL, TT_END); } return PS_NORMAL; } void MCChoose::exec_ctxt(MCExecContext &ctxt) { MCAutoStringRef t_string; if (!ctxt . EvalOptionalExprAsStringRef(etool, kMCEmptyString, EE_CHOOSE_BADEXP, &t_string)) return; MCInterfaceExecChooseTool(ctxt, *t_string, littool); } MCConvert::~MCConvert() { delete container; delete source; } Parse_stat MCConvert::parse(MCScriptPoint &sp) { initpoint(sp); MCerrorlock++; container = new (nothrow) MCChunk(True); MCScriptPoint tsp(sp); if (container->parse(sp, False) != PS_NORMAL) { sp = tsp; MCerrorlock--; delete container; container = NULL; if (sp.parseexp(False, True, &source) != PS_NORMAL) { MCperror->add (PE_CONVERT_NOCONTAINER, sp); return PS_ERROR; } } else MCerrorlock--; if (sp.skip_token(SP_FACTOR, TT_FROM) == PS_NORMAL) { if (parsedtformat(sp, fform, fsform) != PS_NORMAL) return PS_ERROR; } if (sp.skip_token(SP_FACTOR, TT_TO) != PS_NORMAL) { MCperror->add (PE_CONVERT_NOTO, sp); return PS_ERROR; } if (parsedtformat(sp, pform, sform) != PS_NORMAL) return PS_ERROR; return PS_NORMAL; } Parse_stat MCConvert::parsedtformat(MCScriptPoint &sp, Convert_form &firstform, Convert_form &secondform) { const LT *te; Symbol_type type; Boolean needformat = True; Convert_form localeform = CF_UNDEFINED; while (True) { if (sp.next(type) != PS_NORMAL) { if (needformat) { MCperror->add (PE_CONVERT_NOFORMAT, sp); return PS_ERROR; } else return PS_NORMAL; } if (sp.lookup(SP_CONVERT, te) != PS_NORMAL) { if (needformat) { MCperror->add (PE_CONVERT_NOTFORMAT, sp); return PS_ERROR; } else { sp.backup(); break; } } switch (te->which) { case CF_ABBREVIATED: case CF_SHORT: case CF_LONG: case CF_INTERNET: if (firstform == CF_UNDEFINED) firstform = (Convert_form)te->which; else secondform = (Convert_form)te->which; break; case CF_SYSTEM: case CF_ENGLISH: localeform = (Convert_form)te->which; break; case CF_DATE: case CF_TIME: if (firstform == CF_UNDEFINED) firstform = (Convert_form)(te->which + 1 + localeform); else if (firstform > CF_INTERNET) { if (secondform == CF_UNDEFINED) secondform = (Convert_form)(te->which + 1 + localeform); else { uint2 dummy = secondform; dummy += te->which + localeform; secondform = (Convert_form)dummy; } } else { uint2 dummy = firstform; dummy += te->which + localeform; firstform = (Convert_form)dummy; } needformat = False; break; default: firstform = (Convert_form)te->which; return PS_NORMAL; } if (sp.skip_token(SP_FACTOR, TT_BINOP, O_AND) == PS_NORMAL) { if (needformat) { MCperror->add (PE_CONVERT_BADAND, sp); return PS_ERROR; } else needformat = True; } else if (!needformat) break; } return PS_NORMAL; } void MCConvert::exec_ctxt(MCExecContext& ctxt) { MCAutoStringRef t_input; MCAutoStringRef t_output; if (container != NULL) { if (!ctxt . EvalExprAsStringRef(container, EE_CONVERT_CANTGET, &t_input)) return; } else { if (!ctxt . EvalExprAsStringRef(source, EE_CONVERT_CANTGET, &t_input)) return; } if (container == NULL) MCDateTimeExecConvertIntoIt(ctxt, *t_input, fform, fsform, pform, sform); else { MCDateTimeExecConvert(ctxt, *t_input, fform, fsform, pform, sform, &t_output); container -> set(ctxt, PT_INTO, *t_output); if (ctxt . HasError()) ctxt . LegacyThrow(EE_CONVERT_CANTSET, *t_output); } } MCDo::~MCDo() { delete source; delete alternatelang; delete widget; } Parse_stat MCDo::parse(MCScriptPoint &sp) { initpoint(sp); if (sp.parseexp(False, True, &source) != PS_NORMAL) { MCperror->add(PE_DO_BADEXP, sp); return PS_ERROR; } if (sp.skip_token(SP_FACTOR, TT_IN, PT_IN) == PS_NORMAL) { if (sp.skip_token(SP_SUGAR, TT_UNDEFINED, SG_BROWSER) == PS_NORMAL) browser = True; else if (sp.skip_token(SP_SUGAR, TT_UNDEFINED, SG_CALLER) == PS_NORMAL) caller = true; else { widget = new (nothrow) MCChunk(False); if (widget->parse(sp, False) != PS_NORMAL) { MCperror->add(PE_DO_BADENV, sp); return PS_ERROR; } } return PS_NORMAL; } if (sp.skip_token(SP_FACTOR, TT_PREP, PT_AS) == PS_NORMAL) { if (sp.parseexp(False, True, &alternatelang) != PS_NORMAL) { MCperror->add(PE_DO_BADLANG, sp); return PS_ERROR; } } return PS_NORMAL; } void MCDo::exec_ctxt(MCExecContext& ctxt) { MCAutoStringRef t_script; if (!ctxt . EvalExprAsStringRef(source, EE_DO_BADEXP, &t_script)) return; if (widget) { MCObject *t_object; uint32_t t_parid; if (!widget->getobj(ctxt, t_object, t_parid, True) || t_object->gettype() != CT_WIDGET) { ctxt.LegacyThrow(EE_DO_BADWIDGETEXP); return; } MCInterfaceExecDoInWidget(ctxt, *t_script, (MCWidget*)t_object); return; } if (browser) { MCLegacyExecDoInBrowser(ctxt, *t_script); return; } if (alternatelang != NULL) { MCAutoStringRef t_language; if (!ctxt . EvalExprAsStringRef(alternatelang, EE_DO_BADLANG, &t_language)) return; MCScriptingExecDoAsAlternateLanguage(ctxt, *t_script, *t_language); return; } if (debug) { MCDebuggingExecDebugDo(ctxt, *t_script, line, pos); return; } // AL-2014-11-17: [[ Bug 14044 ]] Do in caller not implemented if (caller) { MCEngineExecDoInCaller(ctxt, *t_script, line, pos); return; } MCEngineExecDo(ctxt, *t_script, line, pos); } MCDoMenu::~MCDoMenu() { delete source; } Parse_stat MCDoMenu::parse(MCScriptPoint &sp) { initpoint(sp); if (sp.parseexp(False, True, &source) != PS_NORMAL) { MCperror->add (PE_DOMENU_BADEXP, sp); return PS_ERROR; } return PS_NORMAL; } void MCDoMenu::exec_ctxt(MCExecContext& ctxt) { MCAutoStringRef t_option; if (!ctxt . EvalExprAsStringRef(source, EE_DOMENU_BADEXP, &t_option)) return; MCLegacyExecDoMenu(ctxt, *t_option); } MCEdit::~MCEdit() { delete target; delete m_at; } Parse_stat MCEdit::parse(MCScriptPoint &sp) { initpoint(sp); sp.skip_token(SP_FACTOR, TT_THE); if (sp.skip_token(SP_FACTOR, TT_PROPERTY) != PS_NORMAL) { MCperror->add(PE_EDIT_NOSCRIPT, sp); return PS_ERROR; } if (sp.skip_token(SP_FACTOR, TT_OF) != PS_NORMAL) { MCperror->add(PE_EDIT_NOOF, sp); return PS_ERROR; } target = new (nothrow) MCChunk(False); if (target->parse(sp, False) != PS_NORMAL) { MCperror->add(PE_EDIT_NOTARGET, sp); return PS_ERROR; } // MERG 2013-9-13: [[ EditScriptChunk ]] Added line and column if (sp.skip_token(SP_FACTOR, TT_PREP, PT_AT) == PS_NORMAL) { if (sp.parseexp(False, True, &m_at) != PS_NORMAL) { MCperror->add(PE_EDIT_NOAT, sp); return PS_ERROR; } } return PS_NORMAL; } void MCEdit::exec_ctxt(MCExecContext& ctxt) { MCObject *optr; uint4 parid; if (!target->getobj(ctxt, optr, parid, True)) { ctxt . LegacyThrow(EE_EDIT_BADTARGET); return; } // MERG 2013-9-13: [[ EditScriptChunk ]] Added at expression that's passed through as a second parameter to editScript MCAutoStringRef t_at; if (!ctxt . EvalOptionalExprAsNullableStringRef(m_at, EE_EDIT_BADAT, &t_at)) return; MCIdeExecEditScriptOfObject(ctxt, optr, *t_at); } MCFind::~MCFind() { delete tofind; delete field; } Parse_stat MCFind::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); if (sp.next(type) != PS_NORMAL) { MCperror->add (PE_FIND_NOSTRING, sp); return PS_ERROR; } if (sp.lookup(SP_FIND, te) == PS_NORMAL) mode = (Find_mode)te->which; else sp.backup(); if (sp.parseexp(False, True, &tofind) != PS_NORMAL) { MCperror->add (PE_FIND_BADSTRING, sp); return PS_ERROR; } if (sp.skip_token(SP_FACTOR, TT_IN) == PS_NORMAL) { field = new (nothrow) MCChunk(False); if (field->parse(sp, False) != PS_NORMAL) { MCperror->add (PE_FIND_BADFIELD, sp); return PS_ERROR; } } return PS_NORMAL; } void MCFind::exec_ctxt(MCExecContext& ctxt) { MCAutoStringRef t_needle; if (!ctxt . EvalExprAsStringRef(tofind, EE_FIND_BADSTRING, &t_needle)) return; MCInterfaceExecFind(ctxt, mode, *t_needle, field); // SN-2014-03-21: [[ Bug 11949 ]] 'find' shouldn't throw an error on a failure // but MCInterfaceExecFind would cause the context to be set on error if finding fails ctxt . IgnoreLastError(); } MCGet::~MCGet() { delete value; } Parse_stat MCGet::parse(MCScriptPoint &sp) { initpoint(sp); if (sp.parseexp(False, True, &value) != PS_NORMAL) { MCperror->add (PE_GET_BADEXP, sp); return PS_ERROR; } return PS_NORMAL; } void MCGet::exec_ctxt(MCExecContext& ctxt) { MCExecValue t_value; if (!ctxt . EvaluateExpression(value, EE_GET_BADEXP, t_value)) return; MCEngineExecGet(ctxt, t_value); } MCMarking::~MCMarking() { delete where; delete tofind; delete field; delete card; } Parse_stat MCMarking::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); if (sp.next(type) != PS_NORMAL) { MCperror->add (PE_MARK_NOCARDS, sp); return PS_ERROR; } if (sp.lookup(SP_MARK, te) != PS_NORMAL) { sp.backup(); card = new (nothrow) MCChunk(False); if (card->parse(sp, False) != PS_NORMAL) { MCperror->add (PE_MARK_NOTCARDS, sp); return PS_ERROR; } return PS_NORMAL; } if (te->which == MC_ALL) { if (sp.skip_token(SP_MARK, TT_UNDEFINED, MC_CARDS) != PS_NORMAL) { MCperror->add (PE_MARK_NOCARDS, sp); return PS_ERROR; } return PS_NORMAL; } if (te->which != MC_CARDS) { MCperror->add (PE_MARK_NOCARDS, sp); return PS_ERROR; } if (sp.next(type) != PS_NORMAL) { MCperror->add (PE_MARK_NOBYORWHERE, sp); return PS_ERROR; } if (sp.lookup(SP_MARK,te) != PS_NORMAL || (te->which != MC_BY && te->which != MC_WHERE)) { MCperror->add (PE_MARK_NOTBYORWHERE, sp); return PS_ERROR; } if (te->which == MC_WHERE) { if (sp.parseexp(False, True, &where) != PS_NORMAL) { MCperror->add (PE_MARK_BADWHEREEXP, sp); return PS_ERROR; } return PS_NORMAL; } if (sp.skip_token(SP_MARK, TT_UNDEFINED, MC_FINDING) != PS_NORMAL) { MCperror->add (PE_MARK_NOFINDING, sp); return PS_ERROR; } if (sp.next(type) != PS_NORMAL) { MCperror->add (PE_MARK_NOSTRING, sp); return PS_ERROR; } if (sp.lookup(SP_FIND, te) == PS_NORMAL) mode = (Find_mode)te->which; else sp.backup(); if (sp.parseexp(False, True, &tofind) != PS_NORMAL) { MCperror->add (PE_MARK_BADSTRING, sp); return PS_ERROR; } if (sp.skip_token(SP_FACTOR, TT_IN) == PS_NORMAL) { field = new (nothrow) MCChunk(False); if (field->parse(sp, False) != PS_NORMAL) { MCperror->add (PE_MARK_BADFIELD, sp); return PS_ERROR; } } return PS_NORMAL; } void MCMarking::exec_ctxt(MCExecContext &ctxt) { if (card != NULL) { MCObjectPtr t_object; if (!card->getobj(ctxt, t_object, True) || t_object . object -> gettype() != CT_CARD) { ctxt . LegacyThrow(EE_MARK_BADCARD); return; } if (mark) MCInterfaceExecMarkCard(ctxt, t_object); else MCInterfaceExecUnmarkCard(ctxt, t_object); } // SN-2014-03-21 [[ Bug 11950 ]]: 'mark' shouldn't throw an error when failing to mark when card is nil // Any error set is discarded in the end of this block - unless is was triggered by a bad string. else { if (tofind == nil) { if (mark) { if (where != nil) MCInterfaceExecMarkCardsConditional(ctxt, where); else MCInterfaceExecMarkAllCards(ctxt); } else { if (where != nil) MCInterfaceExecUnmarkCardsConditional(ctxt, where); else MCInterfaceExecUnmarkAllCards(ctxt); } } else { MCAutoStringRef t_needle; if (!ctxt . EvalExprAsStringRef(tofind, EE_MARK_BADSTRING, &t_needle)) return; if (mark) MCInterfaceExecMarkFind(ctxt, mode, *t_needle, field); else MCInterfaceExecUnmarkFind(ctxt, mode, *t_needle, field); } ctxt . IgnoreLastError(); } } MCPut::~MCPut() { delete source; delete dest; // cookie delete name; delete domain; delete path; delete expires; } // put [ unicode | binary ] // put [ unicode ] ( content | markup ) // put [ new ] header // put [ secure ] [ httponly ] cookie [ for ] [ on ] to [ until ] // put ( into | after | before ) message [ box ] // put ( into | after | before ) // Parse_stat MCPut::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); // IM-2011-08-22: [[ SERVER ]] Add support for new put variant. // Parse: put [ secure ] [ httponly ] cookie [ for path ] [ on domain ] with [ until expires ] if (sp . skip_token(SP_SERVER, TT_SERVER, SK_SECURE) == PS_NORMAL) is_secure = true; if (sp . skip_token(SP_SERVER, TT_SERVER, SK_HTTPONLY) == PS_NORMAL) is_httponly = true; if (sp . skip_token(SP_SERVER, TT_PREP, PT_COOKIE) == PS_NORMAL) { prep = PT_COOKIE; if (sp . parseexp(True, False, &name) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } if (sp . skip_token(SP_REPEAT, TT_UNDEFINED, RF_FOR) == PS_NORMAL) { if (sp . parseexp(True, False, &path) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } } if (sp . skip_token(SP_FACTOR, TT_OF, PT_ON) == PS_NORMAL) { if (sp . parseexp(True, False, &domain) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } } sp . skip_token(SP_REPEAT, TT_UNDEFINED, RF_WITH); if (sp . parseexp(True, False, &source) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } if (sp . skip_token(SP_REPEAT, TT_UNDEFINED, RF_UNTIL) == PS_NORMAL) { if (sp . parseexp(True, False, &expires) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } } return PS_NORMAL; } if (is_secure || is_httponly) { MCperror->add(PE_PUT_BADPREP, sp); return PS_ERROR; } // MW-2011-06-22: [[ SERVER ]] Add support for new put variant. // Parse: put new header if (sp.skip_token(SP_SERVER, TT_SERVER, SK_NEW) == PS_NORMAL) { if (sp.skip_token(SP_SERVER, TT_PREP, PT_HEADER) == PS_NORMAL) { prep = PT_NEW_HEADER; if (sp . parseexp(False, True, &source) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } return PS_NORMAL; } else sp.backup(); } // MW-2012-02-23: [[ UnicodePut ]] Store whether 'unicode' was present // in the ast. if (sp . skip_token(SP_SERVER, TT_SERVER, SK_UNICODE) == PS_NORMAL) is_unicode = true; // MW-2011-06-22: [[ SERVER ]] Add support for new put variant. // Parse: put [ unicode ] ( header | content | markup ) if (sp.next(type) == PS_NORMAL) { if (type == ST_ID && sp.lookup(SP_SERVER, te) == PS_NORMAL && te -> type == TT_PREP) { prep = (Preposition_type)te -> which; if (is_unicode && (prep == PT_HEADER || prep == PT_BINARY)) { MCperror->add(PE_PUT_BADPREP, sp); return PS_ERROR; } if (sp . parseexp(False, True, &source) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } return PS_NORMAL; } else sp.backup(); } if (sp.parseexp(False, True, &source) != PS_NORMAL) { MCperror->add(PE_PUT_BADEXP, sp); return PS_ERROR; } if (sp.next(type) != PS_NORMAL) return PS_NORMAL; if (sp.lookup(SP_FACTOR, te) != PS_NORMAL || te->type != TT_PREP) { MCperror->add(PE_PUT_BADPREP, sp); return PS_ERROR; } prep = (Preposition_type)te->which; if (prep != PT_BEFORE && prep != PT_INTO && prep != PT_AFTER) { MCperror->add(PE_PUT_BADPREP, sp); return PS_ERROR; } if (sp.skip_token(SP_SHOW, TT_UNDEFINED, SO_MESSAGE) == PS_NORMAL) { sp.skip_token(SP_SHOW, TT_UNDEFINED, SO_MESSAGE); // "box" return PS_NORMAL; } dest = new (nothrow) MCChunk(True); if (dest->parse(sp, False) != PS_NORMAL) { MCperror->add(PE_PUT_BADCHUNK, sp); return PS_ERROR; } return PS_NORMAL; } void MCPut::exec_ctxt(MCExecContext& ctxt) { MCExecValue t_value; if (!ctxt . EvaluateExpression(source, EE_PUT_BADEXP, t_value)) return; if (dest != nil) { // MCAutoValueRef t_valueref; // MCExecTypeConvertAndReleaseAlways(ctxt, t_value . type, &t_value, kMCExecValueTypeValueRef, &(&t_valueref)); // dest -> set(ctxt, prep, *t_valueref, is_unicode); dest -> set(ctxt, prep, t_value, is_unicode); } else { if (ctxt . HasError()) return; MCAutoValueRef t_val; if (is_unicode && (prep == PT_UNDEFINED || prep == PT_CONTENT || prep == PT_MARKUP)) { MCExecTypeConvertAndReleaseAlways(ctxt, t_value . type, &t_value, kMCExecValueTypeDataRef, &(&t_val)); if (ctxt . HasError()) { ctxt . LegacyThrow(EE_CHUNK_CANTSETDEST); return; } } else { MCExecTypeConvertAndReleaseAlways(ctxt, t_value . type, &t_value, kMCExecValueTypeStringRef, &(&t_val)); if (ctxt . HasError()) { ctxt . LegacyThrow(EE_CHUNK_CANTSETDEST); return; } } // Defined for convenience MCStringRef t_string = (MCStringRef)*t_val; MCDataRef t_data = (MCDataRef)*t_val; if (prep == PT_COOKIE) { MCAutoStringRef t_name; if (!ctxt . EvalOptionalExprAsStringRef(name, kMCEmptyString, EE_PUT_CANTSETINTO, &t_name)) return; uinteger_t t_expires; if (!ctxt . EvalOptionalExprAsUInt(expires, 0, EE_PUT_CANTSETINTO, t_expires)) return; MCAutoStringRef t_path; if (!ctxt . EvalOptionalExprAsStringRef(path, kMCEmptyString, EE_PUT_CANTSETINTO, &t_path)) return; MCAutoStringRef t_domain; if (!ctxt . EvalOptionalExprAsStringRef(domain, kMCEmptyString, EE_PUT_CANTSETINTO, &t_domain)) return; MCServerExecPutCookie(ctxt, *t_name, t_string, t_expires, *t_path, *t_domain, is_secure, is_httponly); } else if (prep == PT_UNDEFINED) { if (is_unicode) MCEngineExecPutOutputUnicode(ctxt, t_data); else MCEngineExecPutOutput(ctxt, t_string); } else if (prep == PT_INTO || prep == PT_AFTER || prep == PT_BEFORE) MCDebuggingExecPutIntoMessage(ctxt, t_string, prep); else if (prep == PT_HEADER || prep == PT_NEW_HEADER) MCServerExecPutHeader(ctxt, t_string, prep == PT_NEW_HEADER); else if (prep == PT_CONTENT) { if (is_unicode) MCServerExecPutContentUnicode(ctxt, t_data); else MCServerExecPutContent(ctxt, t_string); } else if (prep == PT_MARKUP) { if (is_unicode) MCServerExecPutMarkupUnicode(ctxt, t_data); else MCServerExecPutMarkup(ctxt, t_string); } else if (prep == PT_BINARY) MCServerExecPutBinaryOutput(ctxt, t_data); } } MCQuit::~MCQuit() { delete retcode; } Parse_stat MCQuit::parse(MCScriptPoint &sp) { initpoint(sp); MCerrorlock++; sp.parseexp(False, True, &retcode); MCerrorlock--; return PS_NORMAL; } void MCQuit::exec_ctxt(MCExecContext& ctxt) { integer_t t_retcode; if (!ctxt . EvalOptionalExprAsInt(retcode, 0, EE_UNDEFINED, t_retcode)) return; MCEngineExecQuit(ctxt, t_retcode); } Parse_stat MCReset::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); sp.skip_token(SP_FACTOR, TT_THE); if (sp.next(type) != PS_NORMAL) { MCperror->add (PE_CHOOSE_NOTOKEN, sp); return PS_ERROR; } if (sp.lookup(SP_RESET, te) != PS_NORMAL) { MCperror->add (PE_RESET_NOTYPE, sp); return PS_ERROR; } which = (Reset_type)te->which; return PS_NORMAL; } void MCReset::exec_ctxt(MCExecContext& ctxt) { switch (which) { case RT_CURSORS: MCInterfaceExecResetCursors(ctxt); break; case RT_PAINT: MCGraphicsExecResetPaint(ctxt); break; case RT_PRINTING: MCPrintingExecResetPrinting(ctxt); break; default: MCInterfaceExecResetTemplate(ctxt, which); break; } return; } MCReturn::~MCReturn() { delete source; delete extra_source; } Parse_stat MCReturn::parse(MCScriptPoint &sp) { initpoint(sp); if (sp.parseexp(False, True, &source) != PS_NORMAL) { MCperror->add (PE_RETURN_BADEXP, sp); return PS_ERROR; } if (sp.skip_token(SP_REPEAT, TT_UNDEFINED, RF_FOR) == PS_NORMAL) { if (sp.skip_token(SP_SUGAR, TT_UNDEFINED, SG_VALUE) == PS_NORMAL) { kind = kReturnValue; } else if (sp.skip_token(SP_SUGAR, TT_UNDEFINED, SG_ERROR) == PS_NORMAL) { kind = kReturnError; } else { MCperror->add(PE_RETURN_BADFOR, sp); return PS_ERROR; } Handler_type t_handler_type; t_handler_type = sp.gethandler()->gettype(); if (t_handler_type != HT_MESSAGE && t_handler_type != HT_FUNCTION) { MCperror->add(PE_RETURN_BADFORMINCONTEXT, sp); return PS_ERROR; } } else if (sp.skip_token(SP_REPEAT, TT_UNDEFINED, RF_WITH) == PS_NORMAL) { kind = kReturnWithUrlResult; if (sp.skip_token(SP_SUGAR, TT_UNDEFINED, SG_URL_RESULT) != PS_NORMAL || sp.parseexp(False, True, &extra_source) != PS_NORMAL) { MCperror->add(PE_RETURN_BADEXP, sp); return PS_ERROR; } } return PS_NORMAL; } // MW-2007-07-03: [[ Bug 4570 ]] - Using the return command now causes a // RETURN_HANDLER status rather than EXIT_HANDLER. This is used to not // clear the result in this case. (see MCHandler::exec). void MCReturn::exec_ctxt(MCExecContext &ctxt) { MCAutoValueRef t_result; if (!ctxt . EvalExprAsValueRef(source, EE_RETURN_BADEXP, &t_result)) return; if (kind == kReturn) { MCEngineExecReturn(ctxt, *t_result); } else if (kind == kReturnValue) { MCEngineExecReturnValue(ctxt, *t_result); } else if (kind == kReturnError) { MCEngineExecReturnError(ctxt, *t_result); } else if (kind == kReturnWithUrlResult) { MCAutoValueRef t_extra_result; if (!ctxt . EvalExprAsValueRef(extra_source, EE_RETURN_BADEXP, &t_extra_result)) return; MCNetworkExecReturnValueAndUrlResult(ctxt, *t_result, *t_extra_result); } if (!ctxt . HasError()) ctxt . SetIsReturnHandler(); } uint4 MCReturn::linecount() { return 0; } Parse_stat MCScriptError::parse(MCScriptPoint &sp) { return PS_ERROR; // catch on/function/getprop/setprop } MCSet::~MCSet() { delete target; delete value; } Parse_stat MCSet::parse(MCScriptPoint &sp) { initpoint(sp); if (sp.skip_token(SP_FACTOR, TT_THE) == PS_ERROR) { MCperror->add(PE_SET_NOTHE, sp); return PS_ERROR; } target = new (nothrow) MCProperty; if (target->parse(sp, True) != PS_NORMAL) { MCperror->add(PE_SET_NOPROP, sp); return PS_ERROR; } if (sp.skip_token(SP_FACTOR, TT_TO) != PS_NORMAL) { MCperror->add(PE_SET_NOTO, sp); return PS_ERROR; } if (sp.parseexp(False, True, &value) != PS_NORMAL) { MCperror->add(PE_SET_BADEXP, sp); return PS_ERROR; } return PS_NORMAL; } void MCSet::exec_ctxt(MCExecContext& ctxt) { MCAutoValueRef t_value; if (!ctxt . EvalExprAsValueRef(value, EE_SET_BADEXP, &t_value)) return; ctxt . SetTheResultToEmpty(); MCEngineExecSet(ctxt, target, *t_value); } MCSort::~MCSort() { delete of; delete by; } Parse_stat MCSort::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); while (True) { if (sp.next(type) != PS_NORMAL) { if (of == NULL && chunktype == CT_FIELD) { MCperror->add (PE_SORT_NOTARGET, sp); return PS_ERROR; } else break; } if (sp.lookup(SP_SORT, te) == PS_NORMAL) { switch (te->which) { case ST_OF: of = new (nothrow) MCChunk(True); if (of->parse(sp, False) != PS_NORMAL) { MCperror->add (PE_SORT_BADTARGET, sp); return PS_ERROR; } break; case ST_BY: if (chunktype == CT_FIELD && of == NULL) chunktype = CT_CARD; if (sp.parseexp(False, True, &by) != PS_NORMAL) { MCperror->add (PE_SORT_BADEXPRESSION, sp); return PS_ERROR; } if (of == NULL && chunktype == CT_FIELD) { MCperror->add (PE_SORT_NOTARGET, sp); return PS_ERROR; } return PS_NORMAL; case ST_LINES: chunktype = CT_LINE; break; case ST_ITEMS: chunktype = CT_ITEM; break; case ST_MARKED: chunktype = CT_MARKED; break; case ST_CARDS: if (chunktype != CT_MARKED) chunktype = CT_CARD; break; case ST_TEXT: case ST_BINARY: case ST_NUMERIC: case ST_INTERNATIONAL: case ST_DATETIME: format = (Sort_type)te->which; break; case ST_ASCENDING: case ST_DESCENDING: direction = (Sort_type)te->which; break; } } else { sp.backup(); if (of == NULL) { of = new (nothrow) MCChunk(True); if (of->parse(sp, False) != PS_NORMAL) { MCperror->add (PE_SORT_BADTARGET, sp); return PS_ERROR; } } else break; } } return PS_NORMAL; } void MCSort::exec_ctxt(MCExecContext& ctxt) { MCObjectPtr t_object; MCAutoStringRef t_target; // SN-2014-03-21: [[ Bug 11953 ]] sort card does not work t_object . object = nil; t_object . part_id = 0; if (of != NULL) { MCerrorlock++; of->getoptionalobj(ctxt, t_object, False); if (t_object . object == nil || t_object . object->gettype() == CT_BUTTON) { MCerrorlock--; if (!ctxt . EvalExprAsStringRef(of, EE_SORT_BADTARGET, &t_target)) return; } else { MCerrorlock--; } if (t_object . object != nil && t_object . object->gettype() > CT_GROUP && chunktype <= CT_GROUP) chunktype = CT_LINE; } // SN-2015-04-01: [[ Bug 14885 ]] Make sure that the default stack is used // if none is specified. else t_object . object = MCdefaultstackptr; if (chunktype == CT_CARD || chunktype == CT_MARKED) { if (t_object . object == nil || t_object . object -> gettype() != CT_STACK) { ctxt . LegacyThrow(EE_SORT_CANTSORT); return; } MCInterfaceExecSortCardsOfStack(ctxt, (MCStack *)t_object . object, direction == ST_ASCENDING, format, by, chunktype == CT_MARKED); } else if (t_object . object == nil || t_object . object->gettype() == CT_BUTTON) { MCStringRef t_sorted_target; if (*t_target == nil) t_sorted_target = MCValueRetain(kMCEmptyString); else t_sorted_target = MCValueRetain(*t_target); MCInterfaceExecSortContainer(ctxt, t_sorted_target, chunktype, direction == ST_ASCENDING, format, by); if (!ctxt . HasError()) of -> set(ctxt, PT_INTO, t_sorted_target); MCValueRelease(t_sorted_target); } else { if (t_object . object->gettype() != CT_FIELD || !of->notextchunks()) { ctxt . LegacyThrow(EE_SORT_CANTSORT); return; } MCInterfaceExecSortField(ctxt, t_object, chunktype, direction == ST_ASCENDING, format, by); } } MCWait::~MCWait() { delete duration; } Parse_stat MCWait::parse(MCScriptPoint &sp) { Symbol_type type; const LT *te; initpoint(sp); if (sp.next(type) != PS_NORMAL) { MCperror->add (PE_WAIT_NODURATION, sp); return PS_ERROR; } if (sp.lookup(SP_REPEAT, te) == PS_NORMAL) condition = (Repeat_form)te->which; else { condition = RF_FOR; sp.backup(); } if (sp.skip_token(SP_MOVE, TT_UNDEFINED, MM_MESSAGES) == PS_NORMAL) messages = True; else { if (sp.parseexp(False, True, &duration) != PS_NORMAL) { MCperror->add (PE_WAIT_BADCOND, sp); return PS_ERROR; } if (condition == RF_FOR) { if (sp.skip_token(SP_FACTOR, TT_FUNCTION, F_MILLISECS) == PS_NORMAL) units = F_MILLISECS; else if (sp.skip_token(SP_FACTOR, TT_FUNCTION, F_SECONDS) == PS_NORMAL || sp.skip_token(SP_FACTOR, TT_CHUNK, CT_SECOND) == PS_NORMAL) units = F_SECONDS; else if (sp.skip_token(SP_FACTOR, TT_FUNCTION, F_TICKS) == PS_NORMAL) units = F_TICKS; else units = F_TICKS; } if (sp.skip_token(SP_REPEAT, TT_UNDEFINED, RF_WITH) == PS_NORMAL) { sp.skip_token(SP_MOVE, TT_UNDEFINED, MM_MESSAGES); messages = True; } } return PS_NORMAL; } void MCWait::exec_ctxt(MCExecContext& ctxt) { if (duration == NULL) MCEngineExecWaitFor(ctxt, MCmaxwait, F_UNDEFINED, messages == True); else { switch (condition) { case RF_FOR: { double t_delay; if (!ctxt . EvalExprAsDouble(duration, EE_WAIT_BADEXP, t_delay)) return; MCEngineExecWaitFor(ctxt, t_delay, units, messages == True); break; } case RF_UNTIL: MCEngineExecWaitUntil(ctxt, duration, messages == True); break; case RF_WHILE: MCEngineExecWaitWhile(ctxt, duration, messages == True); break; default: return; } } } MCInclude::~MCInclude(void) { delete filename; } Parse_stat MCInclude::parse(MCScriptPoint& sp) { initpoint(sp); if (sp . parseexp(False, True, &filename) != PS_NORMAL) { MCperror -> add(PE_INCLUDE_BADFILENAME, sp); return PS_ERROR; } return PS_NORMAL; } void MCInclude::exec_ctxt(MCExecContext& ctxt) { MCAutoStringRef t_filename; if (!ctxt . EvalExprAsStringRef(filename, EE_INCLUDE_BADFILENAME, &t_filename)) return; MCServerExecInclude(ctxt, *t_filename, is_require); } MCEcho::~MCEcho(void) { if (data != nil) MCValueRelease(data); } Parse_stat MCEcho::parse(MCScriptPoint& sp) { initpoint(sp); data = MCValueRetain(sp . gettoken_stringref()); return PS_NORMAL; } void MCEcho::exec_ctxt(MCExecContext& ctxt) { MCServerExecEcho(ctxt, data); return; } MCResolveImage::~MCResolveImage(void) { delete m_relative_object; delete m_id_or_name; } Parse_stat MCResolveImage::parse(MCScriptPoint &p_sp) { Parse_stat t_stat; t_stat = PS_NORMAL; if (t_stat == PS_NORMAL) t_stat = p_sp.skip_token(SP_FACTOR, TT_CHUNK, CT_IMAGE); // Parse the optional 'id' token m_is_id = (PS_NORMAL == p_sp . skip_token(SP_FACTOR, TT_PROPERTY, P_ID)); // Parse the id_or_name expression if (t_stat == PS_NORMAL) t_stat = p_sp . parseexp(False, True, &m_id_or_name); if (t_stat != PS_NORMAL) { MCperror->add (PE_RESOLVE_BADIMAGE, p_sp); return PS_ERROR; } // Parse the 'relative to' tokens if (t_stat == PS_NORMAL) t_stat = p_sp . skip_token(SP_FACTOR, TT_TO, PT_RELATIVE); if (t_stat == PS_NORMAL) t_stat = p_sp . skip_token(SP_FACTOR, TT_TO, PT_TO); // Parse the target object clause if (t_stat == PS_NORMAL) { m_relative_object = new (nothrow) MCChunk(false); t_stat = m_relative_object -> parse(p_sp, False); } else { MCperror->add (PE_RESOLVE_BADOBJECT, p_sp); return PS_ERROR; } return t_stat; } void MCResolveImage::exec_ctxt(MCExecContext &ctxt) { uint4 t_part_id; MCObject *t_relative_object; if (!m_relative_object -> getobj(ctxt, t_relative_object, t_part_id, True)) { ctxt . Throw(); return; } if (m_is_id) { uinteger_t t_id; if (!ctxt . EvalExprAsUInt(m_id_or_name, EE_RESOLVE_IMG_BADEXP, t_id)) return; MCInterfaceExecResolveImageById(ctxt, t_relative_object, t_id); } else { MCAutoStringRef t_name; if (!ctxt . EvalExprAsStringRef(m_id_or_name, EE_RESOLVE_IMG_BADEXP, &t_name)) return; MCInterfaceExecResolveImageByName(ctxt, t_relative_object, *t_name); } } //////////////////////////////////////////////////////////////////////////////// // // MW-2013-11-14: [[ AssertCmd ]] Implementation of 'assert' command. // MCAssertCmd::~MCAssertCmd(void) { delete m_expr; } // assert // assert true // assert false // assert success // assert failure Parse_stat MCAssertCmd::parse(MCScriptPoint& sp) { initpoint(sp); // See if there is a type token MCScriptPoint temp_sp(sp); if (sp . skip_token(SP_SUGAR, TT_UNDEFINED, SG_TRUE) == PS_NORMAL) m_type = ASSERT_TYPE_TRUE; else if (sp . skip_token(SP_SUGAR, TT_UNDEFINED, SG_FALSE) == PS_NORMAL) m_type = ASSERT_TYPE_FALSE; else if (sp . skip_token(SP_SUGAR, TT_UNDEFINED, SG_SUCCESS) == PS_NORMAL) m_type = ASSERT_TYPE_SUCCESS; else if (sp . skip_token(SP_SUGAR, TT_UNDEFINED, SG_FAILURE) == PS_NORMAL) m_type = ASSERT_TYPE_FAILURE; else m_type = ASSERT_TYPE_NONE; // Now try to parse an expression if (sp.parseexp(False, True, &m_expr) == PS_NORMAL) return PS_NORMAL; // Now if we are not of NONE type, then backup and try for just an // expression (ASSERT_TYPE_NONE). if (m_type != ASSERT_TYPE_NONE) { MCperror -> clear(); sp = temp_sp; } // Parse the expression again (if not NONE, otherwise we already have // a badexpr error to report). if (m_type == ASSERT_TYPE_NONE || sp.parseexp(False, True, &m_expr) != PS_NORMAL) { MCperror -> add(PE_ASSERT_BADEXPR, sp); return PS_ERROR; } // We must be of type none. m_type = ASSERT_TYPE_NONE; return PS_NORMAL; } void MCAssertCmd::exec_ctxt(MCExecContext& ctxt) { bool t_success, t_result; t_success = ctxt . EvalExprAsNonStrictBool(m_expr, EE_UNDEFINED, t_result); if (!t_success) ctxt . IgnoreLastError(); MCDebuggingExecAssert(ctxt, m_type, t_success, t_result); }