docs: Add new samples for N MR1
- AppShortcuts - CommitContentSampleApp - CommitContentSampleIME Change-Id: I3cefc134839f944b1c0c5efc943fb779c7e7ee70
This commit is contained in:
@@ -0,0 +1,286 @@
|
||||
/*
|
||||
* 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;
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user