WeChatとQQは、他のアプリのオープンリストに自分のアプリを追加し、ファイルパスを取得することができます
プロジェクトの要件として、WeChatやQQなどのサードパーティアプリケーションから自分のアプリケーションに直接ファイルを共有したいのです。実際にそれを行う過程で、いくつかの問題に遭遇しましたので、ここに記録します。
以前、自分のアプリに画像共有をしたことがあったので、以下のコードを追加すれば済むと思ったのです。
<! --add own app to app list when calling share or send -->
<intent-filter>
<action android:name="android.intent.action.SEND" />
<category android:name="android.intent.category.DEFAULT" />
<! -- Allow all types of files -- >
<data android:mimeType="*/*" />
<! -- Allow only image files -- >
<! --<data android:mimeType="image/*"/>-->
</intent-filter>
その結果、WeChatでファイルを開いて送信を選択すると、表示されるリストには私のアプリがありますが、ファイルを開いてquot;他のアプリで開くを選択しても、リストには私のアプリはなく、しばらく検索しました。
<! -- Call other apps in WeChat to open the list to add their own apps -->
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<data android:scheme="content"/>
<! -- Allow all types of files -- >
<data android:mimeType="*/*" />
<! -- Allow only image files -- >
<! --<data android:mimeType="image/*"/>-->
</intent-filter>
WeChatは使えるがQQは使えないことが判明し、クラッシュした。
Baiduでしばらく調べると解決策が見つかり、android:schemeの中身をfileに変更するとうまくいった
<! -- Calling other apps in QQ to open the list to add their own apps -->
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<data android:scheme="file"/>
<! -- Allow all types of files -- >
<data android:mimeType="*/*" />
<! -- Allow only image files -- >
<! --<data android:mimeType="image/*"/>-->
</intent-filter>
アプリケーション一覧は解決、あとはファイルパスの取得に問題あり、どーも~~~。
actionがandroid.intent.action.SENDの場合、get pathメソッドで取得します。
Uri uri = getIntent().getExtras().getParcelable(Intent.EXTRA_STREAM);
String url = uri.toString();
Log.e("url", url);
このアプリを選択して自分のファイルマネージャで共有する場合、印刷されるURLアドレスはcontent://media/external/file/85139となります。
QQブラウザを使用してWeChatで開き、このアプリを送信すると、印刷されるURLは次のようになります。
ファイル:///storage/emulated/0/tencent/MicroMsg/Download/111.doc
actionがandroid.intent.action.Viewの場合、pathメソッドを取得します。
Intent intent = getIntent();
Uri uri = intent.getData();
String url = intent.getDataString();
Log.e("url", url);
このアプリを選択して自分のファイルマネージャーで開くと、urlの値は content://media/external/file/85139 になります。
このアプリケーションを選択してWeChatで開くと、urlの値は次のようになります。
content://com.tencent.mm.external.fileprovider/external/tencent/MicroMsg/Download/111.doc
このアプリケーションを選択してQQで開くと、urlの値が
file:///storage/emulated/0/tencent/MicroMsg/Download/111.doc
新バージョンのurlアドレスも、以下のようにfileproviderを使用するように変更されているようです。
content://com.tencent.mobileqq.fileprovider/external_files/storage/emulated/0/Android/data/com.tencent.mobileqq/Tencent/QQfile_ recv/1111.doc recv/1111.doc
ローカルファイルを取得する ファイル形式のパスは直接使用することができます、URIはファイル形式に変換する必要があります、ここでは2つのケースで説明します。
メディアライブラリのクエリURIであれば、content://media/external/file/85139のような形式をファイルメソッドにします。
public static String getFilePathFromContentUri(Uri selectedVideoUri,
Activity context) {
String filePath = "";
String[] filePathColumn = {MediaStore;}
// Cursor cursor = contentResolver.query(selectedVideoUri, filePathColumn, null, null, null);
// You can also get the cursor in the following way
Cursor cursor = context.managedQuery(selectedVideoUri, filePathColumn, null, null, null);
if (cursor ! = null) {
cursor.moveToFirst();
int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
filePath = cursor.getString(columnIndex);
try {
//automatic closure for versions 4.0 and above (4.0--14;; 4.0.3--15)
if (Integer.parseInt(Build.VERSION.SDK) < 14) {
cursor.close();
}
} catch (Exception e) {
Log.e("Convert address", "error:" + e);
}
}
return filePath;
}
fileproviderが提供するcontenturiであれば、content://com.tencent.mm.external.fileprovider/external/tencent/MicroMsg/Download/111.docといったファイル形式方式(ここでは https://www.jianshu.com/p/0ca6989f2bc2 から参照、借用)に変換する。
public static String getFPUriToPath(Context context, Uri uri) {
try {
List<PackageInfo> packs = context.getPackageManager().getInstalledPackages(PackageManager.GET_PROVIDERS);
if (packs ! = null) {
String fileProviderClassName = FileProvider.class.getName();
for (PackageInfo pack : packs) {
ProviderInfo[] providers = pack.providers;
if (providers ! = null) {
for (ProviderInfo provider : providers) {
if (uri.getAuthority().equals(provider.authority)) {
if (provider.name.equalsIgnoreCase(fileProviderClassName)) { if (provider.name.equalsIgnoreCase(fileProviderClassName)) {
Class<FileProvider> fileProviderClass = FileProvider.class;
try {
Method getPathStrategy = fileProviderClass.getDeclaredMethod("getPathStrategy", Context.class, String.class);
getPathStrategy.setAccessible(true);
Object invoke = getPathStrategy.invoke(null, context, uri.getAuthority());
if (invoke ! = null) {
String PathStrategyStringClass = FileProvider.class.getName() + "$PathStrategy";
Class<? > PathStrategy = Class.forName(PathStrategyStringClass);
Method getFileForUri = PathStrategy.getDeclaredMethod("getFileForUri", Uri.class);
getFileForUri.setAccessible(true);
Object invoke1 = getFileForUri.invoke(invoke, uri);
if (invoke1 instanceof File) {
String filePath = ((File) invoke1).getAbsolutePath();
return filePath;
}
}
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
break;
}
break;
}
}
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
ついに完成!!!
友達のメッセージが失敗することが多かったので、自分で再挑戦したところ、getFPUriToPathメソッドを実行すると、アドレスがうまく変換されないことが続き、調べたところ、WeChatのfileproviderはcom. tencent.mm.external.fileprovider、QQのfileproviderはcom.tencent.mobileqq.fileproviderで、どちらもこのコード provider.name.equalsIgnoreCase(fileProviderClassName) で provider を取得していることがわかりました。 という名前のFileProvider、つまりv4パッケージの下ですが、今はASのバージョンが高く、プロジェクトをビルドすると、それらはすべてandroidxパッケージ、つまりandroidx.core.content.FileProviderを使っているので、につながります。 プロバイダ名が矛盾しているのでurlアドレスがurlに変換されておらず、v4パッケージの中にFileProvider.というクラスを導入したらうまくいったということです。もちろん、他のuriからurlへのメソッドを探せばいいのですが、今のところ見つからなかったので、見つかったら追記します。
完全なコードは、以下を参照してください。 githubアドレス
関連
-
プログラム "git.exe "を実行できない場合の正しい解決方法です。CreateProcessエラー=2
-
cygwinのダウンロード、インストールチュートリアル、およびCDTの「makeプログラムがパスに見つからない」バグの解消
-
デフォルトのアクティビティが見つからない場合の対処法
-
アプリはGoogle検索でインデックスされません Androidmanifestのクソみたいな黄色い警告
-
最新のandroidプロジェクトディレクトリにあるarmeabi-v7aとarmeabiの具体的な意味とその違いを教えてください。
-
Android基本アプレット
-
SpinnerのOnItemSelectedListenerのonItemSelectedメソッドの4つのパラメーターの意味
-
Androidカスタムドロップダウンリストボックスコントロール
-
Android TextViewは、テキスト内容が表示省略記号を超えているかどうかを判断する
-
AndroidサポートデザインライブラリのFloatingActionButtonについて
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
Android端末にADBが接続できない!を解決。理由: デバイスが認証されていない!
-
Android.mk:7: *** セパレータがありません。
-
Androidで発生した問題、解決策とヒント
-
Android開発で「Attempt to invoke virtual method 'XXX()' on null object reference」というヌルポインター例外に遭遇する。
-
アンドロイドスタジオのエラーを解決する --> Error:(1, 0) id 'com.android.application' を持つプラグインが見つかりません。
-
Androidで、onTouchEventでダブルクリックを実装し、ダブルクリックイベントとして判定する方法
-
アンドロイドのエリプサイズを使用する
-
Android ProgressBarの色を変更する
-
Android--shape--描画のコーナー、グラデーション、パディング、サイズ、ソリッド、ストロークのプロパティを指定する。
-
アンドロイドシェイプ、グラデーション、角丸、ボーダーラインの設定