summaryrefslogtreecommitdiff
path: root/VideoPlayerLite/VideoPlayerLite.Android/Renderers/MediaRenderingViewRenderer.cs
blob: 092ac45be87c77b5a836abeb13d3bfdc77661082 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
/*
 * Copyright (c) 2017 Samsung Electronics Co., Ltd
 *
 * Licensed under the Flora License, Version 1.1 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://floralicense.org/license/
 *
 * 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.
 */

using Android.Widget;
using Xamarin.Forms;
using Xamarin.Forms.Platform.Android;
using System.ComponentModel;
using VideoPlayerLite.Views;
using System.Threading;
using VideoPlayerLite.Android;
using Android.Util;

[assembly: ExportRenderer(typeof(MediaRenderingView), typeof(MediaRenderingViewRenderer))]
namespace VideoPlayerLite.Android
{
    /// <summary>
    /// MediaRenderingViewRenderer implements the VideoView Custom Renderer on the Android Platform.
    /// </summary>
    public class MediaRenderingViewRenderer : ViewRenderer<MediaRenderingView, VideoView>
    {
        private static readonly string TAG = "VideoViewRenderer";
        /// <summary>
        /// Timer for updating the playing position on the video player UI.
        /// </summary>
        private Timer playPositionTimer;

        /// <summary>
        /// When the Element, which is created on the Xamarin Forms, is changed, this method is going to be invoked.
        /// If the control is null, assign Android VideoView to the Control property with the SetNativeControl method.
        /// If OldElement is not null, unsubscribe VideoViewCompletion event from event handler.
        /// If NewElement is not null, subscribe VideoViewCompletion event to the event handler.
        /// </summary>
        /// <param name="e">Data of the event</param>
        protected override void OnElementChanged(ElementChangedEventArgs<MediaRenderingView> e)
        {
            base.OnElementChanged(e);

            if (Control == null)
            {
                SetNativeControl(new VideoView(Context));
            }

            if (e.OldElement != null)
            {
                Control.Completion -= VideoViewCompletion;
                StopTimer();
            }

            if (e.NewElement != null)
            {
                Control.Completion += VideoViewCompletion;
                StopTimer();
            }
        }

        /// <summary>
        /// OnElementPropertyChanged method is called whenever any bindable properties in the MediaRenderingView change MediaRenderingView.
        /// PlayingItem property will change VideoPlayer to start with new video item.
        /// MediaRenderingView.PlayerStatus property will change when VideoPlayer receive UI control event.
        /// </summary>
        /// <param name="sender">The object what got the event</param>
        /// <param name="e">Data of the event</param>
        protected override void OnElementPropertyChanged(object sender, PropertyChangedEventArgs e)
        {
            base.OnElementPropertyChanged(sender, e);

            if (e.PropertyName == MediaRenderingView.CurrentVideoProperty.PropertyName)
            {
                ((IElementController)Element).SetValueFromRenderer(MediaRenderingView.CurrentStatusProperty, PlayerStatus.Preparing);

                StopTimer();
                if (Control.IsPlaying)
                {
                    Control.StopPlayback();
                }
                Control.SetVideoPath(Element.VideoPath);

                ((IElementController)Element).SetValueFromRenderer(MediaRenderingView.CurrentStatusProperty, PlayerStatus.Playing);
            }
            else if (e.PropertyName == MediaRenderingView.CurrentStatusProperty.PropertyName)
            {
                switch (Element.CurrentStatus)
                {
                    case PlayerStatus.Playing:
                        Control.Start();
                        StartTimer();
                        break;
                    case PlayerStatus.Paused:
                        Control.Pause();
                        /// Should pause the timer
                        StopTimer();
                        break;
                    case PlayerStatus.Stopped:
                        Control.StopPlayback();
                        Control.SetVideoPath(Element.VideoPath);
                        StopTimer();
                        break;
                    default:
                        break;
                }
            }
        }

        /// <summary>
        /// It is called when playing is completed. Then, this method changes player status of mediaRendering view to the stopped status
        /// </summary>
        /// <param name="sender">The object what got the event</param>
        /// <param name="e">Data of the event</param>
        private void VideoViewCompletion(object sender, System.EventArgs e)
        {
            ((IElementController)Element).SetValueFromRenderer(MediaRenderingView.CurrentStatusProperty, PlayerStatus.Stopped);
        }

        /// <summary>
        /// Creates new timer for updating plyaing position at VideoPlayer
        /// Timer gets current playing position from Android VideoView every 250ms and updates the position.
        /// </summary>
        private void StartTimer()
        {
            if (playPositionTimer != null)
            {
                return;
            }

            playPositionTimer = new Timer((x) =>
            {
                Device.BeginInvokeOnMainThread(() =>
                {
                    if (playPositionTimer != null)
                    {
                        int playPosition = 0;

                        try
                        {
                            playPosition = (Control != null) ? Control.CurrentPosition : 0;
                        }
                        catch (System.Exception e)
                        {
                            playPosition = 0;
                            Log.Error(TAG, "Failed to get current position. ("+e.Message+")");
                        }

                        if (Element == null)
                        {
                            StopTimer();
                        }
                        else
                        {
                            ((IElementController)Element).SetValueFromRenderer(MediaRenderingView.CurrentPositionProperty, playPosition);
                        }
                    }
                });
            }, null, 0, 250);
        }
        /// <summary>
        /// Destroys the playing position update timer
        /// </summary>
        private void StopTimer()
        {
            playPositionTimer?.Dispose();
            playPositionTimer = null;
        }
    }
}