henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 1 | /* |
kjellander | b24317b | 2016-02-10 07:54:43 -0800 | [diff] [blame] | 2 | * Copyright 2013 The WebRTC project authors. All Rights Reserved. |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 3 | * |
kjellander | b24317b | 2016-02-10 07:54:43 -0800 | [diff] [blame] | 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 9 | */ |
| 10 | |
| 11 | package org.webrtc; |
| 12 | |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 13 | import java.util.IdentityHashMap; |
Magnus Jedvert | 6062f37 | 2017-11-16 16:53:12 +0100 | [diff] [blame] | 14 | import java.util.ArrayList; |
| 15 | import java.util.List; |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 16 | |
| 17 | /** Java version of VideoTrackInterface. */ |
| 18 | public class VideoTrack extends MediaStreamTrack { |
Magnus Jedvert | 6062f37 | 2017-11-16 16:53:12 +0100 | [diff] [blame] | 19 | private final List<VideoRenderer> renderers = new ArrayList<>(); |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 20 | private final IdentityHashMap<VideoSink, Long> sinks = new IdentityHashMap<VideoSink, Long>(); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 21 | |
| 22 | public VideoTrack(long nativeTrack) { |
| 23 | super(nativeTrack); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 24 | } |
| 25 | |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 26 | /** |
| 27 | * Adds a VideoSink to the track. |
| 28 | * |
| 29 | * A track can have any number of VideoSinks. VideoSinks will replace |
| 30 | * renderers. However, converting old style texture frames will involve costly |
| 31 | * conversion to I420 so it is not recommended to upgrade before all your |
| 32 | * sources produce VideoFrames. |
| 33 | */ |
| 34 | public void addSink(VideoSink sink) { |
Magnus Jedvert | b7700d3 | 2018-06-13 17:26:59 +0200 | [diff] [blame^] | 35 | if (sink == null) { |
| 36 | throw new IllegalArgumentException("The VideoSink is not allowed to be null"); |
| 37 | } |
| 38 | // We allow calling addSink() with the same sink multiple times. This is similar to the C++ |
| 39 | // VideoTrack::AddOrUpdateSink(). |
| 40 | if (!sinks.containsKey(sink)) { |
| 41 | final long nativeSink = nativeWrapSink(sink); |
| 42 | sinks.put(sink, nativeSink); |
| 43 | nativeAddSink(nativeTrack, nativeSink); |
| 44 | } |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 45 | } |
| 46 | |
| 47 | /** |
| 48 | * Removes a VideoSink from the track. |
| 49 | * |
| 50 | * If the VideoSink was not attached to the track, this is a no-op. |
| 51 | */ |
| 52 | public void removeSink(VideoSink sink) { |
Magnus Jedvert | b7700d3 | 2018-06-13 17:26:59 +0200 | [diff] [blame^] | 53 | final Long nativeSink = sinks.remove(sink); |
| 54 | if (nativeSink != null) { |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 55 | nativeRemoveSink(nativeTrack, nativeSink); |
| 56 | nativeFreeSink(nativeSink); |
| 57 | } |
| 58 | } |
| 59 | |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 60 | public void addRenderer(VideoRenderer renderer) { |
| 61 | renderers.add(renderer); |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 62 | nativeAddSink(nativeTrack, renderer.nativeVideoRenderer); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 63 | } |
| 64 | |
| 65 | public void removeRenderer(VideoRenderer renderer) { |
| 66 | if (!renderers.remove(renderer)) { |
| 67 | return; |
| 68 | } |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 69 | nativeRemoveSink(nativeTrack, renderer.nativeVideoRenderer); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 70 | renderer.dispose(); |
| 71 | } |
| 72 | |
Sami Kalliomäki | bde473e | 2017-10-30 13:34:41 +0100 | [diff] [blame] | 73 | @Override |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 74 | public void dispose() { |
Magnus Jedvert | 6062f37 | 2017-11-16 16:53:12 +0100 | [diff] [blame] | 75 | for (VideoRenderer renderer : renderers) { |
| 76 | nativeRemoveSink(nativeTrack, renderer.nativeVideoRenderer); |
| 77 | renderer.dispose(); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 78 | } |
Magnus Jedvert | 6062f37 | 2017-11-16 16:53:12 +0100 | [diff] [blame] | 79 | renderers.clear(); |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 80 | for (long nativeSink : sinks.values()) { |
| 81 | nativeRemoveSink(nativeTrack, nativeSink); |
| 82 | nativeFreeSink(nativeSink); |
| 83 | } |
| 84 | sinks.clear(); |
fischman@webrtc.org | 32001ef | 2013-08-12 23:26:21 +0000 | [diff] [blame] | 85 | super.dispose(); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 86 | } |
| 87 | |
Magnus Jedvert | 84d8ae5 | 2017-12-20 15:12:10 +0100 | [diff] [blame] | 88 | private static native void nativeAddSink(long track, long nativeSink); |
| 89 | private static native void nativeRemoveSink(long track, long nativeSink); |
sakal | 0ba43b5 | 2017-08-14 05:17:49 -0700 | [diff] [blame] | 90 | private static native long nativeWrapSink(VideoSink sink); |
Magnus Jedvert | 84d8ae5 | 2017-12-20 15:12:10 +0100 | [diff] [blame] | 91 | private static native void nativeFreeSink(long sink); |
henrike@webrtc.org | 28e2075 | 2013-07-10 00:45:36 +0000 | [diff] [blame] | 92 | } |