├── .gitignore
├── LICENSE
├── README.md
├── build.gradle
├── gradle.properties
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── gradlew
├── gradlew.bat
├── sample
├── build.gradle
└── src
│ └── main
│ ├── AndroidManifest.xml
│ ├── java
│ └── com
│ │ └── wroclawstudio
│ │ └── example
│ │ ├── Constants.java
│ │ ├── ListExampleActivity.java
│ │ ├── MainActivity.java
│ │ └── ScrollExampleActivity.java
│ └── res
│ ├── drawable-hdpi
│ └── ic_launcher.png
│ ├── drawable-mdpi
│ └── ic_launcher.png
│ ├── drawable-xhdpi
│ └── ic_launcher.png
│ ├── drawable-xxhdpi
│ └── ic_launcher.png
│ ├── layout
│ ├── activity_list_example.xml
│ ├── activity_main.xml
│ ├── activity_scroll_exmaple.xml
│ └── row_main.xml
│ └── values
│ ├── dimens.xml
│ └── strings.xml
├── settings.gradle
└── ytLib
├── build.gradle
├── libs
├── YouTubeAndroidPlayerApi.jar
└── guava-18.0.jar
└── src
└── main
├── AndroidManifest.xml
└── java
└── com
└── wroclawstudio
└── ytoverlay
├── LogHelper.java
└── YoutubeOverlayFragment.java
/.gitignore:
--------------------------------------------------------------------------------
1 | # Created by https://www.gitignore.io
2 |
3 | ### Android ###
4 | # Built application files
5 | *.apk
6 | *.ap_
7 |
8 | # Files for the Dalvik VM
9 | *.dex
10 |
11 | # Java class files
12 | *.class
13 |
14 | # Generated files
15 | bin/
16 | gen/
17 |
18 | # Gradle files
19 | .gradle/
20 | build/
21 |
22 | # Local configuration file (sdk path, etc)
23 | local.properties
24 |
25 | # Proguard folder generated by Eclipse
26 | proguard/
27 |
28 | # Log Files
29 | *.log
30 |
31 |
32 | ### Intellij ###
33 | # Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm
34 |
35 | *.iml
36 |
37 | ## Directory-based project format:
38 | .idea/
39 | # if you remove the above rule, at least ignore the following:
40 |
41 | # User-specific stuff:
42 | # .idea/workspace.xml
43 | # .idea/tasks.xml
44 | # .idea/dictionaries
45 |
46 | # Sensitive or high-churn files:
47 | # .idea/dataSources.ids
48 | # .idea/dataSources.xml
49 | # .idea/sqlDataSources.xml
50 | # .idea/dynamic.xml
51 | # .idea/uiDesigner.xml
52 |
53 | # Gradle:
54 | # .idea/gradle.xml
55 | # .idea/libraries
56 |
57 | # Mongo Explorer plugin:
58 | # .idea/mongoSettings.xml
59 |
60 | ## File-based project format:
61 | *.ipr
62 | *.iws
63 |
64 | ## Plugin-specific files:
65 |
66 | # IntelliJ
67 | out/
68 |
69 | # mpeltonen/sbt-idea plugin
70 | .idea_modules/
71 |
72 | # JIRA plugin
73 | atlassian-ide-plugin.xml
74 |
75 | # Crashlytics plugin (for Android Studio and IntelliJ)
76 | com_crashlytics_export_strings.xml
77 | crashlytics.properties
78 | crashlytics-build.properties
79 |
80 |
81 | ### Gradle ###
82 | .gradle
83 | build/
84 |
85 | # Ignore Gradle GUI config
86 | gradle-app.setting
87 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2015 Jakub
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
23 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | [](https://android-arsenal.com/details/1/1324)
2 |
3 |
4 | AndroidYoutubeOverlay
5 | =====================
6 |
7 | Library for handling Youtube Player in scrollable Views (ListView, GridView, ScrollView...)
8 |
9 | Features:
10 | ---------
11 |
12 | * Allows seamless transition between portrait and landscape (without rebufering)
13 | * Allows playback of one video at time
14 | * Attaches it self to view in scrollable container and scrolls with it
15 | * Supports AbsListView and ScrollView
16 | * Works from API level 17 (like YT API)
17 |
18 | Todo:
19 | -----
20 |
21 | * Test with Horizontal ScrollView
22 | * Added Touch event passing to underlying scrollable View
23 | * Added RecyclerView support
24 |
25 | How to Use:
26 | -----------
27 |
28 | - Add following line to Activity declaration in AndroidManifest.xml, which will handle YT playback
29 | ```
30 | android:configChanges="orientation|keyboardHidden|screenSize"
31 | ```
32 |
33 | - Use builder to create Fragment instance (Remember to set proper YT_DEVELOPER_KEY for your own project)
34 | ```
35 | ytPlayer = YoutubeOverlayFragment.newBuilder(YT_DEVELOPER_KEY, this).setListId(SCROLLABLE_CONTAINER_ID).buildAndAdd();
36 | ```
37 |
38 | - Handle view click in your adapter getView()
39 | ```
40 | @Override
41 | public View getView(final int position, View convertView, ViewGroup parent) {
42 | (...)
43 | final String videoId = getItem(position);
44 | holder.rowMainImage.setOnClickListener(new View.OnClickListener() {
45 | @Override
46 | public void onClick(View v) {
47 | ytPlayer.onClick(holder.rowMainImage, videoId, position);
48 | }
49 | });
50 | return convertView;
51 | }
52 | ```
53 |
54 | - Or in your ScrollView
55 | ```
56 | @Override
57 | protected void onCreate(Bundle savedInstanceState) {
58 | (...)
59 | final View img = findViewById(R.id.rowMainImage);
60 | img.setOnClickListener(new View.OnClickListener() {
61 | @Override
62 | public void onClick(View v) {
63 | ytPlayer.onClick(img, VIDEO_ID);
64 | }
65 | });
66 | }
67 | ```
68 |
69 | - Override onBackPressed, so fragment can handle back button while in landscape
70 | ```
71 | @Override
72 | public void onBackPressed() {
73 | if (ytPlayer.onBackPressed()) {
74 | // handled by ytPlayer fragment
75 | } else {
76 | super.onBackPressed();
77 | }
78 | }
79 | ```
80 |
--------------------------------------------------------------------------------
/build.gradle:
--------------------------------------------------------------------------------
1 | // Top-level build file where you can add configuration options common to all sub-projects/modules.
2 |
3 | buildscript {
4 | repositories {
5 | jcenter()
6 | }
7 | dependencies {
8 | classpath 'com.android.tools.build:gradle:1.0.0'
9 |
10 | // NOTE: Do not place your application dependencies here; they belong
11 | // in the individual module build.gradle files
12 | }
13 | }
14 |
15 | allprojects {
16 | repositories {
17 | jcenter()
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Wed Apr 10 15:27:10 PDT 2013
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.2.1-all.zip
7 |
--------------------------------------------------------------------------------
/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # For Cygwin, ensure paths are in UNIX format before anything is touched.
46 | if $cygwin ; then
47 | [ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
48 | fi
49 |
50 | # Attempt to set APP_HOME
51 | # Resolve links: $0 may be a link
52 | PRG="$0"
53 | # Need this for relative symlinks.
54 | while [ -h "$PRG" ] ; do
55 | ls=`ls -ld "$PRG"`
56 | link=`expr "$ls" : '.*-> \(.*\)$'`
57 | if expr "$link" : '/.*' > /dev/null; then
58 | PRG="$link"
59 | else
60 | PRG=`dirname "$PRG"`"/$link"
61 | fi
62 | done
63 | SAVED="`pwd`"
64 | cd "`dirname \"$PRG\"`/" >&-
65 | APP_HOME="`pwd -P`"
66 | cd "$SAVED" >&-
67 |
68 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
69 |
70 | # Determine the Java command to use to start the JVM.
71 | if [ -n "$JAVA_HOME" ] ; then
72 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
73 | # IBM's JDK on AIX uses strange locations for the executables
74 | JAVACMD="$JAVA_HOME/jre/sh/java"
75 | else
76 | JAVACMD="$JAVA_HOME/bin/java"
77 | fi
78 | if [ ! -x "$JAVACMD" ] ; then
79 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
80 |
81 | Please set the JAVA_HOME variable in your environment to match the
82 | location of your Java installation."
83 | fi
84 | else
85 | JAVACMD="java"
86 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
87 |
88 | Please set the JAVA_HOME variable in your environment to match the
89 | location of your Java installation."
90 | fi
91 |
92 | # Increase the maximum file descriptors if we can.
93 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
94 | MAX_FD_LIMIT=`ulimit -H -n`
95 | if [ $? -eq 0 ] ; then
96 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
97 | MAX_FD="$MAX_FD_LIMIT"
98 | fi
99 | ulimit -n $MAX_FD
100 | if [ $? -ne 0 ] ; then
101 | warn "Could not set maximum file descriptor limit: $MAX_FD"
102 | fi
103 | else
104 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
105 | fi
106 | fi
107 |
108 | # For Darwin, add options to specify how the application appears in the dock
109 | if $darwin; then
110 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
111 | fi
112 |
113 | # For Cygwin, switch paths to Windows format before running java
114 | if $cygwin ; then
115 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
116 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
117 |
118 | # We build the pattern for arguments to be converted via cygpath
119 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
120 | SEP=""
121 | for dir in $ROOTDIRSRAW ; do
122 | ROOTDIRS="$ROOTDIRS$SEP$dir"
123 | SEP="|"
124 | done
125 | OURCYGPATTERN="(^($ROOTDIRS))"
126 | # Add a user-defined pattern to the cygpath arguments
127 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
128 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
129 | fi
130 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
131 | i=0
132 | for arg in "$@" ; do
133 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
134 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
135 |
136 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
137 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
138 | else
139 | eval `echo args$i`="\"$arg\""
140 | fi
141 | i=$((i+1))
142 | done
143 | case $i in
144 | (0) set -- ;;
145 | (1) set -- "$args0" ;;
146 | (2) set -- "$args0" "$args1" ;;
147 | (3) set -- "$args0" "$args1" "$args2" ;;
148 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
149 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
150 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
151 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
152 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
153 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
154 | esac
155 | fi
156 |
157 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
158 | function splitJvmOpts() {
159 | JVM_OPTS=("$@")
160 | }
161 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
162 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
163 |
164 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
165 |
--------------------------------------------------------------------------------
/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/sample/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion 21
5 | buildToolsVersion "20.0.0"
6 |
7 | defaultConfig {
8 | applicationId "com.wroclawstudio.example"
9 | minSdkVersion 17
10 | targetSdkVersion 21
11 | versionCode 1
12 | versionName "1.0"
13 | }
14 |
15 | }
16 |
17 | dependencies {
18 | compile fileTree(dir: 'libs', include: ['*.jar'])
19 | compile project(":ytLib");
20 | }
21 |
--------------------------------------------------------------------------------
/sample/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 |
6 |
11 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
25 |
29 |
30 |
31 |
32 |
33 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/wroclawstudio/example/Constants.java:
--------------------------------------------------------------------------------
1 | package com.wroclawstudio.example;
2 |
3 | /**
4 | * Created by jakubszczygiel on 04/01/15.
5 | */
6 | public interface Constants {
7 | public static final String YT_DEVELOPER_KEY = "AIzaSyDugQYgHDcyOwx1pj-qBDPOFjFjouhS7Ms";
8 | }
9 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/wroclawstudio/example/ListExampleActivity.java:
--------------------------------------------------------------------------------
1 | package com.wroclawstudio.example;
2 |
3 | import android.app.Activity;
4 | import android.content.Context;
5 | import android.os.Bundle;
6 | import android.view.LayoutInflater;
7 | import android.view.View;
8 | import android.view.ViewGroup;
9 | import android.widget.ArrayAdapter;
10 | import android.widget.ListView;
11 |
12 | import com.wroclawstudio.ytoverlay.YoutubeOverlayFragment;
13 |
14 | /**
15 | * Created by jakubszczygiel on 15/11/14.
16 | */
17 | public class ListExampleActivity extends Activity {
18 |
19 |
20 | public String[] youtubeUrls = new String[]{
21 | "1KOaT1vdLmc",
22 | "PZbkF-15ObM",
23 | "tvY7Nw1i6Kw",
24 | "YwmbbcMHiQ0",
25 | "2b03hoW0TKc",
26 | };
27 | private YoutubeOverlayFragment ytPlayer;
28 |
29 | @Override
30 | protected void onCreate(Bundle savedInstanceState) {
31 | super.onCreate(savedInstanceState);
32 | setContentView(R.layout.activity_list_example);
33 |
34 | ListView list = (ListView) findViewById(android.R.id.list);
35 | list.setAdapter(new YoutubeListAdapter(this, youtubeUrls));
36 |
37 | ytPlayer = YoutubeOverlayFragment.newBuilder(Constants.YT_DEVELOPER_KEY, this).setScrollableViewId(android.R.id.list).buildAndAdd();
38 |
39 | }
40 |
41 | class YoutubeListAdapter extends ArrayAdapter {
42 |
43 | private final LayoutInflater inflater;
44 |
45 | public YoutubeListAdapter(Context context, String[] list) {
46 | super(context, 0, list);
47 | this.inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
48 | }
49 |
50 | @Override
51 | public View getView(final int position, View convertView, ViewGroup parent) {
52 | final ViewHolder holder;
53 | if (convertView == null) {
54 | convertView = inflater.inflate(R.layout.row_main, parent, false);
55 | holder = new ViewHolder();
56 | holder.rowMainImage = convertView.findViewById(R.id.rowMainImage);
57 | convertView.setTag(holder);
58 | } else {
59 | holder = (ViewHolder) convertView.getTag();
60 | }
61 |
62 | final String videoId = getItem(position);
63 | holder.rowMainImage.setOnClickListener(new View.OnClickListener() {
64 | @Override
65 | public void onClick(View v) {
66 | ytPlayer.onClick(holder.rowMainImage, videoId, position);
67 | }
68 | });
69 |
70 | return convertView;
71 | }
72 |
73 | class ViewHolder {
74 | View rowMainImage;
75 | }
76 | }
77 |
78 | @Override
79 | public void onBackPressed() {
80 | if (ytPlayer.onBackPressed()) {
81 | // handled by fragment
82 | } else {
83 | super.onBackPressed();
84 | }
85 | }
86 | }
87 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/wroclawstudio/example/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.wroclawstudio.example;
2 |
3 | import android.app.Activity;
4 | import android.content.Intent;
5 | import android.os.Bundle;
6 | import android.view.View;
7 |
8 | /**
9 | * Created by jakubszczygiel on 04/01/15.
10 | */
11 | public class MainActivity extends Activity {
12 | @Override
13 | protected void onCreate(Bundle savedInstanceState) {
14 | super.onCreate(savedInstanceState);
15 | setContentView(R.layout.activity_main);
16 | findViewById(R.id.buttonListExample).setOnClickListener(new View.OnClickListener() {
17 | @Override
18 | public void onClick(View v) {
19 | Intent intent= new Intent(MainActivity.this, ListExampleActivity.class);
20 | startActivity(intent);
21 | }
22 | });
23 |
24 | findViewById(R.id.buttonScrollExample).setOnClickListener(new View.OnClickListener() {
25 | @Override
26 | public void onClick(View v) {
27 | Intent intent= new Intent(MainActivity.this, ScrollExampleActivity.class);
28 | startActivity(intent);
29 | }
30 | });
31 | }
32 | }
33 |
--------------------------------------------------------------------------------
/sample/src/main/java/com/wroclawstudio/example/ScrollExampleActivity.java:
--------------------------------------------------------------------------------
1 | package com.wroclawstudio.example;
2 |
3 | import android.app.Activity;
4 | import android.os.Bundle;
5 | import android.view.View;
6 |
7 | import com.wroclawstudio.ytoverlay.YoutubeOverlayFragment;
8 |
9 | /**
10 | * Created by jakubszczygiel on 04/01/15.
11 | */
12 | public class ScrollExampleActivity extends Activity {
13 |
14 | private static final String VIDEO_ID = "2b03hoW0TKc";
15 |
16 | private YoutubeOverlayFragment ytPlayer;
17 |
18 | @Override
19 | protected void onCreate(Bundle savedInstanceState) {
20 | super.onCreate(savedInstanceState);
21 | setContentView(R.layout.activity_scroll_exmaple);
22 | ytPlayer = YoutubeOverlayFragment.newBuilder(Constants.YT_DEVELOPER_KEY, this).setScrollableViewId(R.id.scrollView).buildAndAdd();
23 |
24 | final View img = findViewById(R.id.rowMainImage);
25 | img.setOnClickListener(new View.OnClickListener() {
26 | @Override
27 | public void onClick(View v) {
28 | ytPlayer.onClick(img, VIDEO_ID);
29 | }
30 | });
31 | }
32 |
33 | @Override
34 | public void onBackPressed() {
35 | if (ytPlayer.onBackPressed()) {
36 | // handled by fragment
37 | } else {
38 | super.onBackPressed();
39 | }
40 | }
41 | }
42 |
--------------------------------------------------------------------------------
/sample/src/main/res/drawable-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/sample/src/main/res/drawable-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/drawable-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/sample/src/main/res/drawable-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/drawable-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/sample/src/main/res/drawable-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/drawable-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/sample/src/main/res/drawable-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/sample/src/main/res/layout/activity_list_example.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
10 |
11 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
6 |
7 |
12 |
13 |
18 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/activity_scroll_exmaple.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
8 |
9 |
14 |
15 |
20 |
21 |
28 |
29 |
33 |
34 |
35 |
36 |
37 |
--------------------------------------------------------------------------------
/sample/src/main/res/layout/row_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 |
14 |
--------------------------------------------------------------------------------
/sample/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 10dp
4 |
--------------------------------------------------------------------------------
/sample/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | YoutubeOverlayExample
5 | List Example
6 | ScrollView Example
7 | Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam ligula libero, ultricies et ante vel, faucibus posuere justo. Suspendisse varius, ex eget sodales consequat, ante odio eleifend arcu, at pulvinar ipsum metus eget erat. Aliquam tristique mi in feugiat fermentum. Curabitur nisl elit, blandit non risus a, porta auctor diam. Cras in tortor non dui interdum gravida sed et est. Aenean interdum in nibh et laoreet. Vivamus in interdum velit, eu aliquet mauris. Integer lacinia venenatis ante. Quisque ac mi ut augue rhoncus vehicula sed ac sem. Fusce tempus eleifend eros.
8 |
9 |
10 |
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':sample', 'ytLib'
2 |
--------------------------------------------------------------------------------
/ytLib/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.library'
2 |
3 | android {
4 | compileSdkVersion 21
5 | buildToolsVersion "20.0.0"
6 |
7 | defaultConfig {
8 | minSdkVersion 17
9 | targetSdkVersion 21
10 | versionCode 1
11 | versionName "1.0"
12 | }
13 |
14 | }
15 |
16 | dependencies {
17 | compile fileTree(dir: 'libs', include: ['*.jar'])
18 | }
19 |
--------------------------------------------------------------------------------
/ytLib/libs/YouTubeAndroidPlayerApi.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/ytLib/libs/YouTubeAndroidPlayerApi.jar
--------------------------------------------------------------------------------
/ytLib/libs/guava-18.0.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jszczygiel/AndroidYoutubeOverlay/3c4bbc71e8340b43e7a42f75eb66bcf8158210a2/ytLib/libs/guava-18.0.jar
--------------------------------------------------------------------------------
/ytLib/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/ytLib/src/main/java/com/wroclawstudio/ytoverlay/LogHelper.java:
--------------------------------------------------------------------------------
1 | package com.wroclawstudio.ytoverlay;
2 |
3 | import android.util.Log;
4 |
5 | /**
6 | * Created by jakubszczygiel on 03/01/15.
7 | */
8 | public class LogHelper {
9 | public static final String LOG_TAG = "YtOverlay";
10 |
11 | public static void logMessage(String message) {
12 | if(BuildConfig.DEBUG) {
13 | Log.i(LOG_TAG, message);
14 | }
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/ytLib/src/main/java/com/wroclawstudio/ytoverlay/YoutubeOverlayFragment.java:
--------------------------------------------------------------------------------
1 | package com.wroclawstudio.ytoverlay;
2 |
3 | import android.app.Activity;
4 | import android.app.FragmentManager;
5 | import android.content.Context;
6 | import android.content.res.Configuration;
7 | import android.graphics.Color;
8 | import android.graphics.Point;
9 | import android.os.Bundle;
10 | import android.view.Display;
11 | import android.view.LayoutInflater;
12 | import android.view.View;
13 | import android.view.ViewGroup;
14 | import android.view.ViewTreeObserver;
15 | import android.view.Window;
16 | import android.view.WindowManager;
17 | import android.widget.AbsListView;
18 | import android.widget.ListView;
19 | import android.widget.ScrollView;
20 |
21 | import com.google.android.youtube.player.YouTubeInitializationResult;
22 | import com.google.android.youtube.player.YouTubePlayer;
23 | import com.google.android.youtube.player.YouTubePlayerFragment;
24 | import com.google.common.base.Preconditions;
25 | import com.google.common.base.Strings;
26 | import com.google.common.primitives.Ints;
27 |
28 | import java.util.ArrayList;
29 |
30 | public class YoutubeOverlayFragment extends YouTubePlayerFragment implements YouTubePlayer.OnInitializedListener {
31 |
32 | private static final String HIDEABLE_VIEWS = "HIDEABLE_VIEWS";
33 | private static final String SCROLLABLE_VIEW_ID = "SCROLLABLE_VIEW_ID";
34 | private static final String VIDEO_ID = "VIDEO_ID";
35 | private static final String YT_DEVELOPER_KEY = "YT_DEVELOPER_KEY";
36 |
37 | private YouTubePlayer player; // yt player
38 | private String videoId; // yt video id
39 | private boolean fullscreen; // holds current player playback mode
40 | private String ytKey;
41 |
42 | private View videoContainer; // video container
43 | private View attachedImgContainer; // img container to which video container is attached
44 | private Display display; // used for getting screen size
45 | private View attachedListView; // listView to which fragment is attached
46 | private int[] hideableViews; // videoContainer ids that are hidden in fullscreen playback
47 | private View actionBarView; // actionBarView
48 |
49 | private YoutubeOverlayFragment() {
50 | setRetainInstance(true);
51 | }
52 |
53 |
54 | @Override
55 | public void onCreate(Bundle savedInstanceState) {
56 | LogHelper.logMessage("onCreate: " + savedInstanceState);
57 | super.onCreate(savedInstanceState);
58 | if (getArguments() != null) {
59 | this.videoId = getArguments().getString(VIDEO_ID, null);
60 | this.hideableViews = getArguments().getIntArray(HIDEABLE_VIEWS);
61 | this.attachedListView = getActivity().findViewById(getArguments().getInt(SCROLLABLE_VIEW_ID));
62 | this.ytKey = getArguments().getString(YT_DEVELOPER_KEY);
63 |
64 | }
65 | initialize(ytKey, this);
66 | WindowManager wm = (WindowManager) getActivity().getSystemService(Context.WINDOW_SERVICE);
67 | display = wm.getDefaultDisplay();
68 | actionBarView = getActionBarView(getActivity());
69 | }
70 |
71 | private View getActionBarView(Activity activity) {
72 | Window window = activity.getWindow();
73 | View v = window.getDecorView();
74 | int resId = getResources().getIdentifier("action_bar_container", "id", "android");
75 | return v.findViewById(resId);
76 | }
77 |
78 | /**
79 | * Set fragment color to black and hides it after creation
80 | *
81 | * @return fragemnt videoContainer container
82 | */
83 | @Override
84 | public View onCreateView(LayoutInflater layoutInflater, ViewGroup viewGroup, Bundle bundle) {
85 | videoContainer = super.onCreateView(layoutInflater, viewGroup, bundle);
86 | if (videoContainer != null) {
87 | videoContainer.setBackgroundColor(Color.BLACK);
88 | videoContainer.setVisibility(View.GONE);
89 | }
90 | return videoContainer;
91 | }
92 |
93 | @Override
94 | public void onInitializationSuccess(YouTubePlayer.Provider provider, final YouTubePlayer player, boolean restored) {
95 | LogHelper.logMessage("onInitializationSuccess: " + restored + " videoId: " + videoId);
96 | this.player = player;
97 | player.addFullscreenControlFlag(YouTubePlayer.FULLSCREEN_FLAG_CONTROL_SYSTEM_UI);
98 | player.addFullscreenControlFlag(YouTubePlayer.FULLSCREEN_FLAG_CUSTOM_LAYOUT);
99 | player.setOnFullscreenListener(new YouTubePlayer.OnFullscreenListener() {
100 |
101 | @Override
102 | public void onFullscreen(boolean fullscreen) {
103 | YoutubeOverlayFragment.this.fullscreen = fullscreen;
104 | }
105 | });
106 | if (!restored && videoId != null) {
107 | player.loadVideo(videoId);
108 | }
109 | }
110 |
111 | /**
112 | * Handles screen orientation change. Assigns proper size and position of video container based on screen orientation.
113 | *
114 | * @param newConfig new screen configuration
115 | */
116 | @Override
117 | public void onConfigurationChanged(Configuration newConfig) {
118 | LogHelper.logMessage("onConfigurationChanged: is landscape" + (newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE));
119 | super.onConfigurationChanged(newConfig);
120 | if (videoId != null && player != null) {
121 | ViewGroup.LayoutParams params = videoContainer.getLayoutParams();
122 | if (newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE) {
123 | if (getActivity() != null) {
124 | actionBarView.setVisibility(View.GONE);
125 | for (int id : hideableViews) {
126 | getActivity().findViewById(id).setVisibility(View.GONE);
127 | }
128 | }
129 | Point size = new Point();
130 | display.getRealSize(size);
131 | params.height = size.y;
132 | params.width = size.x;
133 | LogHelper.logMessage("x:" + size.x + " y:" + size.y);
134 | videoContainer.setX(0);
135 | videoContainer.setY(0);
136 | player.setFullscreen(true);
137 |
138 | } else if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) {
139 | if (getActivity() != null) {
140 | actionBarView.setVisibility(View.VISIBLE);
141 | for (int id : hideableViews) {
142 | getActivity().findViewById(id).setVisibility(View.VISIBLE);
143 | }
144 | }
145 | setVideoPostion(this.attachedImgContainer);
146 | player.setFullscreen(false);
147 | }
148 | videoContainer.setLayoutParams(params);
149 | }
150 |
151 | }
152 |
153 |
154 | @Override
155 | public void onInitializationFailure(YouTubePlayer.Provider provider, YouTubeInitializationResult result) {
156 | LogHelper.logMessage("onInitializationFailure:" + result.name());
157 | this.player = null;
158 | }
159 |
160 | /**
161 | * Exit fullscreen mode of yt player
162 | */
163 | public void exitFullScreen() {
164 | player.setFullscreen(false);
165 | }
166 |
167 | /**
168 | * @return returns if fragment is in fullscreen mode
169 | */
170 | public boolean isFullScreen() {
171 | return fullscreen;
172 | }
173 |
174 | /**
175 | * Starts playback of passed youtubeID if player is available and youtubeID is not null
176 | *
177 | * @param videoId yt video id to be played
178 | */
179 | public void setVideoId(String videoId) {
180 | this.videoId = videoId;
181 | if (player != null && this.videoId != null) {
182 | player.loadVideo(this.videoId);
183 | videoContainer.setVisibility(View.VISIBLE);
184 | } else {
185 | ViewGroup.LayoutParams params = videoContainer.getLayoutParams();
186 | params.height = 0;
187 | params.width = 0;
188 | videoContainer.setLayoutParams(params);
189 | videoContainer.setVisibility(View.GONE);
190 | }
191 | }
192 |
193 | /**
194 | * Attaches to img container handling img container scrolling, which updates video container position.
195 | *
196 | * @param attachedContainer img container to which video container is going to be attached
197 | * @param position list item position of img container
198 | */
199 | private void attachToView(final View attachedContainer, final int position) {
200 | this.attachedImgContainer = attachedContainer;
201 | setVideoPostion(this.attachedImgContainer);
202 | this.attachedImgContainer.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() {
203 | @Override
204 | public void onScrollChanged() {
205 | if (!fullscreen) {
206 | setVideoPostion(YoutubeOverlayFragment.this.attachedImgContainer);
207 | }
208 | }
209 | });
210 | if (attachedListView instanceof AbsListView) {
211 | ((AbsListView) attachedListView).setOnScrollListener(new AbsListView.OnScrollListener() {
212 | @Override
213 | public void onScrollStateChanged(AbsListView view, int scrollState) {
214 |
215 | }
216 |
217 | @Override
218 | public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
219 | if ((position < (firstVisibleItem + visibleItemCount)) && (position >= firstVisibleItem)) {
220 | YoutubeOverlayFragment.this.videoContainer.setVisibility(View.VISIBLE);
221 | } else {
222 | YoutubeOverlayFragment.this.videoContainer.setVisibility(View.GONE);
223 | }
224 | }
225 | });
226 | }
227 |
228 | }
229 |
230 | private void attachToView(View attachedContainer) {
231 | this.attachedImgContainer = attachedContainer;
232 | setVideoPostion(this.attachedImgContainer);
233 | this.attachedImgContainer.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() {
234 | @Override
235 | public void onScrollChanged() {
236 | if (!fullscreen) {
237 | setVideoPostion(YoutubeOverlayFragment.this.attachedImgContainer);
238 | }
239 | }
240 | });
241 | }
242 |
243 | /**
244 | * Updates video container position based on attachedView current position
245 | *
246 | * @param attachedView videoContainer based on which video container position is updated
247 | */
248 | private void setVideoPostion(View attachedView) {
249 | int postion[] = new int[2];
250 | attachedView.getLocationOnScreen(postion);
251 | videoContainer.setX(postion[0]);
252 | videoContainer.setY(postion[1] - getStatusBarHeight() - getActionBarHeight());
253 | ViewGroup.LayoutParams params = videoContainer.getLayoutParams();
254 | params.height = attachedView.getHeight();
255 | params.width = attachedView.getWidth();
256 | videoContainer.setLayoutParams(params);
257 | }
258 |
259 | /**
260 | * Returns ActionBarHeight
261 | *
262 | * @return ActionBarHeight
263 | */
264 | private int getActionBarHeight() {
265 | if (actionBarView != null) {
266 | return actionBarView.getHeight();
267 | }
268 | return 0;
269 | }
270 |
271 | /**
272 | * Returns status bar height
273 | *
274 | * @return status bar height
275 | */
276 | private int getStatusBarHeight() {
277 | int result = 0;
278 | int resourceId = getResources().getIdentifier("status_bar_height", "dimen", "android");
279 | if (resourceId > 0) {
280 | result = getResources().getDimensionPixelSize(resourceId);
281 | }
282 | return result;
283 | }
284 |
285 | /**
286 | * Handles exiting from yt full screen videoContainer
287 | *
288 | * @param activity activity in which on backpressed is handled
289 | * @return returns if action was handled
290 | */
291 | public static boolean onBackPressed(Activity activity) {
292 | if (activity != null) {
293 | FragmentManager fragmentManager = activity.getFragmentManager();
294 | YoutubeOverlayFragment yt = (YoutubeOverlayFragment) fragmentManager.findFragmentByTag(YoutubeOverlayFragment.class.getName());
295 | if (yt != null) {
296 | yt.onBackPressed();
297 | }
298 | }
299 | return false;
300 | }
301 |
302 | /**
303 | * Handles exiting from yt full screen videoContainer
304 | *
305 | * @return returns if action was handled
306 | */
307 | public boolean onBackPressed() {
308 | if (isFullScreen()) {
309 | exitFullScreen();
310 | return true;
311 | }
312 | return false;
313 | }
314 |
315 | /**
316 | * Performs video playback of passed videoId in fragment video container.
317 | * Video container is attached to img container of an row item.
318 | *
319 | * @param activity activity context in which adapter is set
320 | * @param img img container to which video container is going to be attached
321 | * @param videoId yt video id which is going to be played
322 | * @param position list item position of img container
323 | */
324 | public static void onClick(Activity activity, View img, String videoId, int position) {
325 | if (activity != null) {
326 | FragmentManager fragmentManager = activity.getFragmentManager();
327 | YoutubeOverlayFragment yt = (YoutubeOverlayFragment) fragmentManager.findFragmentByTag(YoutubeOverlayFragment.class.getName());
328 | if (yt != null) {
329 | yt.onClick(img, videoId, position);
330 | }
331 | }
332 | }
333 |
334 | /**
335 | * Used for handling onclick in AbsListView. Overlay takes size of view passed.
336 | *
337 | * @param videoId yt video Id
338 | * @param view view to which yt overlay attaches it self
339 | * @param position position in list for handling player visibility
340 | */
341 | public void onClick(View view, String videoId, int position) {
342 | setVideoId(videoId);
343 | attachToView(view, position);
344 | }
345 |
346 | /**
347 | * Used for handling onclick in ScrollView. Overlay takes size of view passed.
348 | *
349 | * @param videoId yt video Id
350 | * @param view view to which yt overlay attaches it self
351 | */
352 | public void onClick(View view, String videoId) {
353 | setVideoId(videoId);
354 | attachToView(view);
355 | }
356 |
357 | public static Builder newBuilder(String ytKey, Activity activity) {
358 | return new Builder(ytKey, activity);
359 | }
360 |
361 | public static class Builder {
362 |
363 | private final String ytKey;
364 | private int scrollableViewId;
365 | private ArrayList hideableViews;
366 | private Activity activity;
367 |
368 | public Builder(String ytKey, Activity activity) {
369 | this.ytKey = ytKey;
370 | this.activity = activity;
371 | this.hideableViews = new ArrayList<>();
372 | }
373 |
374 | /**
375 | * Sets id of scrollable view to which overlay will attach. Required
376 | *
377 | * @param scrollableViewId viewId to hide
378 | * @return returns Builder
379 | */
380 | public Builder setScrollableViewId(int scrollableViewId) {
381 | this.scrollableViewId = scrollableViewId;
382 | return this;
383 | }
384 |
385 | /**
386 | * View id that is at the same hierarchy level as yt overlay. Will be hidden while player is in landscape mode
387 | * Optional
388 | *
389 | * @param viewId viewId to hide
390 | * @return returns Builder
391 | */
392 | public Builder addHideableView(int viewId) {
393 | hideableViews.add(viewId);
394 | return this;
395 | }
396 |
397 | /**
398 | * Builds Fragment and adds it to view hierarchy
399 | * @return returns YoutubeOverlayFragment instance that was added to hierarchy
400 | */
401 | public YoutubeOverlayFragment buildAndAdd() {
402 | Preconditions.checkNotNull(activity);
403 | Preconditions.checkArgument(!Strings.isNullOrEmpty(ytKey));
404 | Preconditions.checkArgument(scrollableViewId != 0);
405 |
406 | FragmentManager fragmentManager = activity.getFragmentManager();
407 | View scrollableView = activity.findViewById(scrollableViewId);
408 | YoutubeOverlayFragment yt = (YoutubeOverlayFragment) fragmentManager.findFragmentByTag(YoutubeOverlayFragment.class.getName());
409 | if (yt != null) {
410 | fragmentManager.beginTransaction().remove(yt).commit();
411 | fragmentManager.executePendingTransactions();
412 | }
413 | yt = new YoutubeOverlayFragment();
414 | Bundle bundle = new Bundle();
415 | bundle.putIntArray(YoutubeOverlayFragment.HIDEABLE_VIEWS, Ints.toArray(hideableViews));
416 | bundle.putString(YoutubeOverlayFragment.YT_DEVELOPER_KEY, ytKey);
417 | bundle.putInt(SCROLLABLE_VIEW_ID, scrollableViewId);
418 | yt.setArguments(bundle);
419 | fragmentManager.beginTransaction().add(((View) scrollableView.getParent()).getId(), yt, YoutubeOverlayFragment.class.getName()).commit();
420 | return yt;
421 | }
422 |
423 | }
424 | }
--------------------------------------------------------------------------------