/**************************************************************************** Copyright (c) 2018 Xiamen Yaji Software Co., Ltd. http://www.cocos2d-x.org Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ****************************************************************************/ #include "EditBox.h" #include "platform/CCApplication.h" #include "platform/ios/CCEAGLView-ios.h" #include "cocos/scripting/js-bindings/jswrapper/SeApi.h" #include "cocos/scripting/js-bindings/manual/jsb_global.h" #import #import #define TEXT_LINE_HEIGHT 40 #define TEXT_VIEW_MAX_LINE_SHOWN 3 #define BUTTON_HIGHT (TEXT_LINE_HEIGHT - 2) #define BUTTON_WIDTH 60 #define TO_TEXT_VIEW(textinput) ((UITextView*)textinput) #define TO_TEXT_FIELD(textinput) ((UITextField*)textinput) /************************************************************************* Inner class declarations. ************************************************************************/ // MARK: class declaration @interface ButtonHandler : NSObject -(IBAction) buttonTapped:(UIButton *)button; @end @interface KeyboardEventHandler : NSObject -(void)keyboardWillShow: (NSNotification*) notification; -(void)keyboardWillHide: (NSNotification*) notification; @end @interface TextFieldDelegate : NSObject - (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string; - (void)textFieldDidChange:(UITextField *)textField; - (BOOL)textFieldShouldReturn:(UITextField *)textField; @end @interface TextViewDelegate : NSObject - (BOOL) textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text; - (void) textViewDidChange:(UITextView *)textView; @end /************************************************************************* Global variables and functions. ************************************************************************/ // MARK: global variables and functions namespace { bool g_isMultiline = false; bool g_confirmHold = false; int g_maxLength = INT_MAX; KeyboardEventHandler* g_keyboardHandler = nil; // "#1fa014", a color of dark green, was used for confirm button background static UIColor* g_darkGreen = [UIColor colorWithRed:31/255.0 green:160/255.0 blue:20/255.0 alpha:0.8]; UITextField* g_textField = nil; TextFieldDelegate* g_textFieldDelegate = nil; UIButton* g_textFieldConfirmButton = nil; ButtonHandler* g_textFieldConfirmButtonHandler = nil; UITextView* g_textView = nil; TextViewDelegate* g_textViewDelegate = nil; UIButton* g_textViewConfirmButton = nil; ButtonHandler* g_textViewConfirmButtonHander = nil; UIView* getCurrentView() { if (g_isMultiline) return g_textView; else return g_textField; } NSString* getCurrentText() { if (g_isMultiline) return g_textView.text; else return g_textField.text; } void setText(NSString* text) { if (g_isMultiline) g_textView.text = text; else g_textField.text = text; } se::Value textInputCallback; void getTextInputCallback() { if (! textInputCallback.isUndefined()) return; auto global = se::ScriptEngine::getInstance()->getGlobalObject(); se::Value jsbVal; if (global->getProperty("jsb", &jsbVal) && jsbVal.isObject()) { jsbVal.toObject()->getProperty("onTextInput", &textInputCallback); // free globle se::Value before ScriptEngine clean up se::ScriptEngine::getInstance()->addBeforeCleanupHook([](){ textInputCallback.setUndefined(); }); } } void callJSFunc(const std::string& type, const std::string& text) { getTextInputCallback(); se::AutoHandleScope scope; se::ValueArray args; args.push_back(se::Value(type)); args.push_back(se::Value(text)); textInputCallback.toObject()->call(args, nullptr); } int getTextInputHeight() { if (g_isMultiline) return TEXT_LINE_HEIGHT * TEXT_VIEW_MAX_LINE_SHOWN; else return TEXT_LINE_HEIGHT; } void createButton(UIButton** button, ButtonHandler** buttonHandler, const CGRect& viewRect, const std::string& title) { ButtonHandler *btnHandler = [[ButtonHandler alloc] init]; UIButton* btn = [UIButton buttonWithType:UIButtonTypeRoundedRect]; [btn addTarget:btnHandler action:@selector(buttonTapped:) forControlEvents:UIControlEventTouchUpInside]; btn.frame = CGRectMake(0, 0, BUTTON_WIDTH, BUTTON_HIGHT); btn.backgroundColor = g_darkGreen; [btn setTitle: [NSString stringWithUTF8String:title.c_str()] forState:UIControlStateNormal]; [btn setTitleColor: [UIColor whiteColor] forState:UIControlStateNormal]; *button = btn; *buttonHandler = btnHandler; } void setTexFiledKeyboardType(UITextField* textField, const std::string& inputType) { if (0 == inputType.compare("password")) { textField.secureTextEntry = TRUE; textField.keyboardType = UIKeyboardTypeDefault; } else { textField.secureTextEntry = FALSE; if (0 == inputType.compare("email")) textField.keyboardType = UIKeyboardTypeEmailAddress; else if (0 == inputType.compare("number")) textField.keyboardType = UIKeyboardTypeDecimalPad; else if (0 == inputType.compare("url")) textField.keyboardType = UIKeyboardTypeURL; else if (0 == inputType.compare("text")) textField.keyboardType = UIKeyboardTypeDefault; } } void setTextFieldReturnType(UITextField* textField, const std::string& returnType) { if (0 == returnType.compare("done")) textField.returnKeyType = UIReturnKeyDone; else if (0 == returnType.compare("next")) textField.returnKeyType = UIReturnKeyNext; else if (0 == returnType.compare("search")) textField.returnKeyType = UIReturnKeySearch; else if (0 == returnType.compare("go")) textField.returnKeyType = UIReturnKeyGo; else if (0 == returnType.compare("send")) textField.returnKeyType = UIReturnKeySend; } NSString* getConfirmButtonTitle(const std::string& returnType) { NSString* titleKey = [NSString stringWithUTF8String: returnType.c_str()]; return NSLocalizedString(titleKey, nil); // get i18n string to be the title } void initTextField(const CGRect& rect, const cocos2d::EditBox::ShowInfo& showInfo) { if (! g_textField) { g_textField = [[UITextField alloc] initWithFrame:rect]; g_textField.textColor = [UIColor blackColor]; g_textField.backgroundColor = [UIColor whiteColor]; [g_textField setBorderStyle:UITextBorderStyleLine]; g_textField.backgroundColor = [UIColor whiteColor]; g_textFieldDelegate = [[TextFieldDelegate alloc] init]; g_textField.delegate = g_textFieldDelegate; // Assign the overlay button to a stored text field createButton(&g_textFieldConfirmButton, &g_textFieldConfirmButtonHandler, rect, showInfo.confirmType); g_textField.rightView = g_textFieldConfirmButton; g_textField.rightViewMode = UITextFieldViewModeAlways; [g_textField addTarget:g_textFieldDelegate action:@selector(textFieldDidChange:) forControlEvents:UIControlEventEditingChanged]; } g_textField.frame = rect; setTextFieldReturnType(g_textField, showInfo.confirmType); setTexFiledKeyboardType(g_textField, showInfo.inputType); g_textField.text = [NSString stringWithUTF8String: showInfo.defaultValue.c_str()]; [g_textFieldConfirmButton setTitle:getConfirmButtonTitle(showInfo.confirmType) forState:UIControlStateNormal]; } void initTextView(const CGRect& viewRect, const CGRect& btnRect, const cocos2d::EditBox::ShowInfo& showInfo) { if (!g_textView) { g_textView = [[UITextView alloc] initWithFrame:btnRect]; g_textView.textColor = [UIColor blackColor]; g_textView.backgroundColor = [UIColor whiteColor]; g_textViewDelegate = [[TextViewDelegate alloc] init]; g_textView.delegate = g_textViewDelegate; createButton(&g_textViewConfirmButton, &g_textViewConfirmButtonHander, btnRect, showInfo.confirmType); g_textViewConfirmButton.frame = CGRectMake(viewRect.size.width - BUTTON_WIDTH, 0, BUTTON_WIDTH, BUTTON_HIGHT); [g_textView addSubview:g_textViewConfirmButton]; } g_textView.frame = btnRect; g_textView.text = [NSString stringWithUTF8String: showInfo.defaultValue.c_str()]; [g_textViewConfirmButton setTitle:getConfirmButtonTitle(showInfo.confirmType) forState:UIControlStateNormal]; } CGRect getSafeAreaRect() { UIView* view = (UIView*)cocos2d::Application::getInstance()->getView(); CGRect viewRect = view.frame; // safeAreaInsets is avaible since iOS 11. if (@available(iOS 11.0, *)) { auto safeAreaInsets = view.safeAreaInsets; UIInterfaceOrientation sataus = [UIApplication sharedApplication].statusBarOrientation; if (UIInterfaceOrientationLandscapeLeft == sataus) { viewRect.origin.x = 0; viewRect.size.width -= safeAreaInsets.right; } else { viewRect.origin.x += safeAreaInsets.left; viewRect.size.width -= safeAreaInsets.left; } } return viewRect; } void addTextInput(const cocos2d::EditBox::ShowInfo& showInfo) { auto safeAreaRect = getSafeAreaRect(); int height = getTextInputHeight(); CGRect rect = CGRectMake(safeAreaRect.origin.x, safeAreaRect.size.height - height, safeAreaRect.size.width, height); if (showInfo.isMultiline) initTextView(safeAreaRect, rect, showInfo); else initTextField(rect, showInfo); UIView* textInput = getCurrentView(); UIView* view = (UIView*)cocos2d::Application::getInstance()->getView(); [view addSubview:textInput]; [textInput becomeFirstResponder]; } void addKeyboardEventLisnters() { if (!g_keyboardHandler) g_keyboardHandler = [[KeyboardEventHandler alloc] init]; [[NSNotificationCenter defaultCenter] addObserver:g_keyboardHandler selector:@selector(keyboardWillShow:) name:UIKeyboardWillShowNotification object:nil]; [[NSNotificationCenter defaultCenter] addObserver:g_keyboardHandler selector:@selector(keyboardWillHide:) name:UIKeyboardWillHideNotification object:nil]; } void removeKeyboardEventLisnters() { if (!g_keyboardHandler) return; [[NSNotificationCenter defaultCenter] removeObserver:g_keyboardHandler]; } } /************************************************************************* Class implementations. ************************************************************************/ // MARK: class implementation @implementation KeyboardEventHandler -(void)keyboardWillShow: (NSNotification*) notification { UIView* textView = getCurrentView(); if (!textView) return; NSDictionary* keyboardInfo = [notification userInfo]; NSValue* keyboardFrame = [keyboardInfo objectForKey:UIKeyboardFrameEndUserInfoKey]; CGSize kbSize = [keyboardFrame CGRectValue].size; int textHeight = getTextInputHeight(); CGRect screenRect = getSafeAreaRect(); textView.frame = CGRectMake(screenRect.origin.x, screenRect.size.height - textHeight - kbSize.height, screenRect.size.width, textHeight); } -(void)keyboardWillHide: (NSNotification*) notification { NSDictionary *info = [notification userInfo]; CGRect beginKeyboardRect = [[info objectForKey:UIKeyboardFrameBeginUserInfoKey] CGRectValue]; CGRect endKeyboardRect = [[info objectForKey:UIKeyboardFrameEndUserInfoKey] CGRectValue]; CGFloat yOffset = endKeyboardRect.origin.y - beginKeyboardRect.origin.y; if (yOffset <= 0) { cocos2d::EditBox::complete(); cocos2d::EditBox::hide(); } } @end @implementation TextFieldDelegate - (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string { // REFINE: check length limit before text changed return YES; } - (void)textFieldDidChange:(UITextField *)textField { if (textField.markedTextRange != nil) return; // check length limit after text changed, a little rude if (textField.text.length > g_maxLength) { NSRange rangeIndex = [textField.text rangeOfComposedCharacterSequenceAtIndex:g_maxLength]; auto newText = [textField.text substringToIndex:rangeIndex.location]; dispatch_async(dispatch_get_main_queue(), ^{ textField.text = newText; }); } callJSFunc("input", [textField.text UTF8String]); } -(BOOL) textFieldShouldReturn:(UITextField *)textField { cocos2d::EditBox::complete(); return YES; } @end @implementation ButtonHandler -(IBAction) buttonTapped:(UIButton *)button { const std::string text([getCurrentText() UTF8String]); callJSFunc("confirm", text); if (!g_confirmHold) cocos2d::EditBox::complete(); } @end @implementation TextViewDelegate - (BOOL) textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text { // REFINE: check length limit before text changed return YES; } - (void)textViewDidChange:(UITextView *)textView { if (textView.markedTextRange != nil) return; // check length limit after text changed, a little rude if (textView.text.length > g_maxLength) { auto newText = [textView.text substringToIndex:g_maxLength]; // fix undo crash dispatch_async(dispatch_get_main_queue(), ^{ textView.text = newText; }); } callJSFunc("input", [textView.text UTF8String]); } @end /************************************************************************* Implementation of EditBox. ************************************************************************/ // MARK: EditBox NS_CC_BEGIN void EditBox::show(const cocos2d::EditBox::ShowInfo& showInfo) { // Should initialize them at first. g_maxLength = showInfo.maxLength; g_isMultiline = showInfo.isMultiline; g_confirmHold = showInfo.confirmHold; [(CCEAGLView*)cocos2d::Application::getInstance()->getView() setPreventTouchEvent:true]; addKeyboardEventLisnters(); addTextInput(showInfo); } void EditBox::hide() { removeKeyboardEventLisnters(); UIView* view = getCurrentView(); if (view) { [view removeFromSuperview]; [view resignFirstResponder]; } [(CCEAGLView*)cocos2d::Application::getInstance()->getView() setPreventTouchEvent:false]; } void EditBox::updateRect(int x, int y, int width, int height) { // not supported ... } void EditBox::complete() { NSString* text = getCurrentText(); callJSFunc("complete", [text UTF8String]); EditBox::hide(); } NS_CC_END