いまのところデベロッパー側に好意的に受け入れられている気配を感じないMulti-Window機能ですが、自分がプライベート開発しているアプリでは可能性を感じたので、そのあたりで調べたことを書きます。あと5/25(水)に行われたpotatotips #29の発表内容+α版です。
対象アプリ:AAびゅーわ - AA表示アプリ - Google Play の Android アプリ
ついでにここで主張したいのですが、AAの表示がズレる環境のまま資料や記事に公開する行為は、思った以上にAA職人の精神にダメージがあります(個人差があります)。注意してください。
前置き:やれなかったD&D機能
本来はMulti-Windowに合わせて実装された「アプリ間のドラッグ・アンド・ドロップ(D&D)機能」を使い、テキストをドロップすることでビューワを更新させたかったのですが、ドラッグする側(ブラウザだったり、ソフトキーボードアプリだったり)がドラッグ機能を実装してくれない限りは意味がないことに気がついたので、今回の実装は諦めることになりました。
ただこの画面の構成上、ブラウザ側の共有機能でビューワの画面を更新できるんじゃないかと思って、その辺りの動きを見ることにしました。
なお、確認にはAndroid N Preview 3公開時点での最新のAndroid Emulatorで行っております。
本題
Multi-Window上で「共有」をすると何が起こるのか
というわけで、とりあえず現在のアプリの構成のまま(android:resizeableActivity=true
は実装済み)、ブラウザの共有機能からビューワを呼んでみました。
↓結果
まあ、予想はしてましたがブラウザ側のWindowでビューワが起動してしまいました。何も考えずに呼び出すと、もう一方のWindowの状態は関係なく新規タスクとしてアプリが起動してしまうようです。
では、呼ばれる側(ビューワ)がlaunchMode=singleTask
の場合、どうなるでしょうか。
<activity
android:name="jp.yamaglo.aaviewer.MainActivity"
android:label="@string/app_name"
android:launchMode="singleTask"
>
更新されました!
launchMode=singleTask
の場合は新規でタスクを作成せず、現在表示中のビューワのonNewIntent()
が呼ばれていました。
Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT
を付けて呼び出した時の動作
とりあえずlaunchMode=singleTask
であれば、「もう一方のWindowでアプリを開いている場合」は更新されることがわかりました。しかし「もう一方のWindowでアプリを開いていない場合」では、呼び出した側のWindowでアプリが立ち上がってしまいます。
となると、「共有先は必ず片方のWindowで開きたい」場合はどうすればいいのでしょうか。
ここで公式のドキュメントを見ると、以下のことが書かれていました。
マルチ ウィンドウのサポート | Android Developers
新しいアクティビティを起動するときに、可能であれば、新しいアクティビティを現在のアクティビティの隣に表示する必用があるかどうかをシステムに示すことができます。 そうするには、フラグ Intent.FLAG_ACTIVITY_LAUNCH_TO_ADJACENT を使用します。 このフラグを渡すと、次の動作がリクエストされます。
ということで、 Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT
(公式には_TO_
が入っているが、誤字)を使って呼び出してみました。
Button button = (Button)findViewById(R.id.button_launch);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent();
intent.setAction(Intent.ACTION_SEND);
intent.setType("text/plain");
intent.putExtra(Intent.EXTRA_TEXT, "text");
intent.setFlags(Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT);
startActivity(intent);
}
});
で、結果は以下のようになりました。
lanchMode=Default |
lanchMode=singleTask |
|
---|---|---|
片方のWindowで表示中 | 呼び出し側で立ち上がる | 開いているアプリのonNewIntent() が呼ばれる |
片方のWindowで非表示 | 呼び出し側で立ち上がる | 片方のWindowでアプリが立ち上がる |
と、わざわざ表にするまでもなかったですが、launchMode=singleTask
のアプリであれば、もう片方のWindowでアプリが立ち上がってくれるようです。
なぜsingleTaskである必要があるのかは、よくわかりません……。
いちおう先ほどのドキュメントを読む限り、ActivityOptions.setLaunchBounds()
を使って何かしら対応できそうなことが書かれていますが、何も変化はありませんでした。
なんとなくですが、ドキュメントのサポートを見てもMulti-Windowモードでの共有動作についてはあまり考慮されていないのかもしれません。
補足:FloatingActionModeでの呼び出しについて
ついでにAndroid 6.0から実装されたFloatingActionModeからでも呼び出せるようにしましたが、こちらの場合では launchModeの設定が如何にせよ、必ず呼び出し側のWindowでアプリが「新規に」立ち上がります。
↓結果(テキストが読み込めてないんですが、こちらの実装ミスです)
これはAndrodi6.0の頃からlaunchModeがどんな設定だろうと新タスクとして起動されてしまうので、バグではなく仕様です(この動きがいいかどうかは置いといて……)。
まとめ
- 呼ばれるアプリ(Activity)のlaunchModeがsingleTaskの場合、表示中のアプリを更新(
onNewIntent()
の呼び出し)が行われる -
Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT
付きのIntent呼び出しの場合、launchModeがsingleTaskなら片方のWindowでアプリが立ち上がる - launchModeがデフォルトだと、必ず呼び出し元のWindowでアプリが立ち上がる
- Window間の連携は、一方のアプリだけ対応してもダメっぽいので辛い