アプリの動作にマッチした、いい感じの背景エフェクトを付加したい
アプリがどのような動作をしているかを視覚的に分かりやすくするために、背景に何かしらのエフェクトを付加したいことがあります。
例えば、ビーコン発信アプリで発信中にいかにも『発信中です』って感じのエフェクトを発生させたい、等々…。
そんなとき、WebViewをアプリの背景として使うと、世の中に出回っているJavaScriptのエフェクトを利用できたり、エフェクト発生部分のソースを全プラットフォームで共通化できたりして便利です。
エフェクトを付加した例
以下のアニメーションは、ビーコン発信中っぽいエフェクトを追加してみた例です(参考)。
実機で動かすともっと綺麗にヌルヌル動くんですけどね。。。
iOS | Android |
---|---|
JavaScriptやCSSの扱いに慣れている人なら、この他にも色々なエフェクトを載せ放題です。
サンプルアプリ
以下にUPしたアプリを例にして、この後の説明を記載しています。
https://github.com/microwavePC/WebViewBackgroundSample
Prism for Xamarin.Forms (Prism.Forms) で作成しています。
サンプルアプリには、以下の背景エフェクトを実装しています。
- マトリックス風エフェクト(参考)
- ビーコン発信中っぽいエフェクト
- ブランク(エフェクトなし)
マトリックス風エフェクトは、以下のような感じです。
iOS | Android |
---|---|
実装手順
背景実装の手順はざっくり以下の通りです。
1. WebViewを継承した、背景用のカスタムコントロールを作成する(※)。
2. カスタムコントロールが背景になるよう、Xamlを修正する。
3. 背景にするWeb系ファイル(HTML、CSS、JavaScript)を用意し、共通プロジェクトに追加する。
4. プラットフォームごとのプロジェクトに、Web系ファイルへのリンクを作成する。
5. ローカルのHTMLファイルを実際に読み込む。
※WebViewの背景を透明化するために必要
1. WebViewを継承した、背景用カスタムコントロールの作成
Xamarin.Forms標準のWebViewは背景を透明にすることができません。マトリックス風エフェクトのように背景を塗りつぶすなら問題ありませんが、そうでないエフェクト用にWebViewをそのまま扱ってしまうと、背景が真っ白になってしまいます。
WebViewには『BackgroundColor』というプロパティがありますが、値を"Transparent"にしても効きません(Xamarin.Forms 2.3.4.231で確認)。その他の色を指定しても適用されません(バグ?)。
また、『Opacity』というプロパティがありますが、この値をいじるとWebViewそのものがエフェクトごとまるまる透明化してしまいます。
iOSなら真っ白の背景で特に問題ないとしても、Androidで背景が白塗りになると大分違和感が出てしまいます。
ということで、背景を透明にしたWebViewを自作してこの点を解決します。
1.1. 共通プロジェクトにCustomWebView.xamlを作成する。
今後さらに機能を追加していくかもしれないので、このような名前です。
具体的には、以下の機能(というか性質)を追加しています。
- 背景の透過
- iOSで、画面上部の電池残量やらの部分とWebViewが被らないような調整
<?xml version="1.0" encoding="UTF-8"?>
<WebView xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="WebViewBackgroundSample.CustomWebView">
<WebView.Margin>
<!-- iOSだけ、上に20pxのMarginを設定する -->
<OnPlatform x:TypeArguments="Thickness"
iOS="0,20,0,0"
Android="0,0,0,0"
WinPhone="0,0,0,0" />
</WebView.Margin>
</WebView>
using Xamarin.Forms;
namespace WebViewBackgroundSample
{
// WebViewを継承しているだけ
public partial class CustomWebView : WebView
{
public CustomWebView()
{
InitializeComponent();
}
}
}
1.2. 各プラットフォームのプロジェクトにCustomWebViewのRendererを作成する。
背景の透過はこの中に実装します。
[2017/04/13追記] iOS側の透過に必要な1行(Opaque = false;)が抜けていたので追加しました。
using WebViewBackgroundSample;
using WebViewBackgroundSample.Droid;
using Xamarin.Forms;
using Xamarin.Forms.Platform.Android;
[assembly: ExportRenderer(typeof(CustomWebView), typeof(CustomWebViewRenderer))]
namespace WebViewBackgroundSample.Droid
{
public class CustomWebViewRenderer : WebViewRenderer
{
protected override void OnElementChanged(ElementChangedEventArgs<WebView> e)
{
base.OnElementChanged(e);
// 背景を透過
this.Control.SetBackgroundColor(Android.Graphics.Color.Transparent);
}
}
}
using UIKit;
using WebViewBackgroundSample;
using WebViewBackgroundSample.iOS;
using Xamarin.Forms;
using Xamarin.Forms.Platform.iOS;
[assembly: ExportRenderer(typeof(CustomWebView), typeof(CustomWebViewRenderer))]
namespace WebViewBackgroundSample.iOS
{
public class CustomWebViewRenderer : WebViewRenderer
{
protected override void OnElementChanged(VisualElementChangedEventArgs e)
{
base.OnElementChanged(e);
// 背景を透過
this.Opaque = false;
this.BackgroundColor = UIColor.Clear;
}
}
}
2. カスタムコントロールが背景として張り付くようXamlを修正
今までのXamlを以下のように改造すると、CustomWebViewを背景に使えるようになります。
MainPage.xamlの背景にCustomWebViewを貼り付けます。
2.1. ContentPageに、カスタムコントロールを利用するためのプロパティを追加する。
<?xml version="1.0" encoding="utf-8"?>
<ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:prism="clr-namespace:Prism.Mvvm;assembly=Prism.Forms"
xmlns:controls="clr-namespace:WebViewBackgroundSample;assembly=WebViewBackgroundSample" ←追加部分
prism:ViewModelLocator.AutowireViewModel="True"
x:Class="WebViewBackgroundSample.Views.MainPage"
Title="MainPage">
(以下省略)
2.2. 元々ContentPageの直下にあったStackLayoutを、Gridの中に移動させてCustomWebViewと並べる。
<?xml version="1.0" encoding="utf-8"?>
<ContentPage (プロパティ省略)>
<StackLayout (プロパティ省略)>
(元々配置していた諸々のコントロール)
</StackLayout>
</ContentPage>
<?xml version="1.0" encoding="utf-8"?>
<ContentPage (プロパティ省略)>
<StackLayout HorizontalOptions="Center" VerticalOptions="Center">
<Grid>
<controls:CustomWebView HorizontalOptions="FillAndExpand"
VerticalOptions="FillAndExpand"
WidthRequest="5000"
HeightRequest="5000"
Source="{Binding Url}">
</controls:CustomWebView>
<StackLayout (プロパティ省略)>
(元々配置していた諸々のコントロール)
</StackLayout>
</Grid>
</StackLayout>
</ContentPage>
WidthRequestとHeightRequestをとりあえず適当にでかくしておけば、CustomWebViewが全画面サイズで張り付いてくれます。
参照するHTMLファイルのパスをViewModelから制御できるようにするため、CustomWebViewのSourceは変数にバインディングするようにします。
3. 背景にするWeb系ファイルの用意・共通プロジェクトへの追加
用意したHTML、CSS、JavaScriptの各ファイルを、以下のように共通プロジェクトに追加します。
各ファイルを設置する階層はお好みです。
共通プロジェクトに置いた上でプラットフォーム固有のプロジェクトがそれを参照するようにすることで、背景部分のソースを共通化できます。
注意
HTMLからCSSやJavaScriptを参照させる場合は、作った階層に対応した相対パスでの指定ではなく、ファイル名のみの指定にしてください。
これらのファイルは各プラットフォームのリソースフォルダに配置されることになりますが、そのときには全て同じ階層に置かれます。
HTMLからのCSS・JavaScriptの参照は、例えば以下のような具合です。
<html>
<head>
<meta charset="UTF-8">
<title>Transmission Background</title>
<link rel="stylesheet" href="Background_Transmission.css">
<script src="jquery-2.1.4.min.js"></script>
<script src="Background_Transmission.js"></script>
</head>
<body>
(省略)
</body>
</html>
4. [プラットフォーム別]Web系ファイルへのリンクの作成
4.1. Android側のリンクの作成
Androidのプロジェクト内に初めから置かれているAssetsフォルダ内にリンクを作成します。
4.1.1. Assetsフォルダを右クリックし、「追加」→「ファイルを追加」を選択する。
4.1.2. 共通プロジェクトに追加しておいたWeb系ファイルを選択する。
4.1.3. 「どう追加するか」を聞かれるので、「ファイルに対するリンクを追加する」を選択する。
この手順を、必要なファイル全てに対して行います。
最終的には、以下のようにAssetsフォルダ直下にWeb系リソースへのリンクが並ぶ形になります。
4.2. iOS側のリンクの作成
同様の手順を、iOSのプロジェクト内にあるResourcesフォルダに対して行います。
なぜリソースフォルダ内に共通プロジェクトと同じようなフォルダ階層を作らないか?
理由はよく分かりませんが、そうしてしまうとビルド時にリンク対象のファイルを取得することができなくなり、ビルドが通らなくなります。
5. ローカルのHTMLファイルを読み込む
サンプルアプリでは、タップしたボタンに応じて背景のエフェクトが変わるようにしています。
「ボタンのタップに紐付いたCommandで、変数Url(この変数がCustomWebViewのSourceにバインドされている)を書き換える」ということをしています。
この部分は、以下のような実装になっています。
_localHtmlFileRootPath = _resourceUtility.GetHtmlFileRootPath();
// マトリックス風の背景に変更するボタンに紐付いているコマンド
ChangeBackgroundMatrixCommand = new DelegateCommand(() => {
Url = _localHtmlFileRootPath + "Background_Matrix.html";
});
// 何かを発信している風の背景に変更するボタンに紐付いているコマンド
ChangeBackgroundTransmissionCommand = new DelegateCommand(() =>
{
Url = _localHtmlFileRootPath + "Background_Transmission.html";
});
// 何もない背景に変更するボタンに紐付いているコマンド
ChangeBackgroundBlankCommand = new DelegateCommand(() =>
{
Url = _localHtmlFileRootPath + "Background_Blank.html";
});
この中にある_resourceUtility.GetHtmlFileRootPath()は、プラットフォーム固有のリソースフォルダのパスを返すように作ったdependency serviceです。
プラットフォームごとの実装内容は以下のようになっています。
- Android
public string GetHtmlFileRootPath()
{
return "file:///android_asset/";
}
- iOS
public string GetHtmlFileRootPath()
{
return NSBundle.MainBundle.BundleUrl.ToString();
}