/* 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 "platform.h" #include "globdefs.h" #include "filedefs.h" #include "osspec.h" #include "typedefs.h" #include "parsedef.h" #include "objdefs.h" #include "unicode.h" #include "exec.h" #include "scriptpt.h" #include "mcerror.h" #include "globals.h" #include "util.h" #include "stack.h" #include "card.h" #include "debug.h" #include "dispatch.h" #include "mccontrol.h" #include "field.h" #include "graphics_util.h" #include "redraw.h" #include "player.h" #include "aclip.h" #include "stacklst.h" #include "clipboard.h" #include "desktop-dc.h" #include "param.h" //////////////////////////////////////////////////////////////////////////////// #if defined(FEATURE_PLATFORM_APPLICATION) void X_main_loop(void) { while(!MCquit) X_main_loop_iteration(); } //////////////////////////////////////////////////////////////////////////////// void MCPlatformHandleApplicationStartup(int p_argc, MCStringRef *p_argv, MCStringRef *p_envp, int& r_error_code, MCStringRef & r_error_message) { struct X_init_options t_options; t_options.argc = p_argc; t_options.argv = p_argv; t_options.envp = p_envp; t_options.app_code_path = nullptr; if (X_init(t_options)) { r_error_code = 0; r_error_message = nil; return; } r_error_code = -1; if (MCresult == nullptr) { /* TODO[2017-04-05] X_init() failed before initialising global * variables. This could be because something horrible happened, or it * could be because it found "-h" in the arguments. It's better to * quit without an error message than to crash, but it the future it * would be good to distinguish between the two. */ r_error_message = MCValueRetain(kMCEmptyString); } else if (MCValueGetTypeCode(MCresult -> getvalueref()) == kMCValueTypeCodeString) { r_error_message = (MCStringRef)MCValueRetain(MCresult->getvalueref()); } else { r_error_message = MCValueRetain(MCSTR("Unknown error occurred")); } } void MCPlatformHandleApplicationShutdown(int& r_exit_code) { r_exit_code = X_close(); } void MCPlatformHandleApplicationShutdownRequest(bool& r_terminate) { switch(MCdefaultstackptr->getcard()->message(MCM_shut_down_request)) { case ES_PASS: case ES_NOT_HANDLED: MCdefaultstackptr->getcard()->message(MCM_shut_down); MCquit = True; MCexitall = True; MCtracestackptr = nil; MCtraceabort = True; MCtracereturn = True; r_terminate = true; break; default: r_terminate = false; break; } } void MCPlatformHandleApplicationSuspend(void) { // FG-2014-09-22: [[ Bugfix 13480 ]] May have no default stack on startup if (MCdefaultstackptr) MCdefaultstackptr -> getcard() -> message(MCM_suspend); MCappisactive = False; } void MCPlatformHandleApplicationResume(void) { // FG-2014-09-22: [[ Bugfix 13480 ]] May have no default stack on startup MCappisactive = True; if (MCdefaultstackptr) MCdefaultstackptr -> getcard() -> message(MCM_resume); } void MCPlatformHandleApplicationRun(bool& r_continue) { X_main_loop_iteration(); r_continue = !MCquit; } #endif // FEATURE_PLATFORM_APPLICATION //////////////////////////////////////////////////////////////////////////////// void MCPlatformHandleScreenParametersChanged(void) { // It is possible for this notification to be sent *before* MCscreen has // been initialized. In this case, we do nothing (the screen info is fetched // on first use so there's no need to do anything before MCscreen is // initialized). if (MCscreen == nil) return; // IM-2014-01-28: [[ HiDPI ]] Use updatedisplayinfo() method to update & compare display details bool t_changed; t_changed = false; MCscreen->updatedisplayinfo(t_changed); // Post a desktop changed message. MCscreen -> delaymessage(MCdefaultstackptr -> getcurcard(), MCM_desktop_changed); } //////////////////////////////////////////////////////////////////////////////// #if defined (FEATURE_PLATFORM_WINDOW) void MCPlatformHandleWindowCloseRequest(MCPlatformWindowRef p_window) { MCdispatcher -> wclose(p_window); } void MCPlatformHandleWindowClose(MCPlatformWindowRef p_window) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; else { t_stack -> kunfocus(); t_stack -> close(); t_stack -> checkdestroy(); } } void MCPlatformHandleWindowCancel(MCPlatformWindowRef p_window) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; // Handle any cancelled popup menus. MCdispatcher -> closemenus(); // Now make sure the stack is indeed closed, it might be a transient. if (t_stack -> getopened() != 0) { t_stack -> kunfocus(); t_stack -> close(); t_stack -> checkdestroy(); } } void MCPlatformHandleWindowReshape(MCPlatformWindowRef p_window) { MCdispatcher -> wreshape(p_window); } void MCPlatformHandleWindowConstrain(MCPlatformWindowRef p_window, MCPoint p_proposed_size, MCPoint& r_wanted_size) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) { r_wanted_size = p_proposed_size; return; } t_stack -> constrain(p_proposed_size, r_wanted_size); } void MCPlatformHandleWindowRedraw(MCPlatformWindowRef p_window, MCPlatformSurfaceRef p_surface, MCGRegionRef p_region) { if (((MCScreenDC *)MCscreen) -> isbackdrop(p_window)) { ((MCScreenDC *)MCscreen) -> redrawbackdrop(p_surface, p_region); return; } MCdispatcher -> wredraw(p_window, p_surface, p_region); } void MCPlatformHandleWindowIconify(MCPlatformWindowRef p_window) { MCdispatcher -> wiconify(p_window); } void MCPlatformHandleWindowUniconify(MCPlatformWindowRef p_window) { MCdispatcher -> wuniconify(p_window); } void MCPlatformHandleWindowFocus(MCPlatformWindowRef p_window) { MCdispatcher -> wkfocus(p_window); } void MCPlatformHandleWindowUnfocus(MCPlatformWindowRef p_window) { MCdispatcher -> wkunfocus(p_window); } void MCPlatformHandleViewFocusSwitched(MCPlatformWindowRef p_window, uint32_t p_view_id) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; if (p_view_id == 0) t_stack -> getcard() -> kfocus(); else { MCControl *t_control; MCAutoStringRef t_id; /* UNCHECKED */ MCStringFormat(&t_id, "%d", p_view_id); t_control = t_stack -> getcard() -> getchild(CT_ID, *t_id, CT_LAYER, CT_UNDEFINED); if (t_control != nil) t_stack -> kfocusset(t_control); else t_stack -> getcard() -> kunfocus(); } } //////////////////////////////////////////////////////////////////////////////// extern Boolean tripleclick; void MCPlatformHandleMouseEnter(MCPlatformWindowRef p_window) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; if (!MCmousestackptr.IsBoundTo(t_stack)) { MCmousestackptr = t_stack; MCmousestackptr -> enter(); } // MW-2014-06-23: [[ Bug 12670 ]] This shouldn't be necessary as mouseEnter // is followed immediately by a mouseMove after the mouseLoc has been // updated. /*MCObject *t_menu; t_menu = MCdispatcher -> getmenu(); if (t_menu == nil) MCmousestackptr -> mfocus(MCmousex, MCmousey); else t_menu -> mfocus(MCmousex, MCmousey);*/ } void MCPlatformHandleMouseLeave(MCPlatformWindowRef p_window) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; if (MCmousestackptr.IsBoundTo(t_stack)) { MCmousestackptr -> munfocus(); MCmousestackptr = nil; } } void MCPlatformHandleMouseMove(MCPlatformWindowRef p_window, MCPoint p_location) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; MCObject *t_menu; t_menu = MCdispatcher -> getmenu(); if (MCmousestackptr.IsBoundTo(t_stack) || t_menu != nil) { MCeventtime = MCPlatformGetEventTime(); MCObject *t_target; t_target = t_menu != nil ? t_menu : MCmousestackptr; // IM-2014-02-12: [[ StackScale ]] mfocus will translate target stack to menu stack coords // so in both cases we pass target stack coords. // IM-2014-02-14: [[ StackScale ]] Don't try to convert if target is null MCPoint t_mouseloc; if (t_stack != nil) t_mouseloc = t_stack->windowtostackloc(p_location); MCmousex = t_mouseloc.x; MCmousey = t_mouseloc.y; //MCLog("MouseMove(%p, %d, %d)", t_target, t_mouseloc . x, t_mouseloc . y); t_target -> mfocus(t_mouseloc . x, t_mouseloc . y); } } void MCPlatformHandleMouseDown(MCPlatformWindowRef p_window, uint32_t p_button, uint32_t p_count) { if (((MCScreenDC *)MCscreen) -> isbackdrop(p_window)) { ((MCScreenDC *)MCscreen) -> mousedowninbackdrop(p_button, p_count); return; } MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; MCObject *t_menu; t_menu = MCdispatcher -> getmenu(); if (MCmousestackptr.IsBoundTo(t_stack) || t_menu != nil) { MCbuttonstate |= (1 << p_button); MCeventtime = MCPlatformGetEventTime(); MCclicklocx = MCmousex; MCclicklocy = MCmousey; MCclickstackptr = MCmousestackptr; MCObject *t_target; t_target = t_menu != nil ? t_menu : MCclickstackptr; tripleclick = p_count == 2; //MCLog("MouseDown(%p, %d, %d)", t_target, p_button, p_count); if (p_count != 1) { if (p_count == 0 && !MCdispatcher -> isdragsource()) { MCallowabledragactions = DRAG_ACTION_COPY; MCdragaction = DRAG_ACTION_NONE; MCdragimageid = 0; MCdragimageoffset . x = 0; MCdragimageoffset . y = 0; // ensure the dragboard is bound correctly MCAutoRefcounted t_dragboard(MCRawClipboard::CreateSystemDragboard()); MCdragboard->Rebind(t_dragboard); } t_target -> mdown(p_button + 1); } else t_target -> doubledown(p_button + 1); } } void MCPlatformHandleMouseUp(MCPlatformWindowRef p_window, uint32_t p_button, uint32_t p_count) { if (((MCScreenDC *)MCscreen) -> isbackdrop(p_window)) { ((MCScreenDC *)MCscreen) -> mouseupinbackdrop(p_button, p_count); return; } MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; MCObject *t_menu; t_menu = MCdispatcher -> getmenu(); if (MCmousestackptr.IsBoundTo(t_stack) || t_menu != nil) { MCbuttonstate &= ~(1 << p_button); MCeventtime = MCPlatformGetEventTime(); // PM-2015-03-30: [[ Bug 15091 ]] When we "go to card X" on mouseDown, MCclickstackptr becomes nil because of MCStack::close(). if (!MCclickstackptr) MCclickstackptr = MCmousestackptr; MCObject *t_target; t_target = t_menu != nil ? t_menu : MCclickstackptr; //MCLog("MouseUp(%p, %d, %d)", t_target, p_button, p_count); if (p_count != 1) t_target -> mup(p_button + 1, false); else t_target -> doubleup(p_button + 1); } } void MCPlatformHandleMouseDrag(MCPlatformWindowRef p_window, uint32_t p_button) { MCdispatcher -> wmdrag(p_window); } void MCPlatformHandleMouseRelease(MCPlatformWindowRef p_window, uint32_t p_button, bool p_was_menu) { if (((MCScreenDC *)MCscreen) -> isbackdrop(p_window)) { ((MCScreenDC *)MCscreen) -> mousereleaseinbackdrop(p_button); return; } MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; MCObject *t_menu; t_menu = MCdispatcher -> getmenu(); if (MCmousestackptr.IsBoundTo(t_stack) || t_menu != nil) { tripleclick = False; MCbuttonstate &= ~(1 << p_button); // If the press was 'released' i.e. cancelled then we stop messages, mup then // dispatch a mouseRelease message ourselves. // FG-2013-10-09 [[ Bugfix 11208 ]] // CS_NO_MESSAGES only applies to the target and not the controls it contains // so the mouse up message (on mouseUp) gets sent when it isn't desired // Hopefully nobody depends on the old behaviour... // FG-2014-06-27 [[ Bugfix 12705 ]] // Turns out that sending a release message directly to the stack won't // work because it will never get delivered to the control with mouse // focus. Instead, the mup handler has been changed to accept a second // parameter that, when true, indicates that this is a release event. MCObject *t_target; t_target = t_menu != nil ? t_menu : MCclickstackptr; // MW-2014-06-11: [[ Bug 12339 ]] Only send a mouseRelease message if this wasn't the result of a popup menu. if (!p_was_menu) t_target -> mup(p_button + 1, true); else { bool old_lock = MClockmessages; MClockmessages = true; t_target -> mup(p_button + 1, true); MClockmessages = old_lock; } //MCLog("MouseRelease(%p, %d)", t_target, p_button); } } void MCPlatformHandleMouseScroll(MCPlatformWindowRef p_window, int p_dx, int p_dy) { MCStack *t_stack; t_stack = MCdispatcher -> findstackd(p_window); if (t_stack == nil) return; if (!MCmousestackptr.IsBoundTo(t_stack)) return; MCObject *mfocused; mfocused = MCmousestackptr->getcard()->getmfocused(); if (mfocused == NULL) mfocused = MCmousestackptr -> getcard(); if (mfocused == NULL) mfocused = MCmousestackptr; if (p_dy != 0) mfocused -> kdown(kMCEmptyString, p_dy < 0 ? XK_WheelUp : XK_WheelDown); mfocused = MCmousestackptr->getcard()->getmfocused(); if (mfocused == NULL) mfocused = MCmousestackptr -> getcard(); if (mfocused == NULL) mfocused = MCmousestackptr; if (p_dx != 0) mfocused -> kdown(kMCEmptyString, p_dx < 0 ? XK_WheelLeft : XK_WheelRight); } ////////// static MCPlatformDragOperation dragaction_to_dragoperation(MCDragAction p_action) { switch(p_action) { case DRAG_ACTION_NONE: return kMCPlatformDragOperationNone; case DRAG_ACTION_MOVE: return kMCPlatformDragOperationMove; case DRAG_ACTION_COPY: return kMCPlatformDragOperationCopy; case DRAG_ACTION_LINK: return kMCPlatformDragOperationLink; } assert(false); return kMCPlatformDragOperationNone; } void MCPlatformHandleDragEnter(MCPlatformWindowRef p_window, MCRawClipboard* p_dragboard, MCPlatformDragOperation& r_operation) { // On some platforms (Mac and iOS), the drag board used for drag-and-drop // operations may not be the main drag board. If a non-NULL clipboard was // supplied for this operation, tell the engine drag board to re-bind to it. if (p_dragboard != NULL) MCdragboard->Rebind(p_dragboard); MCdispatcher->wmdragenter(p_window); r_operation = dragaction_to_dragoperation(MCdragaction); } void MCPlatformHandleDragMove(MCPlatformWindowRef p_window, MCPoint p_location, MCPlatformDragOperation& r_operation) { MCdispatcher -> wmdragmove(p_window, p_location . x, p_location . y); r_operation = dragaction_to_dragoperation(MCdragaction); } void MCPlatformHandleDragLeave(MCPlatformWindowRef p_window) { // On some platforms (Mac and iOS), the drag board used for drag-and-drop // operations may not be the main drag board. Reset the drag board back to // the main one after the drag has left. MCAutoRefcounted t_dragboard(MCRawClipboard::CreateSystemDragboard()); MCdragboard->Rebind(t_dragboard); MCdispatcher -> wmdragleave(p_window); } void MCPlatformHandleDragDrop(MCPlatformWindowRef p_window, bool& r_accepted) { MCdispatcher -> wmdragdrop(p_window); // On some platforms (Mac and iOS), the drag board used for drag-and-drop // operations may not be the main drag board. Reset the drag board back to // the main one after the drag has left. MCAutoRefcounted t_dragboard(MCRawClipboard::CreateSystemDragboard()); MCdragboard->Rebind(t_dragboard); // PLATFORM-TODO: Should we do more than this? i.e. Should the dragDrop // message be able to signal refusal? r_accepted = true; } //////////////////////////////////////////////////////////////////////////////// // SN-2014-09-15: [[ Bug 13423 ]] Added new static variable to keep the last keys pressed // SN-2015-06-23: [[ Bug 3537 ]] Sometimes a keyUp key message can be already // mapped (when nativised). struct MCKeyMessage { MCPlatformKeyCode key_code; codepoint_t mapped_codepoint; codepoint_t unmapped_codepoint; bool needs_mapping; struct MCKeyMessage* next; }; typedef MCKeyMessage MCKeyMessage; // SN-2014-11-03: [[ Bug 13832 ]] Added a message queue for keyUp messages static MCKeyMessage* s_pending_key_down = nil; static MCKeyMessage* s_pending_key_up = nil; // SN-2014-11-03: [[ Bug 13832 ]] Added a message queue parameter void MCKeyMessageClear(MCKeyMessage *&p_message_queue) { MCKeyMessage *t_next; t_next = p_message_queue; while (t_next != nil) { p_message_queue = p_message_queue -> next; delete t_next; t_next = p_message_queue; } p_message_queue = nil; } // SN-2014-11-03: [[ Bug 13832 ]] Added a message queue parameter // SN-2015-06-23: [[ Bug 3537 ]] If the input has already been nativised, then // we don't want to map it with map_key_to_engine. void MCKeyMessageAppend(MCKeyMessage *&p_message_queue, MCPlatformKeyCode p_key_code, codepoint_t p_mapped_codepoint, codepoint_t p_unmapped_codepoint, bool p_needs_mapping = true) { MCKeyMessage *t_new; t_new = new (nothrow) MCKeyMessage; t_new -> key_code = p_key_code; t_new -> mapped_codepoint = p_mapped_codepoint; t_new -> unmapped_codepoint = p_unmapped_codepoint; t_new -> needs_mapping = p_needs_mapping; t_new -> next = nil; if (p_message_queue != nil) { MCKeyMessage *t_ptr; t_ptr = p_message_queue; while (t_ptr -> next != nil) t_ptr = t_ptr -> next; t_ptr -> next = t_new; } else p_message_queue = t_new; } // SN-2014-11-03: [[ Bug 13832 ]] Added a message queue parameter void MCKeyMessageNext(MCKeyMessage *&p_message_queue) { if (p_message_queue) { MCKeyMessage *t_old; t_old = p_message_queue; p_message_queue = p_message_queue -> next; delete t_old; } } // MW-2014-06-25: [[ Bug 12370 ]] Map an input keyCode and mapped codepoint to // engine keysym and char. The engine expects keyCode to be the mapped key // if an ascii character and the raw keycode if not. static void map_key_to_engine(MCPlatformKeyCode p_key_code, codepoint_t p_mapped_codepoint, codepoint_t p_unmapped_codepoint, MCPlatformKeyCode& r_key_code, MCStringRef &r_native_char) { if (p_mapped_codepoint <= 0xffff) { uint16_t t_unicode_char; char_t t_native_char; // MW-2014-08-05: [[ Bug 13042 ]] This was a mis-merge from an updated fix to bug 12747 // (the code previously was using unmapped codepoint). t_unicode_char = p_mapped_codepoint & 0xffff; if (MCUnicodeMapToNative(&t_unicode_char, 1, t_native_char)) { /* UNCHECKED */ MCStringCreateWithNativeChars(&t_native_char, 1, r_native_char); // MW-2014-06-25: [[ Bug 12370 ]] The engine expects keyCode to be the mapped key whenever // the mapped key is ASCII. If the mapped key is not ASCII then the keyCode reflects // the raw (US English) keycode. // SN-2014-12-08: [[ Bug 14067 ]] Avoid to use the native char instead of the key code // the numeric keypad keys. if (isascii(t_native_char) && (p_key_code < kMCPlatformKeyCodeKeypadSpace || p_key_code > kMCPlatformKeyCodeKeypadEqual)) r_key_code = t_native_char; else r_key_code = p_key_code; return; } // SN-2014-12-05: [[ Bug 14162 ]] We can have unicode chars being typed. // We keep the given keycode (the codepoint) as the key code in these conditions. else { /* UNCHECKED */ MCStringCreateWithChars(&t_unicode_char, 1, r_native_char); r_key_code = p_key_code; return; } } r_native_char = MCValueRetain(kMCEmptyString); r_key_code = p_key_code; } void MCPlatformHandleModifiersChanged(MCPlatformModifiers p_modifiers) { MCmodifierstate = 0; if ((p_modifiers & kMCPlatformModifierShift) != 0) MCmodifierstate |= MS_SHIFT; if ((p_modifiers & kMCPlatformModifierControl) != 0) MCmodifierstate |= MS_CONTROL; if ((p_modifiers & kMCPlatformModifierAlt) != 0) MCmodifierstate |= MS_MOD1; if ((p_modifiers & kMCPlatformModifierMeta) != 0) MCmodifierstate |= MS_MOD2; if ((p_modifiers & kMCPlatformModifierCapsLock) != 0) MCmodifierstate |= MS_CAPS_LOCK; } // MW-2014-04-15: [[ Bug 12086 ]] This method is invoked to give us the last key // code that was passed to an IME session. This allows us to correctly synthesize // a keydown / keyup pair if a single character is produced. void MCPlatformHandleRawKeyDown(MCPlatformWindowRef p_window, MCPlatformKeyCode p_key_code, codepoint_t p_mapped_codepoint, codepoint_t p_unmapped_codepoint) { int32_t si, ei; // SN-2014-09-15: [[ Bug 13423 ]] Clear the key sequence if needed, then append // the new key typed. if (MCactivefield && !MCactivefield -> getcompositionrange(si, ei)) MCKeyMessageClear(s_pending_key_down); MCKeyMessageAppend(s_pending_key_down, p_key_code, p_mapped_codepoint, p_unmapped_codepoint); } void MCPlatformHandleKeyDown(MCPlatformWindowRef p_window, MCPlatformKeyCode p_key_code, codepoint_t p_mapped_codepoint, codepoint_t p_unmapped_codepoint) { MCPlatformKeyCode t_mapped_key_code; MCAutoStringRef t_mapped_char; map_key_to_engine(p_key_code, p_mapped_codepoint, p_unmapped_codepoint, t_mapped_key_code, &t_mapped_char); MCdispatcher -> wkdown(p_window, *t_mapped_char, t_mapped_key_code); // SN-2014-11-03: [[ Bug 13832]] Enqueue the event instead of firing it now (we are still in the NSApplication's keyDown) MCKeyMessageAppend(s_pending_key_up, p_key_code, p_mapped_codepoint, p_unmapped_codepoint); } void MCPlatformHandleKeyUp(MCPlatformWindowRef p_window, MCPlatformKeyCode p_key_code, codepoint_t p_mapped_codepoint, codepoint_t p_unmapped_codepoint) { // SN-2014-10-31: [[ Bug 13832 ]] We now output all the key messages that have been queued // (by either MCPlatformHandleKeyDown, or MCPlatformHandleTextInputInsertText) while (s_pending_key_up != nil) { MCPlatformKeyCode t_mapped_key_code; MCAutoStringRef t_mapped_char; // SN-2015-06-23: [[ Bug 3537 ]] We don't want to map a nativised char - // but we want to map key strokes like F1 and such // This is intended to mimic the behaviour of the key down process, // in which MCDispatcher::wkdown is called with the nativised char if (s_pending_key_up -> needs_mapping) map_key_to_engine(s_pending_key_up -> key_code, s_pending_key_up -> mapped_codepoint, s_pending_key_up -> unmapped_codepoint, t_mapped_key_code, &t_mapped_char); else { /* UNCHECKED */ MCStringCreateWithNativeChars((const char_t*)&(s_pending_key_up -> mapped_codepoint), 1, &t_mapped_char); t_mapped_key_code = s_pending_key_up -> key_code; } MCdispatcher -> wkup(p_window, *t_mapped_char, t_mapped_key_code); MCKeyMessageNext(s_pending_key_up); } } void MCPlatformHandleTextInputQueryTextRanges(MCPlatformWindowRef p_window, MCRange& r_marked_range, MCRange& r_selected_range) { if (!MCactivefield) { r_marked_range = MCRangeMake(UINDEX_MAX, 0); r_selected_range = MCRangeMake(UINDEX_MAX, 0); return; } int4 si, ei; MCactivefield -> selectedmark(False, si, ei, False); MCactivefield -> unresolvechars(0, si, ei); r_selected_range = MCRangeMakeMinMax(si, ei); if (MCactivefield -> getcompositionrange(si, ei)) { MCactivefield -> unresolvechars(0, si, ei); r_marked_range = MCRangeMakeMinMax(si, ei); } else r_marked_range = MCRangeMake(UINDEX_MAX, 0); } void MCPlatformHandleTextInputQueryTextIndex(MCPlatformWindowRef p_window, MCPoint p_location, uindex_t& r_index) { if (!MCactivefield) { r_index = 0; return; } MCPoint t_location; t_location = MCactivefield -> getstack() -> windowtostackloc(p_location); int32_t si, ei; if (MCactivefield -> locmarkpoint(p_location, False, False, False, False, si, ei)) MCactivefield -> unresolvechars(0, si, ei); else si = 0; r_index = si; } void MCPlatformHandleTextInputQueryTextRect(MCPlatformWindowRef p_window, MCRange p_range, MCRectangle& r_first_line_rect, MCRange& r_actual_range) { if (!MCactivefield) { r_first_line_rect = MCRectangleMake(0, 0, 0, 0); r_actual_range = MCRangeMake(UINDEX_MAX, 0); return; } int32_t t_si, t_ei; t_si = 0; t_ei = INT32_MAX; MCactivefield -> resolvechars(0, t_si, t_ei, p_range . offset, p_range . length); MCRectangle t_rect; t_rect = MCactivefield -> firstRectForCharacterRange(t_si, t_ei); MCactivefield -> unresolvechars(0, t_si, t_ei); MCPoint t_top_left, t_bottom_right; t_top_left = MCactivefield -> getstack() -> stacktowindowloc(MCPointMake(t_rect . x, t_rect . y)); t_bottom_right = MCactivefield -> getstack() -> stacktowindowloc(MCPointMake(t_rect . x + t_rect . width, t_rect . y + t_rect . height)); r_first_line_rect = MCRectangleMake(t_top_left . x, t_top_left . y, t_bottom_right . x - t_top_left . x, t_bottom_right . y - t_top_left . y); r_actual_range = MCRangeMakeMinMax(t_si, t_ei); } void MCPlatformHandleTextInputQueryText(MCPlatformWindowRef p_window, MCRange p_range, unichar_t*& r_chars, uindex_t& r_char_count, MCRange& r_actual_range) { if (!MCactivefield) { r_chars = nil; r_char_count = 0; r_actual_range = MCRangeMake(p_range . offset, 0); return; } int32_t t_si, t_ei; MCAutoStringRef t_text; t_si = 0; t_ei = INT32_MAX; MCactivefield -> resolvechars(0, t_si, t_ei, p_range . offset, p_range . length); MCactivefield -> exportastext(0, t_si, t_ei, &t_text); MCactivefield -> unresolvechars(0, t_si, t_ei); /* UNCHECKED */ MCStringConvertToUnicode(*t_text, r_chars, r_char_count); r_actual_range = MCRangeMakeMinMax(t_si, t_ei); } void MCPlatformHandleTextInputInsertText(MCPlatformWindowRef p_window, unichar_t *p_chars, uindex_t p_char_count, MCRange p_replace_range, MCRange p_selection_range, bool p_mark) { if (!MCactivefield) return; // SN-2014-12-04: [[ Bug 14152 ]] Locking the screen here doesn't allow the screen to refresh after // text input, inside an MCWait loop // MCRedrawLockScreen(); int32_t t_r_si, t_r_ei; t_r_si = 0; t_r_ei = INT32_MAX; MCactivefield -> resolvechars(0, t_r_si, t_r_ei, p_replace_range . offset, p_replace_range . length); // SN-2014-09-15: [[ Bug 13423 ]] t_was_compositing now used further in the function bool t_was_compositing; t_was_compositing = false; if (!p_mark) { // MW-2014-08-05: [[ Bug 13098 ]] If we have been compositing, then don't synthesise a // keyDown / keyUp. int4 si, ei; if (MCactivefield -> getcompositionrange(si, ei)) { if (si < t_r_si) t_r_si -= MCMin(t_r_si - si, ei - si); if (si < t_r_ei) t_r_ei -= MCMin(t_r_ei - si, ei - si); MCactivefield -> stopcomposition(True, False); t_was_compositing = true; } else t_was_compositing = false; // If the char count is 1 and the replacement range matches the current selection, // the char is native and the requested selection is after the char, then synthesis a // keydown/up pair. // MW-2014-06-25: [[ Bug 12370 ]] If the char is ascii then map appropriately so we get // the keycodes the engine expects. char_t t_char; if (!t_was_compositing && p_char_count == 1 && MCUnicodeMapToNative(p_chars, 1, t_char) && p_selection_range . offset == p_replace_range . offset + 1 && p_selection_range . length == 0) { int32_t t_s_si, t_s_ei; MCactivefield -> selectedmark(False, t_s_si, t_s_ei, False); if (t_s_si == t_r_si && t_s_ei == t_r_ei) { // SN-2014-09-15: [[ Bug 13423 ]] Send the messages for all the characters typed while (s_pending_key_down != nil) { // MW-2014-04-15: [[ Bug 12086 ]] Pass the keycode from the last event that was // passed to the IME. MCAutoStringRef t_mapped_char; MCPlatformKeyCode t_mapped_key_code; map_key_to_engine(s_pending_key_down -> key_code, s_pending_key_down -> mapped_codepoint, s_pending_key_down -> unmapped_codepoint, t_mapped_key_code, &t_mapped_char); // SN-2014-11-03: [[ Bug 13832 ]] Enqueue the event, instead of firing it now (we are still in the NSApplication's keyDown). // PM-2015-05-15: [[ Bug 15372]] call MCKeyMessageAppend before wkdown to prevent a crash if 'wait with messages' is used (since s_pending_key_down might become nil after wkdown MCKeyMessageAppend(s_pending_key_up, s_pending_key_down -> key_code, s_pending_key_down -> mapped_codepoint, s_pending_key_down -> unmapped_codepoint); MCdispatcher -> wkdown(p_window, *t_mapped_char, t_mapped_key_code); MCKeyMessageNext(s_pending_key_down); } return; } } } else { if (p_char_count == 0) MCactivefield -> stopcomposition(True, False); else { int4 si, ei; if (MCactivefield -> getcompositionrange(si, ei)) { if (si < t_r_si) t_r_si -= MCMin(t_r_si - si, ei - si); if (si < t_r_ei) t_r_ei -= MCMin(t_r_ei - si, ei - si); MCactivefield -> stopcomposition(True, False); } } } // SN-2014-09-14: [[ Bug 13423 ]] MCPlatformHandleRawKeyDown gets the US mac layout key, without any modifier included. // We need to update the elements: // if the key pressed leads to an actual char: // this wrong key is replaced by this new 'combined' char // if the key pressed fails to generate a char: // this wrong key is replaced by the dead-key char // SN-2015-04-10: [[ Bug 14205 ]] When using the dictation, there is no // pending key down, but the composition was still on though. // SN-2015-06-23: [[ Bug 3537 ]] We should not cast p_char as a uint1 if it // is not a native char. uint1 t_char[2]; bool t_is_native_char; t_is_native_char = MCUnicodeMapToNative(p_chars, 1, t_char[0]); t_char[1] = 0; if (t_was_compositing && s_pending_key_down && t_is_native_char) { s_pending_key_down -> key_code = (uint1)*t_char; s_pending_key_down -> mapped_codepoint = (uint1)*t_char; s_pending_key_down -> unmapped_codepoint = (uint1)*t_char; // SN-2015-05-18: [[ Bug 15385 ]] Enqueue the first char in the sequence // here - that will be the same as keyDown. // SN-2015-06-23: [[ Bug 3537 ]] In this only case, we don't want this // nativised char to be mapped again in MCPlatformHandleKeyUp. MCKeyMessageAppend(s_pending_key_up, (uint1)*t_char, (uint1)*t_char, (uint1)*t_char, false); } // Set the text. MCactivefield -> seltext(t_r_si, t_r_ei, False); if (p_mark) MCactivefield -> startcomposition(); // SN-2014-09-15: [[ Bug 13423 ]] If the character typed is not Unicode and follows a dead key character, then we send // [Raw]KeyDown/Up and remove the first character from the sequence of keys typed. // If the character successfully combined with the dead char before it in a native char, we don't use finsert // Otherwise, we have the dead char in p_chars, we need to remove the one stored first in the sequence MCAutoStringRef t_string; // SN-2015-01-20: [[ Bug 14406 ]] If we have a series of pending keys, we have two possibilities: // - typing IME characters: the characters are native, so we use the finsertnew // - typing dead characters: the character, if we arrive here, is > 127 // SN-2015-04-13: [[ Bug 14205 ]] Ensure that s_pending_key_down is not nil if (*p_chars > 127 && s_pending_key_down && s_pending_key_down -> next && t_is_native_char) { MCStringCreateWithNativeChars((const char_t *)t_char, 1, &t_string); MCdispatcher -> wkdown(p_window, *t_string, *t_char); MCKeyMessageNext(s_pending_key_down); } else { MCStringCreateWithChars(p_chars, p_char_count, &t_string); // SN-2014-12-05: [[ Bug 14162 ]] In case the character is a Unicode alphanumeric char, // then that's not a combining char - and it deserves its (raw)Key(Down|Up) messages uint32_t t_codepoint; t_codepoint = MCStringGetCodepointAtIndex(*t_string, 0); // SN-2015-05-18: [[ Bug 3537 ]] Use p_mark to determine whether we are // in an IME state // SN-2015-05-05: [[ Bug 15305 ]] Check that s_pending_key_down is not // nil before trying to use it, and use IME only if p_mark says so. if (s_pending_key_down && !p_mark) { MCAutoStringRef t_mapped_char; MCPlatformKeyCode t_mapped_key_code; map_key_to_engine(s_pending_key_down -> key_code, s_pending_key_down -> mapped_codepoint, s_pending_key_down -> unmapped_codepoint, t_mapped_key_code, &t_mapped_char); MCdispatcher -> wkdown(p_window, *t_string, *p_chars); // SN-2015-05-18: [[ Bug 3537 ]] If we were compositing, then we want // to send the same message for keyUp and keyDown - which might be // seeveral character-long if (t_was_compositing) MCdispatcher -> wkup(p_window, *t_string, *p_chars); else MCKeyMessageAppend(s_pending_key_up, *p_chars, s_pending_key_down -> mapped_codepoint, s_pending_key_down -> unmapped_codepoint); MCKeyMessageNext(s_pending_key_down); } else MCactivefield -> finsertnew(FT_IMEINSERT, *t_string, True); } // And update the selection range. int32_t t_s_si, t_s_ei; t_s_si = 0; t_s_ei = INT32_MAX; MCactivefield -> resolvechars(0, t_s_si, t_s_ei, p_selection_range . offset, p_selection_range . length); MCactivefield -> setcompositioncursoroffset(t_s_si - t_r_si); MCactivefield -> seltext(t_s_si, t_s_ei, True); // SN-2014-12-04: [[ Bug 14152 ]] Locking the screen here doesn't allow the screen to refresh after // text input, inside an MCWait loop // MCRedrawUnlockScreen(); } static void synthesize_key_press(MCPlatformWindowRef p_window, char p_char, KeySym p_sym) { MCAutoStringRef t_key; /* UNCHECKED */ MCStringCreateWithNativeChars((char_t*)&p_char, 1, &t_key); MCdispatcher -> wkdown(p_window, *t_key, p_sym); MCdispatcher -> wkup(p_window, *t_key, p_sym); } static void synthesize_move_with_shift(MCField *p_field, Field_translations p_action) { uint2 t_modifier_state; t_modifier_state = MCmodifierstate; MCmodifierstate = MS_SHIFT; p_field -> fmove(p_action, nil, 0); MCmodifierstate = t_modifier_state; } // This is probably never called now as we catch keyDown's for IME input // and dispatch them rather than go through actions. void MCPlatformHandleTextInputAction(MCPlatformWindowRef p_window, MCPlatformTextInputAction p_action) { if (!MCactivefield) return; switch(p_action) { case kMCPlatformTextInputActionCapitalizeWord: break; case kMCPlatformTextInputActionChangeCaseOfLetter: break; case kMCPlatformTextInputActionDeleteBackward: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELBCHAR, nil, 0); break; case kMCPlatformTextInputActionDeleteBackwardByDecomposingPreviousCharacter: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELBSUBCHAR, nil, 0); break; case kMCPlatformTextInputActionDeleteForward: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELFCHAR, nil, 0); break; case kMCPlatformTextInputActionDeleteToBeginningOfLine: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELBOL, nil, 0); break; case kMCPlatformTextInputActionDeleteToBeginningOfParagraph: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELBOP, nil, 0); break; case kMCPlatformTextInputActionDeleteToEndOfLine: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELEOL, nil, 0); break; case kMCPlatformTextInputActionDeleteToEndOfParagraph: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELEOP, nil, 0); break; case kMCPlatformTextInputActionDeleteWordBackward: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELBWORD, nil, 0); break; case kMCPlatformTextInputActionDeleteWordForward: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fdel(FT_DELFWORD, nil, 0); break; case kMCPlatformTextInputActionInsertBacktab: break; case kMCPlatformTextInputActionInsertContainerBreak: break; case kMCPlatformTextInputActionInsertLineBreak: synthesize_key_press(p_window, 11, 0); break; case kMCPlatformTextInputActionInsertNewline: synthesize_key_press(p_window, '\n', XK_Return); break; case kMCPlatformTextInputActionInsertParagraphSeparator: synthesize_key_press(p_window, '\n', XK_Return); break; case kMCPlatformTextInputActionInsertTab: synthesize_key_press(p_window, '\t', XK_Tab); break; case kMCPlatformTextInputActionLowercaseWord: break; case kMCPlatformTextInputActionMoveBackward: MCactivefield -> fmove(FT_BACKCHAR, nil, 0); break; case kMCPlatformTextInputActionMoveBackwardAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_BACKCHAR); break; case kMCPlatformTextInputActionMoveParagraphForwardAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_FORWARDPARA); break; case kMCPlatformTextInputActionMoveParagraphBackwardAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_BACKPARA); break; case kMCPlatformTextInputActionMoveToBeginningOfDocumentAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_BOF); break; case kMCPlatformTextInputActionMoveToEndOfDocumentAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_EOF); break; case kMCPlatformTextInputActionMoveToBeginningOfLineAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_BOL); break; case kMCPlatformTextInputActionMoveToEndOfLineAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_EOL); break; case kMCPlatformTextInputActionMoveToBeginningOfParagraphAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_BOP); break; case kMCPlatformTextInputActionMoveToEndOfParagraphAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_EOP); break; case kMCPlatformTextInputActionMoveToLeftEndOfLine: MCactivefield -> fmove(FT_BOL, nil, 0); break; case kMCPlatformTextInputActionMoveToLeftEndOfLineAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_BOL); break; case kMCPlatformTextInputActionMoveToRightEndOfLine: MCactivefield -> fmove(FT_EOL, nil, 0); break; case kMCPlatformTextInputActionMoveToRightEndOfLineAndModfySelection: synthesize_move_with_shift(MCactivefield, FT_EOL); break; case kMCPlatformTextInputActionMoveDown: MCactivefield -> fmove(FT_DOWN, nil, 0); break; case kMCPlatformTextInputActionMoveDownAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_DOWN); break; case kMCPlatformTextInputActionMoveForward: MCactivefield -> fmove(FT_FORWARDCHAR, nil, 0); break; case kMCPlatformTextInputActionMoveForwardAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_FORWARDCHAR); break; case kMCPlatformTextInputActionMoveLeft: MCactivefield -> fmove(FT_LEFTCHAR, nil, 0); break; case kMCPlatformTextInputActionMoveLeftAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_LEFTCHAR); break; case kMCPlatformTextInputActionMoveRight: MCactivefield -> fmove(FT_RIGHTCHAR, nil, 0); break; case kMCPlatformTextInputActionMoveRightAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_RIGHTCHAR); break; case kMCPlatformTextInputActionMoveToBeginningOfDocument: MCactivefield -> fmove(FT_BOF, nil, 0); break; case kMCPlatformTextInputActionMoveToBeginningOfLine: MCactivefield -> fmove(FT_BOL, nil, 0); break; case kMCPlatformTextInputActionMoveToBeginningOfParagraph: MCactivefield -> fmove(FT_BOP, nil, 0); break; case kMCPlatformTextInputActionMoveToEndOfDocument: MCactivefield -> fmove(FT_EOF, nil, 0); break; case kMCPlatformTextInputActionMoveToEndOfLine: MCactivefield -> fmove(FT_EOL, nil, 0); break; case kMCPlatformTextInputActionMoveToEndOfParagraph: MCactivefield -> fmove(FT_EOF, nil, 0); break; case kMCPlatformTextInputActionMoveUp: MCactivefield -> fmove(FT_UP, nil, 0); break; case kMCPlatformTextInputActionMoveUpAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_UP); break; case kMCPlatformTextInputActionMoveWordBackward: MCactivefield -> fmove(FT_BACKWORD, nil, 0); break; case kMCPlatformTextInputActionMoveWordBackwardAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_BACKWORD); break; case kMCPlatformTextInputActionMoveWordForward: MCactivefield -> fmove(FT_FORWARDWORD, nil, 0); break; case kMCPlatformTextInputActionMoveWordForwardAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_FORWARDWORD); break; case kMCPlatformTextInputActionMoveWordLeft: MCactivefield -> fmove(FT_LEFTWORD, nil, 0); break; case kMCPlatformTextInputActionMoveWordLeftAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_LEFTWORD); break; case kMCPlatformTextInputActionMoveWordRight: MCactivefield -> fmove(FT_RIGHTWORD, nil, 0); break; case kMCPlatformTextInputActionMoveWordRightAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_RIGHTWORD); break; case kMCPlatformTextInputActionPageUp: MCactivefield -> fmove(FT_PAGEUP, nil, 0); break; case kMCPlatformTextInputActionPageUpAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_PAGEUP); break; case kMCPlatformTextInputActionPageDown: MCactivefield -> fmove(FT_PAGEDOWN, nil, 0); break; case kMCPlatformTextInputActionPageDownAndModifySelection: synthesize_move_with_shift(MCactivefield, FT_PAGEDOWN); break; case kMCPlatformTextInputActionScrollToBeginningOfDocument: MCactivefield -> fscroll(FT_SCROLLTOP, nil, 0); break; case kMCPlatformTextInputActionScrollToEndOfDocument: MCactivefield -> fscroll(FT_SCROLLBOTTOM, nil, 0); break; case kMCPlatformTextInputActionScrollLineUp: MCactivefield -> fscroll(FT_SCROLLUP, nil, 0); break; case kMCPlatformTextInputActionScrollLineDown: MCactivefield -> fscroll(FT_SCROLLDOWN, nil, 0); break; case kMCPlatformTextInputActionScrollPageUp: MCactivefield -> fscroll(FT_SCROLLPAGEUP, nil, 0); break; case kMCPlatformTextInputActionScrollPageDown: MCactivefield -> fscroll(FT_SCROLLPAGEDOWN, nil, 0); break; case kMCPlatformTextInputActionSelectAll: break; case kMCPlatformTextInputActionSelectLine: break; case kMCPlatformTextInputActionSelectParagraph: break; case kMCPlatformTextInputActionSelectWord: break; case kMCPlatformTextInputActionTranspose: break; case kMCPlatformTextInputActionTransposeWords: break; case kMCPlatformTextInputActionUppercaseWord: break; case kMCPlatformTextInputActionYank: break; case kMCPlatformTextInputActionCut: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fcut(FT_CUT, nil, 0); break; case kMCPlatformTextInputActionCopy: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fcopy(FT_COPY, nil, 0); break; case kMCPlatformTextInputActionPaste: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fpaste(FT_PASTE, nil, 0); break; case kMCPlatformTextInputActionUndo: if (!MCactivefield -> getflag(F_LOCK_TEXT)) MCactivefield -> fundo(FT_UNDO, nil, 0); break; case kMCPlatformTextInputActionRedo: break; case kMCPlatformTextInputActionDelete: MCactivefield -> deleteselection(False); break; }; } #endif // FEATURE_PLATFORM_WINDOW //////////////////////////////////////////////////////////////////////////////// #if defined(FEATURE_PLATFORM_PLAYER) static MCPlayer *find_player(MCPlatformPlayerRef p_player) { for(MCPlayerHandle t_player = MCplayers; t_player.IsValid(); t_player = t_player -> getnextplayer()) { if (t_player -> getplatformplayer() == p_player) return t_player; } return nil; } void MCPlatformHandlePlayerFrameChanged(MCPlatformPlayerRef p_player) { MCPlayer *t_player; t_player = find_player(p_player); if (t_player == nil) return; t_player -> layer_redrawall(); MCPlatformBreakWait(); } void MCPlatformHandlePlayerMarkerChanged(MCPlatformPlayerRef p_player, MCPlatformPlayerDuration p_time) { MCPlayer *t_player; t_player = find_player(p_player); if (t_player == nil) return; t_player -> markerchanged(p_time); } void MCPlatformHandlePlayerCurrentTimeChanged(MCPlatformPlayerRef p_player) { MCPlayer *t_player; t_player = find_player(p_player); if (t_player == nil) return; t_player -> currenttimechanged(); } void MCPlatformHandlePlayerFinished(MCPlatformPlayerRef p_player) { MCPlayer *t_player; t_player = find_player(p_player); if (t_player == nil) return; // PM-2016-01-18: [[ Bug 16737 ]] Make sure controller is updated and playstopped msg is sent immediately MCPlatformBreakWait(); t_player -> layer_redrawall(); t_player -> moviefinished(); } void MCPlatformHandlePlayerBufferUpdated(MCPlatformPlayerRef p_player) { MCPlayer *t_player; t_player = find_player(p_player); if (t_player == nil) return; // Make sure download progress is updated MCPlatformBreakWait(); t_player -> redrawcontroller(); // PM-2014-11-20: [[ Bug 14035 ]] Make sure movie frames are shown t_player -> layer_redrawall(); } #endif // FEATURE_PLATFORM_PLAYER //////////////////////////////////////////////////////////////////////////////// #if defined(FEATURE_PLATFORM_AUDIO) void MCPlatformHandleSoundFinished(MCPlatformSoundRef p_sound) { if (MCacptr) { MCscreen -> addtimer(MCacptr, MCM_internal, 0); // PM-2014-12-09: [[ Bug 14176 ]] Release and nullify the sound once it is done MCacptr->stop(True); // PM-2014-12-22: [[ Bug 14269 ]] Nullify MCacptr to prevent looping when play audioclip is followed by wait until the sound is done MCacptr = nil; } } #endif ////////////////////////////////////////////////////////////////////////////////