সম্পাদনা পাঠ্য, বাইরের স্পর্শে পরিষ্কার ফোকাস


104

আমার বিন্যাসে রয়েছে ListView, SurfaceViewএবং EditText। আমি যখন ক্লিক করি তখন EditTextএটি ফোকাস পায় এবং অন-স্ক্রীন কীবোর্ড পপ আপ হয়। আমি যখন বাইরের কোথাও ক্লিক করি তখন EditTextতার ফোকাস থাকে (এটি হওয়া উচিত নয়)। আমি অনুমান করি যে আমি OnTouchListenerবিন্যাসে অন্যান্য মতামতগুলিতে সেট আপ করতে এবং ম্যানুয়ালি EditTextএর ফোকাসটি সাফ করতে পারি । তবে খুব হ্যাকিশ মনে হচ্ছে ...

অন্যান্য লেআউটেও আমার একই অবস্থা রয়েছে - বিভিন্ন ধরণের আইটেমগুলির সাথে তালিকার ভিউ, যার মধ্যে কিছু রয়েছে EditText। আমি উপরে যেমন লিখেছি তারা এগুলি আচরণ করে।

কাজটি হ'ল EditTextফোকাস হারাতে যখন ব্যবহারকারী এর বাইরের কিছু স্পর্শ করে।

আমি এখানে অনুরূপ প্রশ্নগুলি দেখেছি, তবে কোনও সমাধান খুঁজে পাইনি ...

উত্তর:


68

আমি এই সমস্ত সমাধান চেষ্টা করেছিলাম। edc598's কাজ করার সবচেয়ে নিকটতম, তবে স্পর্শ ইভেন্টগুলি Viewবিন্যাসে থাকা অন্যান্যগুলির উপর ট্রিগার দেয় না । কারও যদি এই আচরণের প্রয়োজন হয় তবে আমি এই কাজটি শেষ করেছি:

লেআউটে সর্বশেষ হিসাবে আমি একটি (অদৃশ্য) FrameLayoutনামে পরিচিত (যা অদৃশ্য) তৈরি করেছিView যাতে এটি সমস্ত কিছুকে ওভারলে করে ( সম্পাদনা: আপনাকেওRelativeLayout প্যারেন্ট লেআউট হিসাবে ব্যবহার করতে হবে এবং টাচআইন্টারসেপ্টরের fill_parent বৈশিষ্ট্যগুলি দিতে হবে )। তারপরে আমি এটি স্পর্শকে বাধা দেওয়ার জন্য এবং স্পর্শটির উপরে ছিল কিনা তা নির্ধারণ করতে ব্যবহার করেছি EditText:

FrameLayout touchInterceptor = (FrameLayout)findViewById(R.id.touchInterceptor);
touchInterceptor.setOnTouchListener(new OnTouchListener() {
    @Override
    public boolean onTouch(View v, MotionEvent event) {
        if (event.getAction() == MotionEvent.ACTION_DOWN) {
            if (mEditText.isFocused()) {
                Rect outRect = new Rect();
                mEditText.getGlobalVisibleRect(outRect);
                if (!outRect.contains((int)event.getRawX(), (int)event.getRawY())) {
                    mEditText.clearFocus();
                    InputMethodManager imm = (InputMethodManager) v.getContext().getSystemService(Context.INPUT_METHOD_SERVICE); 
                    imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
                }
            }
        }
        return false;
    }
});

টাচ হ্যান্ডলিংয়ের মধ্যে পড়তে দিতে মিথ্যা ফিরুন

এটি হ্যাকি, তবে এটিই আমার পক্ষে কাজ করেছে।


21
আপনার ক্রিয়াকলাপে ডিসপ্যাচ টাচইভেন্ট (মোশনইভেন্ট ইভ) ওভাররাইড করে অতিরিক্ত লেআউট যুক্ত না করে আপনি একই জিনিসটি করতে পারেন।
পিসান

ক্লিয়ারফোকস () ইঙ্গিতের জন্য ধন্যবাদ, এটি অনুসন্ধান ভিউতেও আমার জন্য কাজ করেছে
জিমি ইলেন্লোয়া

4
আমি নীচে @ zMan এর উত্তরটি ইঙ্গিত করতে চাই। এটি এর উপর ভিত্তি করে তৈরি হয়েছে তবে স্ট্যাকওভারফ্লো.com
বয়

এছাড়াও, যদি কেউ এমন পরিস্থিতির মুখোমুখি হয় যে কীবোর্ডটি লুকিয়ে নেই তবে ক্লিয়ারিং ফোকাস করছে, প্রথমে hideSoftInputFromWindow()স্পষ্ট ফোকাসটি আহ্বান করুন
আহমেট গোকডেই

তবে এটি অবশ্যই অ্যান্ড্রয়েড যুক্ত করতে হবে: টপ সর্বাধিক অভিভাবকের কাছে ফোকাসেবলআইনটচমড = "সত্য"।
জি কে মোহাম্মদ ইমন

240

কেনের উত্তরের ভিত্তিতে বিল্ডিং, এখানে সর্বাধিক মডুলার অনুলিপি-এবং-পেস্ট সমাধান।

কোনও এক্সএমএল দরকার নেই।

এটিকে আপনার ক্রিয়াকলাপে রাখুন এবং এটি সেই ক্রিয়াকলাপের মধ্যে থাকা টুকরোগুলির মধ্যে থাকা সমস্ত এডিটেক্সটকে প্রযোজ্য হবে।

@Override
public boolean dispatchTouchEvent(MotionEvent event) {
    if (event.getAction() == MotionEvent.ACTION_DOWN) {
        View v = getCurrentFocus();
        if ( v instanceof EditText) {
            Rect outRect = new Rect();
            v.getGlobalVisibleRect(outRect);
            if (!outRect.contains((int)event.getRawX(), (int)event.getRawY())) {
                v.clearFocus();
                InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
                imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
            }
        }
    }
    return super.dispatchTouchEvent( event );
}

12
ভাল লাগছে। তবুও আমার একটি সমস্যা হচ্ছে, আমার সমস্ত সম্পাদনা পাঠ্যটি একটি স্ক্রোল ভিউয়ের মধ্যে রয়েছে এবং শীর্ষ সম্পাদনা পাঠ্যটিতে সর্বদা কার্সার দৃশ্যমান থাকে। এমনকি আমি যখন বাইরে ক্লিক করি তখনও ফোকাসটি হারিয়ে যায় এবং কীবোর্ড চলে যায়, তবে শীর্ষ সম্পাদনা পাঠ্যে কার্সারটি এখনও দৃশ্যমান।
বৈভব গুপ্ত 20'15

4
সেরা সমাধানটি আমি পেলাম !! এর আগে আমি @ ওভাররাইড পাবলিক বুলিয়ান অন টচ (ভি দেখুন, মোশনইভেন্ট ইভেন্ট) ব্যবহার করছিলাম {যদি (v.getId ()! = অনুসন্ধান_box.getId ()) {যদি (অনুসন্ধান_বক্স.হ্যাস ফোকাস ()) put ইনপুটমেথডম্যানেজার ইমি = (ইনপুটমেথডম্যানেজার) গিস্টিস্টেম সার্ভিস ( প্রসঙ্গ। INPUT_METHOD_SERVICE); im.hideSoftInputFromWindow (Search_box.getWindowToken (), 0); অনুসন্ধান_বক্স.ক্রিয়ারফোকস (); false মিথ্যা প্রত্যাবর্তন; false মিথ্যা প্রত্যাবর্তন; }
প্রদীপ কুমার কুশওয়াহা

4
একটি শট সমাধান :)
কার্তিক কোলাঞ্জি

16
এর আগে সবচেয়ে ভাল সমাধান আমি এর আগে দেখিনি! আমি আমার ছেলে এবং নাতির কাছে যাওয়ার জন্য এই কোডটি আমার টুকরোটিতে সংরক্ষণ করব।
অনুরাগী অ্যাপলিন

4
ব্যবহারকারী যদি অন্য সম্পাদনা পাঠ্যে ক্লিক করে থাকে তবে কীবোর্ডটি বন্ধ হয়ে যায় এবং সঙ্গে সঙ্গে আবার খোলে। এই আমি পরিবর্তন সমাধানের জন্য MotionEvent.ACTION_DOWNকরতে MotionEvent.ACTION_UPআপনার কোড উপর। দুর্দান্ত উত্তর বিটিডব্লিউ!
থানাসিস 11101

36

এডিটেক্সট এর প্যারেন্ট ভিউয়ের জন্য, নিম্নলিখিত 3 টি বৈশিষ্ট্যটিকে " সত্য " হতে দিন:
ক্লিকযোগ্য , ফোকাসযোগ্য , ফোকাসযোগ্য ইনটচমড

যদি কোনও ভিউ ফোকাস পেতে চায় তবে এটি অবশ্যই এই 3 শর্ত পূরণ করবে।

অ্যান্ড্রয়েড.ভিউ দেখুন :

public boolean onTouchEvent(MotionEvent event) {
    ...
    if (((viewFlags & CLICKABLE) == CLICKABLE || 
        (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {
        ...
        if (isFocusable() && isFocusableInTouchMode()
            && !isFocused()) {
                focusTaken = requestFocus();
        }
        ...
    }
    ...
}

আশা করি এটা সাহায্য করবে.


সম্পূর্ণরূপে ঠিক: গ্রেপকোড.com
file

4
ফোকাস সাফ করার জন্য অন্যান্য ফোকাসযোগ্য ভিউ অবশ্যই ফোকাসযুক্ত ভিউর পিতা হতে হবে। ফোকাস করার দৃষ্টিভঙ্গি অন্য শ্রেণিবিন্যাসে থাকলে এটি কাজ করবে না।
এক্সিএফেক্ট

প্রযুক্তিগতভাবে, আপনার বক্তব্য ভুল। পিতামাতার (clickable) *OR* (focusable *AND* focusableInTouchMode)
মতামতটি

26

এই বৈশিষ্ট্যগুলি কেবল শীর্ষ পিতামাতার মধ্যে রাখুন।

android:focusableInTouchMode="true"
android:clickable="true"
android:focusable="true" 

ধন্যবাদ মানুষ. আমি গত দুই ঘন্টা এই সমস্যাটি নিয়ে লড়াই করে যাচ্ছিলাম। শীর্ষ পিতামাতার এই লাইনগুলি যুক্ত করার কাজ করেছে।
দিশিশ শর্মি

এফওয়াইআই: যদিও এটি বেশিরভাগ ক্ষেত্রে কাজ করে, এটি বিন্যাসের অভ্যন্তরীণ উপাদানগুলির জন্য কাজ করছে না।
এস ভি মাধব রেড্ডি

17

কেনের উত্তরটি কার্যকর, তবে এটি হ্যাকি। প্যাকানরা উত্তরের মন্তব্যে ইঙ্গিত হিসাবে, একই কাজটি ডিসপ্যাচটচ এভেন্টেও করা যেতে পারে। এই সমাধানটি পরিষ্কারতর কারণ এটি একটি স্বচ্ছ, ডামি ফ্রেমলাউট সহ এক্সএমএল হ্যাক করা এড়ানো হয়। যা দেখতে দেখতে তা এখানে:

@Override
public boolean dispatchTouchEvent(MotionEvent event) {
    EditText mEditText = findViewById(R.id.mEditText);
    if (event.getAction() == MotionEvent.ACTION_DOWN) {
        View v = getCurrentFocus();
        if (mEditText.isFocused()) {
            Rect outRect = new Rect();
            mEditText.getGlobalVisibleRect(outRect);
            if (!outRect.contains((int)event.getRawX(), (int)event.getRawY())) {
                mEditText.clearFocus();
                //
                // Hide keyboard
                //
                InputMethodManager imm = (InputMethodManager) v.getContext().getSystemService(Context.INPUT_METHOD_SERVICE); 
                imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
            }
        }
    }
    return super.dispatchTouchEvent(event);
}

7

আমার জন্য কাজ করা নীচে -

1. যোগ করার পদ্ধতি android:clickable="true"এবং android:focusableInTouchMode="true"করতে parentLayoutএর EditTextঅর্থাতandroid.support.design.widget.TextInputLayout

<android.support.design.widget.TextInputLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:clickable="true"
    android:focusableInTouchMode="true">
<EditText
    android:id="@+id/employeeID"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:ems="10"
    android:inputType="number"
    android:hint="Employee ID"
    tools:layout_editor_absoluteX="-62dp"
    tools:layout_editor_absoluteY="16dp"
    android:layout_marginTop="42dp"
    android:layout_alignParentTop="true"
    android:layout_alignParentRight="true"
    android:layout_alignParentEnd="true"
    android:layout_marginRight="36dp"
    android:layout_marginEnd="36dp" />
    </android.support.design.widget.TextInputLayout>

2. উপেক্ষা dispatchTouchEventভ্রমণ ক্লাসে এবং ঢোকাতে hideKeyboard()ফাংশন

@Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        if (ev.getAction() == MotionEvent.ACTION_DOWN) {
            View view = getCurrentFocus();
            if (view != null && view instanceof EditText) {
                Rect r = new Rect();
                view.getGlobalVisibleRect(r);
                int rawX = (int)ev.getRawX();
                int rawY = (int)ev.getRawY();
                if (!r.contains(rawX, rawY)) {
                    view.clearFocus();
                }
            }
        }
        return super.dispatchTouchEvent(ev);
    }

    public void hideKeyboard(View view) {
        InputMethodManager inputMethodManager =(InputMethodManager)getSystemService(Activity.INPUT_METHOD_SERVICE);
        inputMethodManager.hideSoftInputFromWindow(view.getWindowToken(), 0);
    }

3. সম্পাদনা পাঠের জন্য যুক্ত করা হচ্ছেsetOnFocusChangeListener

EmployeeId.setOnFocusChangeListener(new View.OnFocusChangeListener() {
            @Override
            public void onFocusChange(View v, boolean hasFocus) {
                if (!hasFocus) {
                    hideKeyboard(v);
                }
            }
        });

5

আপনি সম্ভবত এই সমস্যার উত্তর ইতিমধ্যে খুঁজে পেয়েছেন তবে আমি কীভাবে এটি সমাধান করব তা খুঁজছিলাম এবং এখনও আমি ঠিক কী খুঁজছিলাম তা খুঁজে পাচ্ছি না তাই আমি বুঝতে পেরেছিলাম যে আমি এটি এখানে পোস্ট করব।

আমি যা করেছি তা হ'ল:

প্রথমে এডিটেক্সট এবং আপনার প্রোগ্রামটিতে থাকা কোনও অন্য মতামত একটি একক দর্শন দ্বারা মোড়ানো রয়েছে। আমার ক্ষেত্রে আমি সমস্ত কিছু মোড়ানোর জন্য লিনিয়ারলআউট ব্যবহার করেছি।

<LinearLayout 
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/mainLinearLayout">
 <EditText
  android:id="@+id/editText"/>
 <ImageView
  android:id="@+id/imageView"/>
 <TextView
  android:id="@+id/textView"/>
 </LinearLayout>

তারপরে আপনার কোডে আপনাকে আপনার মূল লিনিয়ারলআউটে একটি টাচ শ্রোতা সেট করতে হবে।

final EditText searchEditText = (EditText) findViewById(R.id.editText);
mainLinearLayout.setOnTouchListener(new View.OnTouchListener() {

        @Override
        public boolean onTouch(View v, MotionEvent event) {
            // TODO Auto-generated method stub
            if(searchEditText.isFocused()){
                if(event.getY() >= 72){
                    //Will only enter this if the EditText already has focus
                    //And if a touch event happens outside of the EditText
                    //Which in my case is at the top of my layout
                    //and 72 pixels long
                    searchEditText.clearFocus();
                    InputMethodManager imm = (InputMethodManager) v.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
                    imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
                }
            }
            Toast.makeText(getBaseContext(), "Clicked", Toast.LENGTH_SHORT).show();
            return false;
        }
    });

আমি আশা করি এটি কিছু লোককে সহায়তা করে। অথবা কমপক্ষে তাদের সমস্যার সমাধান শুরু করতে সহায়তা করে।


হ্যাঁ, আমিও একই সমাধানে এসেছি। তারপরে আমি ওপেনগলটিতে সি ++ এ অ্যান্ড্রয়েড ভিউ সিস্টেমের একটি উপসেট পোর্ট করেছি এবং এতে এই বৈশিষ্ট্যটি তৈরি করেছি)
নোট173

এটি সত্যিই খুব ঝরঝরে উদাহরণ। আমি এই জাতীয় কোড খুঁজছিলাম কিন্তু আমি যে সব খুঁজে পেলাম তা জটিল সমাধান। আমি XY স্থানাঙ্ক উভয়ই আরও সুনির্দিষ্ট হিসাবে ব্যবহার করেছি কারণ আমার কাছে কয়েকটি মাল্টি-লাইন সম্পাদনা পাঠও ছিল। ধন্যবাদ!
সুমিতক

3

আমি একটি আছে ListViewগঠিত EditTextমতামত। দৃশ্যে বলা হয়েছে যে এক বা একাধিক সারিতে পাঠ্য সম্পাদনা করার পরে আমাদের "সমাপ্তি" নামক বোতামটিতে ক্লিক করা উচিত। আমি ব্যবহৃত onFocusChangedউপর EditTextদৃশ্যে ভিতরে listViewকিন্তু সংরক্ষণ করা হচ্ছে ডেটা ফিনিস ক্লিক করার পর। যোগ করে সমস্যাটি সমাধান করা হয়েছিল

listView.clearFocus();

ভিতরে onClickListener"শেষ" বোতাম এবং তথ্য জন্য সফলভাবে সংরক্ষিত হয়েছে।


ধন্যবাদ আপনি আমার সময় সাশ্রয় করেছেন, আমার এই দৃশ্যটি রিসাইক্লারভিউতে ছিল এবং বোতামটি ক্লিক করার পরে শেষ সম্পাদনা পাঠ্য মানটি হারাচ্ছিলাম, শেষ সম্পাদনাটির পাঠ্য ফোকাস পরিবর্তিত হয়নি এবং তাই ফোকাস চ্যাঞ্জড বলা হয়নি। আমি এমন একটি সমাধান খুঁজে পেতে চাই যা সর্বশেষ সম্পাদনা পাঠ্যপুস্তকের বাইরে ফোকাস হারাবে, একই বোতামটি ক্লিক করুন ... এবং আপনার সমাধানটি সন্ধান করুন। এটি দুর্দান্ত কাজ করেছে!
রেহেন ফারশবাফ

3

কেবলমাত্র এর পিতামাতার দুটি বৈশিষ্ট্য হিসাবে সংজ্ঞায়িত করুন EditText:

android:clickable="true"
android:focusableInTouchMode="true"

সুতরাং ব্যবহারকারী যখন EditTextক্ষেত্রের বাইরে স্পর্শ করবেন তখন ফোকাস সরানো হবে কারণ ফোকাস প্যারেন্ট ভিউতে স্থানান্তরিত হবে।


2

আমি সত্যিই মনে করি এটি ব্যবহারের getLocationOnScreenচেয়ে আরও শক্তিশালী উপায় getGlobalVisibleRect। কারণ আমি একটি সমস্যা পূরণ। একটি তালিকাভিউ রয়েছে যা কিছু এডিটেক্সট ধারণ করে এবং ajustpanক্রিয়াকলাপে সেট করে। আমি এতে getGlobalVisibleRectএকটি স্ক্রোলওয়াইয়ের মতো দেখে মনে হচ্ছে এমন একটি মান ফিরে পেয়েছি, তবে ইভেন্ট.জেটআরওয়াই সর্বদা পর্দা থাকে। নীচের কোডটি ভাল কাজ করে।

public boolean dispatchTouchEvent(MotionEvent event) {
    if (event.getAction() == MotionEvent.ACTION_DOWN) {
        View v = getCurrentFocus();
        if ( v instanceof EditText) {
            if (!isPointInsideView(event.getRawX(), event.getRawY(), v)) {
                Log.i(TAG, "!isPointInsideView");

                Log.i(TAG, "dispatchTouchEvent clearFocus");
                v.clearFocus();
                InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
                imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
            }
        }
    }
    return super.dispatchTouchEvent( event );
}

/**
 * Determines if given points are inside view
 * @param x - x coordinate of point
 * @param y - y coordinate of point
 * @param view - view object to compare
 * @return true if the points are within view bounds, false otherwise
 */
private boolean isPointInsideView(float x, float y, View view) {
    int location[] = new int[2];
    view.getLocationOnScreen(location);
    int viewX = location[0];
    int viewY = location[1];

    Log.i(TAG, "location x: " + location[0] + ", y: " + location[1]);

    Log.i(TAG, "location xWidth: " + (viewX + view.getWidth()) + ", yHeight: " + (viewY + view.getHeight()));

    // point is inside view bounds
    return ((x > viewX && x < (viewX + view.getWidth())) &&
            (y > viewY && y < (viewY + view.getHeight())));
}

তোমাকে অনেক ধন্যবাদ! এটি রিসাইক্লার ভিউ ইত্যাদির মতো সমস্ত লেআউট দৃশ্যের জন্য কাজ করেছিল। আমি অন্যান্য সমাধানের চেষ্টা করেছি তবে এটি সমস্তই ফিট করে! :) দারূন কাজ!
Woppi

1

অন্যান্য ভিউটি স্পর্শ করা হলে ফোকাসটি হারাতে উভয় মতামতই ভিউ.ফোকাসেবলইটচমড (সত্য) হিসাবে সেট করা উচিত।

তবে মনে হচ্ছে স্পর্শ মোডে ব্যবহারের ফোকাসগুলি প্রস্তাবিত নয়। দয়া করে এখানে একবার দেখুন: http://android-developers.blogspot.com/2008/12/touch-mode.html


হ্যাঁ, আমি এটি পড়েছি তবে এটি আমি যা চাই ঠিক তা নয়। মনোযোগী হওয়ার জন্য আমার অন্য দৃষ্টিভঙ্গির দরকার নেই। যদিও আমি এখনই কোনও বিকল্প উপায় দেখতে পাচ্ছি না ... হতে পারে, ব্যবহারকারী যখন মনোনিবেশযোগ্য বাচ্চাকে ক্লিক করে তখন কোনওভাবে ফোকাস ক্যাপচারের জন্য মূল দৃষ্টিভঙ্গি তৈরি করতে পারে?
নোট 173 14

আমি যতক্ষণ জানি, ফোকাসটি কোনও শ্রেণি দ্বারা পরিচালিত হতে পারে না, এটি অ্যান্ড্রয়েড দ্বারা পরিচালিত হয় ... কোনও ধারণা নেই :(
ফর্মারসিও

1

কোটলিনে

হাইডকিবোর্ড () একটি কোটলিন এক্সটেনশন

fun Activity.hideKeyboard() {
    hideKeyboard(currentFocus ?: View(this))
}

ক্রিয়াকলাপে dispatchTouchEvent যোগ করুন

override fun dispatchTouchEvent(event: MotionEvent): Boolean {
    if (event.action == MotionEvent.ACTION_DOWN) {
        val v: View? = currentFocus
        if (v is EditText) {
            val outRect = Rect()
            v.getGlobalVisibleRect(outRect)
            if (!outRect.contains(event.rawX.toInt(), event.rawY.toInt())) {
                v.clearFocus()
                hideKeyboard()
            }
        }
    }
    return super.dispatchTouchEvent(event)
}

শীর্ষস্থানীয় পিতামাতার মধ্যে এই বৈশিষ্ট্যগুলি যুক্ত করুন

android:focusableInTouchMode="true"
android:focusable="true"

1

এটি zMan কোডের উপর ভিত্তি করে আমার সংস্করণ। যদি পরবর্তী ভিউটি সম্পাদনা পাঠ্য হয় তবে এটি কীবোর্ডটি আড়াল করবে না। যদি ব্যবহারকারী কেবল স্ক্রিনটি স্ক্রোল করে তবে এটি কীবোর্ডটিও আড়াল করবে না।

@Override
public boolean dispatchTouchEvent(MotionEvent event) {
    if (event.getAction() == MotionEvent.ACTION_DOWN) {
        downX = (int) event.getRawX();
    }

    if (event.getAction() == MotionEvent.ACTION_UP) {
        View v = getCurrentFocus();
        if (v instanceof EditText) {
            int x = (int) event.getRawX();
            int y = (int) event.getRawY();
            //Was it a scroll - If skip all
            if (Math.abs(downX - x) > 5) {
                return super.dispatchTouchEvent(event);
            }
            final int reducePx = 25;
            Rect outRect = new Rect();
            v.getGlobalVisibleRect(outRect);
            //Bounding box is to big, reduce it just a little bit
            outRect.inset(reducePx, reducePx);
            if (!outRect.contains(x, y)) {
                v.clearFocus();
                boolean touchTargetIsEditText = false;
                //Check if another editText has been touched
                for (View vi : v.getRootView().getTouchables()) {
                    if (vi instanceof EditText) {
                        Rect clickedViewRect = new Rect();
                        vi.getGlobalVisibleRect(clickedViewRect);
                        //Bounding box is to big, reduce it just a little bit
                        clickedViewRect.inset(reducePx, reducePx);
                        if (clickedViewRect.contains(x, y)) {
                            touchTargetIsEditText = true;
                            break;
                        }
                    }
                }
                if (!touchTargetIsEditText) {
                    InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
                    imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
                }
            }
        }
    }
    return super.dispatchTouchEvent(event);
}

0

সর্বোত্তম উপায় হ'ল ডিফল্ট পদ্ধতিটি ব্যবহার করা clearFocus()

আপনি ঠিক কীভাবে কোডগুলি সমাধান করবেন জানেন onTouchListener?

শুধু কল EditText.clearFocus()। এটি ফোকাস পরিষ্কার করবে last EditText


0

@ প্যাকানদের পরামর্শ অনুসারে আপনি dispatchTouchEvent(MotionEvent event)আপনার ক্রিয়াকলাপে এটি ওভাররাইডিং করতে পারেন ।

এখানে আমরা স্পর্শ স্থানাঙ্ক পেতে এবং সীমা দেখার জন্য তাদের তুলনা। যদি কোনও দেখার বাইরে স্পর্শ করা হয় তবে কিছু করুন।

@Override
public boolean dispatchTouchEvent(MotionEvent event) {
    if (event.getAction() == MotionEvent.ACTION_DOWN) {
        View yourView = (View) findViewById(R.id.view_id);
        if (yourView != null && yourView.getVisibility() == View.VISIBLE) {
            // touch coordinates
            int touchX = (int) event.getX();
            int touchY = (int) event.getY();
            // get your view coordinates
            final int[] viewLocation = new int[2];
            yourView.getLocationOnScreen(viewLocation);

            // The left coordinate of the view
            int viewX1 = viewLocation[0];
            // The right coordinate of the view
            int viewX2 = viewLocation[0] + yourView.getWidth();
            // The top coordinate of the view
            int viewY1 = viewLocation[1];
            // The bottom coordinate of the view
            int viewY2 = viewLocation[1] + yourView.getHeight();

            if (!((touchX >= viewX1 && touchX <= viewX2) && (touchY >= viewY1 && touchY <= viewY2))) {

                Do what you want...

                // If you don't want allow touch outside (for example, only hide keyboard or dismiss popup) 
                return false;
            }
        }
    }
    return super.dispatchTouchEvent(event);
}

এছাড়াও রানটাইম চলাকালীন আপনার ক্রিয়াকলাপের বিন্যাস পরিবর্তন না হলে (উদাহরণস্বরূপ আপনি টুকরোগুলি যোগ করবেন না বা বিন্যাসটি থেকে দর্শনগুলি সরিয়ে ফেলবেন না) যদি দেখার অস্তিত্ব এবং দৃশ্যমানতা পরীক্ষা করা প্রয়োজন হয় না। তবে আপনি যদি কাস্টম কনটেক্সট মেনুটি বন্ধ করতে (বা কিছু কিছু করণীয় করতে চান) (যেমন আইটেমের ওভারফ্লো মেনুটি ব্যবহার করার সময় গুগল প্লে স্টোরের মতো) দেখতে এটির উপস্থিতি যাচাই করা দরকার। অন্যথায় আপনি একটি পাবেন NullPointerException


0

কোডের এই সরল স্নিপেটটি আপনি যা চান তা করে

GestureDetector gestureDetector = new GestureDetector(getContext(), new GestureDetector.SimpleOnGestureListener() {
            @Override
            public boolean onSingleTapConfirmed(MotionEvent e) {
                KeyboardUtil.hideKeyboard(getActivity());
                return true;
            }
        });
mScrollView.setOnTouchListener((v, e) -> gestureDetector.onTouchEvent(e));
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.