- AppShortcuts - CommitContentSampleApp - CommitContentSampleIME Change-Id: I3cefc134839f944b1c0c5efc943fb779c7e7ee70
287 lines
11 KiB
Java
287 lines
11 KiB
Java
/*
|
|
* Copyright (C) 2016 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package com.example.android.commitcontent.ime;
|
|
|
|
import android.app.AppOpsManager;
|
|
import android.content.ClipDescription;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.pm.PackageManager;
|
|
import android.inputmethodservice.InputMethodService;
|
|
import android.net.Uri;
|
|
import android.os.Build;
|
|
import android.support.annotation.NonNull;
|
|
import android.support.annotation.Nullable;
|
|
import android.support.annotation.RawRes;
|
|
import android.support.v13.view.inputmethod.EditorInfoCompat;
|
|
import android.support.v13.view.inputmethod.InputConnectionCompat;
|
|
import android.support.v13.view.inputmethod.InputContentInfoCompat;
|
|
import android.support.v4.content.FileProvider;
|
|
import android.util.Log;
|
|
import android.view.View;
|
|
import android.view.inputmethod.EditorInfo;
|
|
import android.view.inputmethod.InputBinding;
|
|
import android.view.inputmethod.InputConnection;
|
|
import android.widget.Button;
|
|
import android.widget.LinearLayout;
|
|
|
|
import java.io.File;
|
|
import java.io.FileOutputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.io.OutputStream;
|
|
|
|
|
|
public class ImageKeyboard extends InputMethodService {
|
|
|
|
private static final String TAG = "ImageKeyboard";
|
|
private static final String AUTHORITY = "com.example.android.supportv13.sampleime.inputcontent";
|
|
private static final String MIME_TYPE_GIF = "image/gif";
|
|
private static final String MIME_TYPE_PNG = "image/png";
|
|
private static final String MIME_TYPE_WEBP = "image/webp";
|
|
|
|
private File mPngFile;
|
|
private File mGifFile;
|
|
private File mWebpFile;
|
|
private Button mGifButton;
|
|
private Button mPngButton;
|
|
private Button mWebpButton;
|
|
|
|
private boolean isCommitContentSupported(
|
|
@Nullable EditorInfo editorInfo, @NonNull String mimeType) {
|
|
if (editorInfo == null) {
|
|
return false;
|
|
}
|
|
|
|
final InputConnection ic = getCurrentInputConnection();
|
|
if (ic == null) {
|
|
return false;
|
|
}
|
|
|
|
if (!validatePackageName(editorInfo)) {
|
|
return false;
|
|
}
|
|
|
|
final String[] supportedMimeTypes = EditorInfoCompat.getContentMimeTypes(editorInfo);
|
|
for (String supportedMimeType : supportedMimeTypes) {
|
|
if (ClipDescription.compareMimeTypes(mimeType, supportedMimeType)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private void doCommitContent(@NonNull String description, @NonNull String mimeType,
|
|
@NonNull File file) {
|
|
final EditorInfo editorInfo = getCurrentInputEditorInfo();
|
|
|
|
// Validate packageName again just in case.
|
|
if (!validatePackageName(editorInfo)) {
|
|
return;
|
|
}
|
|
|
|
final Uri contentUri = FileProvider.getUriForFile(this, AUTHORITY, file);
|
|
|
|
// As you as an IME author are most likely to have to implement your own content provider
|
|
// to support CommitContent API, it is important to have a clear spec about what
|
|
// applications are going to be allowed to access the content that your are going to share.
|
|
final int flag;
|
|
if (Build.VERSION.SDK_INT >= 25) {
|
|
// On API 25 and later devices, as an analogy of Intent.FLAG_GRANT_READ_URI_PERMISSION,
|
|
// you can specify InputConnectionCompat.INPUT_CONTENT_GRANT_READ_URI_PERMISSION to give
|
|
// a temporary read access to the recipient application without exporting your content
|
|
// provider.
|
|
flag = InputConnectionCompat.INPUT_CONTENT_GRANT_READ_URI_PERMISSION;
|
|
} else {
|
|
// On API 24 and prior devices, we cannot rely on
|
|
// InputConnectionCompat.INPUT_CONTENT_GRANT_READ_URI_PERMISSION. You as an IME author
|
|
// need to decide what access control is needed (or not needed) for content URIs that
|
|
// you are going to expose. This sample uses Context.grantUriPermission(), but you can
|
|
// implement your own mechanism that satisfies your own requirements.
|
|
flag = 0;
|
|
try {
|
|
// TODO: Use revokeUriPermission to revoke as needed.
|
|
grantUriPermission(
|
|
editorInfo.packageName, contentUri, Intent.FLAG_GRANT_READ_URI_PERMISSION);
|
|
} catch (Exception e){
|
|
Log.e(TAG, "grantUriPermission failed packageName=" + editorInfo.packageName
|
|
+ " contentUri=" + contentUri, e);
|
|
}
|
|
}
|
|
|
|
final InputContentInfoCompat inputContentInfoCompat = new InputContentInfoCompat(
|
|
contentUri,
|
|
new ClipDescription(description, new String[]{mimeType}),
|
|
null /* linkUrl */);
|
|
InputConnectionCompat.commitContent(
|
|
getCurrentInputConnection(), getCurrentInputEditorInfo(), inputContentInfoCompat,
|
|
flag, null);
|
|
}
|
|
|
|
private boolean validatePackageName(@Nullable EditorInfo editorInfo) {
|
|
if (editorInfo == null) {
|
|
return false;
|
|
}
|
|
final String packageName = editorInfo.packageName;
|
|
if (packageName == null) {
|
|
return false;
|
|
}
|
|
|
|
// In Android L MR-1 and prior devices, EditorInfo.packageName is not a reliable identifier
|
|
// of the target application because:
|
|
// 1. the system does not verify it [1]
|
|
// 2. InputMethodManager.startInputInner() had filled EditorInfo.packageName with
|
|
// view.getContext().getPackageName() [2]
|
|
// [1]: https://android.googlesource.com/platform/frameworks/base/+/a0f3ad1b5aabe04d9eb1df8bad34124b826ab641
|
|
// [2]: https://android.googlesource.com/platform/frameworks/base/+/02df328f0cd12f2af87ca96ecf5819c8a3470dc8
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
|
return true;
|
|
}
|
|
|
|
final InputBinding inputBinding = getCurrentInputBinding();
|
|
if (inputBinding == null) {
|
|
// Due to b.android.com/225029, it is possible that getCurrentInputBinding() returns
|
|
// null even after onStartInputView() is called.
|
|
// TODO: Come up with a way to work around this bug....
|
|
Log.e(TAG, "inputBinding should not be null here. "
|
|
+ "You are likely to be hitting b.android.com/225029");
|
|
return false;
|
|
}
|
|
final int packageUid = inputBinding.getUid();
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
|
|
final AppOpsManager appOpsManager =
|
|
(AppOpsManager) getSystemService(Context.APP_OPS_SERVICE);
|
|
try {
|
|
appOpsManager.checkPackage(packageUid, packageName);
|
|
} catch (Exception e) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
final PackageManager packageManager = getPackageManager();
|
|
final String possiblePackageNames[] = packageManager.getPackagesForUid(packageUid);
|
|
for (final String possiblePackageName : possiblePackageNames) {
|
|
if (packageName.equals(possiblePackageName)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public void onCreate() {
|
|
super.onCreate();
|
|
|
|
// TODO: Avoid file I/O in the main thread.
|
|
final File imagesDir = new File(getFilesDir(), "images");
|
|
imagesDir.mkdirs();
|
|
mGifFile = getFileForResource(this, R.raw.animated_gif, imagesDir, "image.gif");
|
|
mPngFile = getFileForResource(this, R.raw.dessert_android, imagesDir, "image.png");
|
|
mWebpFile = getFileForResource(this, R.raw.animated_webp, imagesDir, "image.webp");
|
|
}
|
|
|
|
@Override
|
|
public View onCreateInputView() {
|
|
mGifButton = new Button(this);
|
|
mGifButton.setText("Insert GIF");
|
|
mGifButton.setOnClickListener(new View.OnClickListener() {
|
|
@Override
|
|
public void onClick(View view) {
|
|
ImageKeyboard.this.doCommitContent("A waving flag", MIME_TYPE_GIF, mGifFile);
|
|
}
|
|
});
|
|
|
|
mPngButton = new Button(this);
|
|
mPngButton.setText("Insert PNG");
|
|
mPngButton.setOnClickListener(new View.OnClickListener() {
|
|
@Override
|
|
public void onClick(View view) {
|
|
ImageKeyboard.this.doCommitContent("A droid logo", MIME_TYPE_PNG, mPngFile);
|
|
}
|
|
});
|
|
|
|
mWebpButton = new Button(this);
|
|
mWebpButton.setText("Insert WebP");
|
|
mWebpButton.setOnClickListener(new View.OnClickListener() {
|
|
@Override
|
|
public void onClick(View view) {
|
|
ImageKeyboard.this.doCommitContent(
|
|
"Android N recovery animation", MIME_TYPE_WEBP, mWebpFile);
|
|
}
|
|
});
|
|
|
|
final LinearLayout layout = new LinearLayout(this);
|
|
layout.setOrientation(LinearLayout.VERTICAL);
|
|
layout.addView(mGifButton);
|
|
layout.addView(mPngButton);
|
|
layout.addView(mWebpButton);
|
|
return layout;
|
|
}
|
|
|
|
@Override
|
|
public boolean onEvaluateFullscreenMode() {
|
|
// In full-screen mode the inserted content is likely to be hidden by the IME. Hence in this
|
|
// sample we simply disable full-screen mode.
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public void onStartInputView(EditorInfo info, boolean restarting) {
|
|
mGifButton.setEnabled(mGifFile != null && isCommitContentSupported(info, MIME_TYPE_GIF));
|
|
mPngButton.setEnabled(mPngFile != null && isCommitContentSupported(info, MIME_TYPE_PNG));
|
|
mWebpButton.setEnabled(mWebpFile != null && isCommitContentSupported(info, MIME_TYPE_WEBP));
|
|
}
|
|
|
|
private static File getFileForResource(
|
|
@NonNull Context context, @RawRes int res, @NonNull File outputDir,
|
|
@NonNull String filename) {
|
|
final File outputFile = new File(outputDir, filename);
|
|
final byte[] buffer = new byte[4096];
|
|
InputStream resourceReader = null;
|
|
try {
|
|
try {
|
|
resourceReader = context.getResources().openRawResource(res);
|
|
OutputStream dataWriter = null;
|
|
try {
|
|
dataWriter = new FileOutputStream(outputFile);
|
|
while (true) {
|
|
final int numRead = resourceReader.read(buffer);
|
|
if (numRead <= 0) {
|
|
break;
|
|
}
|
|
dataWriter.write(buffer, 0, numRead);
|
|
}
|
|
return outputFile;
|
|
} finally {
|
|
if (dataWriter != null) {
|
|
dataWriter.flush();
|
|
dataWriter.close();
|
|
}
|
|
}
|
|
} finally {
|
|
if (resourceReader != null) {
|
|
resourceReader.close();
|
|
}
|
|
}
|
|
} catch (IOException e) {
|
|
return null;
|
|
}
|
|
}
|
|
}
|