1 /*
2  * Copyright (C) 2012 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.settings.inputmethod;
18 
19 import android.app.Activity;
20 import android.app.Dialog;
21 import android.app.settings.SettingsEnums;
22 import android.content.Context;
23 import android.content.DialogInterface;
24 import android.content.Intent;
25 import android.hardware.input.InputDeviceIdentifier;
26 import android.hardware.input.InputManager;
27 import android.hardware.input.InputManager.InputDeviceListener;
28 import android.hardware.input.KeyboardLayout;
29 import android.os.Bundle;
30 import android.view.InputDevice;
31 import android.view.LayoutInflater;
32 import android.view.View;
33 import android.view.ViewGroup;
34 import android.widget.ArrayAdapter;
35 import android.widget.CheckedTextView;
36 import android.widget.RadioButton;
37 import android.widget.TextView;
38 
39 import androidx.appcompat.app.AlertDialog;
40 import androidx.loader.app.LoaderManager.LoaderCallbacks;
41 import androidx.loader.content.AsyncTaskLoader;
42 import androidx.loader.content.Loader;
43 
44 import com.android.settings.R;
45 import com.android.settings.core.instrumentation.InstrumentedDialogFragment;
46 
47 import java.util.ArrayList;
48 import java.util.Collections;
49 
50 public class KeyboardLayoutDialogFragment extends InstrumentedDialogFragment
51         implements InputDeviceListener, LoaderCallbacks<KeyboardLayoutDialogFragment.Keyboards> {
52     private static final String KEY_INPUT_DEVICE_IDENTIFIER = "inputDeviceIdentifier";
53 
54     private InputDeviceIdentifier mInputDeviceIdentifier;
55     private int mInputDeviceId = -1;
56     private InputManager mIm;
57     private KeyboardLayoutAdapter mAdapter;
58 
KeyboardLayoutDialogFragment()59     public KeyboardLayoutDialogFragment() {
60     }
61 
KeyboardLayoutDialogFragment(InputDeviceIdentifier inputDeviceIdentifier)62     public KeyboardLayoutDialogFragment(InputDeviceIdentifier inputDeviceIdentifier) {
63         mInputDeviceIdentifier = inputDeviceIdentifier;
64     }
65 
66 
67     @Override
getMetricsCategory()68     public int getMetricsCategory() {
69         return SettingsEnums.DIALOG_KEYBOARD_LAYOUT;
70     }
71 
72     @Override
onAttach(Activity activity)73     public void onAttach(Activity activity) {
74         super.onAttach(activity);
75 
76         Context context = activity.getBaseContext();
77         mIm = (InputManager)context.getSystemService(Context.INPUT_SERVICE);
78         mAdapter = new KeyboardLayoutAdapter(context);
79     }
80 
81     @Override
onCreate(Bundle savedInstanceState)82     public void onCreate(Bundle savedInstanceState) {
83         super.onCreate(savedInstanceState);
84 
85         if (savedInstanceState != null) {
86             mInputDeviceIdentifier = savedInstanceState.getParcelable(KEY_INPUT_DEVICE_IDENTIFIER);
87         }
88 
89         getLoaderManager().initLoader(0, null, this);
90     }
91 
92     @Override
onSaveInstanceState(Bundle outState)93     public void onSaveInstanceState(Bundle outState) {
94         super.onSaveInstanceState(outState);
95         outState.putParcelable(KEY_INPUT_DEVICE_IDENTIFIER, mInputDeviceIdentifier);
96     }
97 
98     @Override
onCreateDialog(Bundle savedInstanceState)99     public Dialog onCreateDialog(Bundle savedInstanceState) {
100         Context context = getActivity();
101         LayoutInflater inflater = LayoutInflater.from(context);
102         AlertDialog.Builder builder = new AlertDialog.Builder(context)
103             .setTitle(R.string.keyboard_layout_dialog_title)
104             .setPositiveButton(R.string.keyboard_layout_dialog_setup_button,
105                     new DialogInterface.OnClickListener() {
106                         @Override
107                         public void onClick(DialogInterface dialog, int which) {
108                             onSetupLayoutsButtonClicked();
109                         }
110                     })
111             .setSingleChoiceItems(mAdapter, -1,
112                     new DialogInterface.OnClickListener() {
113                         @Override
114                         public void onClick(DialogInterface dialog, int which) {
115                             onKeyboardLayoutClicked(which);
116                         }
117                     })
118             .setView(inflater.inflate(R.layout.keyboard_layout_dialog_switch_hint, null));
119         updateSwitchHintVisibility();
120         return builder.create();
121     }
122 
123     @Override
onResume()124     public void onResume() {
125         super.onResume();
126 
127         mIm.registerInputDeviceListener(this, null);
128 
129         InputDevice inputDevice =
130                 mIm.getInputDeviceByDescriptor(mInputDeviceIdentifier.getDescriptor());
131         if (inputDevice == null) {
132             dismiss();
133             return;
134         }
135         mInputDeviceId = inputDevice.getId();
136     }
137 
138     @Override
onPause()139     public void onPause() {
140         mIm.unregisterInputDeviceListener(this);
141         mInputDeviceId = -1;
142 
143         super.onPause();
144     }
145 
146     @Override
onCancel(DialogInterface dialog)147     public void onCancel(DialogInterface dialog) {
148         super.onCancel(dialog);
149         dismiss();
150     }
151 
onSetupLayoutsButtonClicked()152     private void onSetupLayoutsButtonClicked() {
153         ((OnSetupKeyboardLayoutsListener)getTargetFragment()).onSetupKeyboardLayouts(
154                 mInputDeviceIdentifier);
155     }
156 
157     @Override
onActivityResult(int requestCode, int resultCode, Intent data)158     public void onActivityResult(int requestCode, int resultCode, Intent data) {
159         super.onActivityResult(requestCode, resultCode, data);
160         show(getActivity().getSupportFragmentManager(), "layout");
161     }
162 
onKeyboardLayoutClicked(int which)163     private void onKeyboardLayoutClicked(int which) {
164         if (which >= 0 && which < mAdapter.getCount()) {
165             KeyboardLayout keyboardLayout = mAdapter.getItem(which);
166             if (keyboardLayout != null) {
167                 mIm.setCurrentKeyboardLayoutForInputDevice(mInputDeviceIdentifier,
168                         keyboardLayout.getDescriptor());
169             }
170             dismiss();
171         }
172     }
173 
174     @Override
onCreateLoader(int id, Bundle args)175     public Loader<Keyboards> onCreateLoader(int id, Bundle args) {
176         return new KeyboardLayoutLoader(getActivity().getBaseContext(), mInputDeviceIdentifier);
177     }
178 
179     @Override
onLoadFinished(Loader<Keyboards> loader, Keyboards data)180     public void onLoadFinished(Loader<Keyboards> loader, Keyboards data) {
181         mAdapter.clear();
182         mAdapter.addAll(data.keyboardLayouts);
183         mAdapter.setCheckedItem(data.current);
184         AlertDialog dialog = (AlertDialog)getDialog();
185         if (dialog != null) {
186             dialog.getListView().setItemChecked(data.current, true);
187         }
188         updateSwitchHintVisibility();
189     }
190 
191     @Override
onLoaderReset(Loader<Keyboards> loader)192     public void onLoaderReset(Loader<Keyboards> loader) {
193         mAdapter.clear();
194         updateSwitchHintVisibility();
195     }
196 
197     @Override
onInputDeviceAdded(int deviceId)198     public void onInputDeviceAdded(int deviceId) {
199     }
200 
201     @Override
onInputDeviceChanged(int deviceId)202     public void onInputDeviceChanged(int deviceId) {
203         if (mInputDeviceId >= 0 && deviceId == mInputDeviceId) {
204             getLoaderManager().restartLoader(0, null, this);
205         }
206     }
207 
208     @Override
onInputDeviceRemoved(int deviceId)209     public void onInputDeviceRemoved(int deviceId) {
210         if (mInputDeviceId >= 0 && deviceId == mInputDeviceId) {
211             dismiss();
212         }
213     }
214 
updateSwitchHintVisibility()215     private void updateSwitchHintVisibility() {
216         AlertDialog dialog = (AlertDialog)getDialog();
217         if (dialog != null) {
218             View customPanel = dialog.findViewById(com.google.android.material.R.id.customPanel);
219             customPanel.setVisibility(mAdapter.getCount() > 1 ? View.VISIBLE : View.GONE);
220         }
221     }
222 
223     private static final class KeyboardLayoutAdapter extends ArrayAdapter<KeyboardLayout> {
224         private final LayoutInflater mInflater;
225         private int mCheckedItem = -1;
226 
KeyboardLayoutAdapter(Context context)227         public KeyboardLayoutAdapter(Context context) {
228             super(context, com.android.internal.R.layout.simple_list_item_2_single_choice);
229             mInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
230         }
231 
setCheckedItem(int position)232         public void setCheckedItem(int position) {
233             mCheckedItem = position;
234         }
235 
236         @Override
getView(int position, View convertView, ViewGroup parent)237         public View getView(int position, View convertView, ViewGroup parent) {
238             KeyboardLayout item = getItem(position);
239             String label, collection;
240             if (item != null) {
241                 label = item.getLabel();
242                 collection = item.getCollection();
243             } else {
244                 label = getContext().getString(R.string.keyboard_layout_default_label);
245                 collection = "";
246             }
247 
248             boolean checked = (position == mCheckedItem);
249             if (collection.isEmpty()) {
250                 return inflateOneLine(convertView, parent, label, checked);
251             } else {
252                 return inflateTwoLine(convertView, parent, label, collection, checked);
253             }
254         }
255 
inflateOneLine(View convertView, ViewGroup parent, String label, boolean checked)256         private View inflateOneLine(View convertView, ViewGroup parent,
257                 String label, boolean checked) {
258             View view = convertView;
259             if (view == null || isTwoLine(view)) {
260                 view = mInflater.inflate(
261                         com.android.internal.R.layout.simple_list_item_single_choice,
262                         parent, false);
263                 setTwoLine(view, false);
264             }
265             CheckedTextView headline = (CheckedTextView) view.findViewById(android.R.id.text1);
266             headline.setText(label);
267             headline.setChecked(checked);
268             return view;
269         }
270 
inflateTwoLine(View convertView, ViewGroup parent, String label, String collection, boolean checked)271         private View inflateTwoLine(View convertView, ViewGroup parent,
272                 String label, String collection, boolean checked) {
273             View view = convertView;
274             if (view == null || !isTwoLine(view)) {
275                 view = mInflater.inflate(
276                         com.android.internal.R.layout.simple_list_item_2_single_choice,
277                         parent, false);
278                 setTwoLine(view, true);
279             }
280             TextView headline = (TextView) view.findViewById(android.R.id.text1);
281             TextView subText = (TextView) view.findViewById(android.R.id.text2);
282             RadioButton radioButton =
283                     (RadioButton)view.findViewById(com.android.internal.R.id.radio);
284             headline.setText(label);
285             subText.setText(collection);
286             radioButton.setChecked(checked);
287             return view;
288         }
289 
isTwoLine(View view)290         private static boolean isTwoLine(View view) {
291             return view.getTag() == Boolean.TRUE;
292         }
293 
setTwoLine(View view, boolean twoLine)294         private static void setTwoLine(View view, boolean twoLine) {
295             view.setTag(Boolean.valueOf(twoLine));
296         }
297     }
298 
299     private static final class KeyboardLayoutLoader extends AsyncTaskLoader<Keyboards> {
300         private final InputDeviceIdentifier mInputDeviceIdentifier;
301 
KeyboardLayoutLoader(Context context, InputDeviceIdentifier inputDeviceIdentifier)302         public KeyboardLayoutLoader(Context context, InputDeviceIdentifier inputDeviceIdentifier) {
303             super(context);
304             mInputDeviceIdentifier = inputDeviceIdentifier;
305         }
306 
307         @Override
loadInBackground()308         public Keyboards loadInBackground() {
309             Keyboards keyboards = new Keyboards();
310             InputManager im = (InputManager)getContext().getSystemService(Context.INPUT_SERVICE);
311             if (mInputDeviceIdentifier == null || NewKeyboardSettingsUtils.getInputDevice(
312                     im, mInputDeviceIdentifier) == null) {
313                 keyboards.keyboardLayouts.add(null); // default layout
314                 keyboards.current = 0;
315                 return keyboards;
316             }
317             String[] keyboardLayoutDescriptors = im.getEnabledKeyboardLayoutsForInputDevice(
318                     mInputDeviceIdentifier);
319             for (String keyboardLayoutDescriptor : keyboardLayoutDescriptors) {
320                 KeyboardLayout keyboardLayout = im.getKeyboardLayout(keyboardLayoutDescriptor);
321                 if (keyboardLayout != null) {
322                     keyboards.keyboardLayouts.add(keyboardLayout);
323                 }
324             }
325             Collections.sort(keyboards.keyboardLayouts);
326 
327             String currentKeyboardLayoutDescriptor =
328                     im.getCurrentKeyboardLayoutForInputDevice(mInputDeviceIdentifier);
329             if (currentKeyboardLayoutDescriptor != null) {
330                 final int numKeyboardLayouts = keyboards.keyboardLayouts.size();
331                 for (int i = 0; i < numKeyboardLayouts; i++) {
332                     if (keyboards.keyboardLayouts.get(i).getDescriptor().equals(
333                             currentKeyboardLayoutDescriptor)) {
334                         keyboards.current = i;
335                         break;
336                     }
337                 }
338             }
339 
340             if (keyboards.keyboardLayouts.isEmpty()) {
341                 keyboards.keyboardLayouts.add(null); // default layout
342                 keyboards.current = 0;
343             }
344             return keyboards;
345         }
346 
347         @Override
onStartLoading()348         protected void onStartLoading() {
349             super.onStartLoading();
350             forceLoad();
351         }
352 
353         @Override
onStopLoading()354         protected void onStopLoading() {
355             super.onStopLoading();
356             cancelLoad();
357         }
358     }
359 
360     public static final class Keyboards {
361         public final ArrayList<KeyboardLayout> keyboardLayouts = new ArrayList<KeyboardLayout>();
362         public int current = -1;
363     }
364 
365     public interface OnSetupKeyboardLayoutsListener {
onSetupKeyboardLayouts(InputDeviceIdentifier mInputDeviceIdentifier)366         public void onSetupKeyboardLayouts(InputDeviceIdentifier mInputDeviceIdentifier);
367     }
368 }
369