Showing
45 changed files
with
1401 additions
and
0 deletions
.gitignore
0 → 100644
.idea/.gitignore
0 → 100644
.idea/compiler.xml
0 → 100644
.idea/gradle.xml
0 → 100644
| 1 | +<?xml version="1.0" encoding="UTF-8"?> | ||
| 2 | +<project version="4"> | ||
| 3 | + <component name="GradleMigrationSettings" migrationVersion="1" /> | ||
| 4 | + <component name="GradleSettings"> | ||
| 5 | + <option name="linkedExternalProjectsSettings"> | ||
| 6 | + <GradleProjectSettings> | ||
| 7 | + <option name="testRunner" value="GRADLE" /> | ||
| 8 | + <option name="distributionType" value="DEFAULT_WRAPPED" /> | ||
| 9 | + <option name="externalProjectPath" value="$PROJECT_DIR$" /> | ||
| 10 | + <option name="modules"> | ||
| 11 | + <set> | ||
| 12 | + <option value="$PROJECT_DIR$" /> | ||
| 13 | + <option value="$PROJECT_DIR$/app" /> | ||
| 14 | + </set> | ||
| 15 | + </option> | ||
| 16 | + </GradleProjectSettings> | ||
| 17 | + </option> | ||
| 18 | + </component> | ||
| 19 | +</project> |
.idea/misc.xml
0 → 100644
| 1 | +<?xml version="1.0" encoding="UTF-8"?> | ||
| 2 | +<project version="4"> | ||
| 3 | + <component name="DesignSurface"> | ||
| 4 | + <option name="filePathToZoomLevelMap"> | ||
| 5 | + <map> | ||
| 6 | + <entry key="..\:/jbl/workspace/android/bigScreem/app/src/main/res/drawable/default_background.xml" value="0.1" /> | ||
| 7 | + <entry key="..\:/jbl/workspace/android/bigScreem/app/src/main/res/layout/activity_details.xml" value="0.24010416666666667" /> | ||
| 8 | + <entry key="..\:/jbl/workspace/android/bigScreem/app/src/main/res/layout/activity_main.xml" value="0.2546875" /> | ||
| 9 | + </map> | ||
| 10 | + </option> | ||
| 11 | + </component> | ||
| 12 | + <component name="ProjectRootManager" version="2" languageLevel="JDK_11" default="true" project-jdk-name="Android Studio default JDK" project-jdk-type="JavaSDK"> | ||
| 13 | + <output url="file://$PROJECT_DIR$/build/classes" /> | ||
| 14 | + </component> | ||
| 15 | + <component name="ProjectType"> | ||
| 16 | + <option name="id" value="Android" /> | ||
| 17 | + </component> | ||
| 18 | +</project> |
.idea/vcs.xml
0 → 100644
AppSignature.jks
0 → 100644
No preview for this file type
app/.gitignore
0 → 100644
| 1 | +/build |
app/build.gradle
0 → 100644
| 1 | +plugins { | ||
| 2 | + id 'com.android.application' | ||
| 3 | + id 'org.jetbrains.kotlin.android' | ||
| 4 | +} | ||
| 5 | + | ||
| 6 | +android { | ||
| 7 | + compileSdk 32 | ||
| 8 | + | ||
| 9 | + defaultConfig { | ||
| 10 | + applicationId "com.apaas.bigscreem" | ||
| 11 | + minSdk 28 | ||
| 12 | + targetSdk 32 | ||
| 13 | + versionCode 1 | ||
| 14 | + versionName "1.0" | ||
| 15 | + | ||
| 16 | + } | ||
| 17 | + | ||
| 18 | + signingConfigs { | ||
| 19 | + config { | ||
| 20 | + storeFile file('./AppSignature.jks') | ||
| 21 | + storePassword '123456' | ||
| 22 | + keyAlias 'study' | ||
| 23 | + keyPassword '123456' | ||
| 24 | + v1SigningEnabled true | ||
| 25 | + v2SigningEnabled true | ||
| 26 | + } | ||
| 27 | + release { | ||
| 28 | + storeFile file('./AppSignature.jks') | ||
| 29 | + storePassword '123456' | ||
| 30 | + keyAlias 'study' | ||
| 31 | + keyPassword '123456' | ||
| 32 | + v1SigningEnabled true | ||
| 33 | + v2SigningEnabled false | ||
| 34 | + } | ||
| 35 | + } | ||
| 36 | + | ||
| 37 | + buildTypes { | ||
| 38 | + release { | ||
| 39 | + debuggable false | ||
| 40 | + jniDebuggable false | ||
| 41 | + // 压缩对齐开关 | ||
| 42 | + zipAlignEnabled true | ||
| 43 | + // 签名信息配置 | ||
| 44 | + signingConfig signingConfigs.release | ||
| 45 | + // 添加清单占位符 | ||
| 46 | + addManifestPlaceholders([ | ||
| 47 | + 'app_name': '@string/app_name' | ||
| 48 | + ]) | ||
| 49 | + minifyEnabled false | ||
| 50 | + proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' | ||
| 51 | + } | ||
| 52 | + } | ||
| 53 | +} | ||
| 54 | + | ||
| 55 | +dependencies { | ||
| 56 | + | ||
| 57 | + implementation 'androidx.core:core-ktx:1.7.0' | ||
| 58 | + implementation 'androidx.leanback:leanback:1.0.0' | ||
| 59 | + implementation 'com.github.bumptech.glide:glide:4.11.0' | ||
| 60 | + implementation 'androidx.browser:browser:1.3.0' | ||
| 61 | +} |
app/proguard-rules.pro
0 → 100644
| 1 | +# Add project specific ProGuard rules here. | ||
| 2 | +# You can control the set of applied configuration files using the | ||
| 3 | +# proguardFiles setting in build.gradle. | ||
| 4 | +# | ||
| 5 | +# For more details, see | ||
| 6 | +# http://developer.android.com/guide/developing/tools/proguard.html | ||
| 7 | + | ||
| 8 | +# If your project uses WebView with JS, uncomment the following | ||
| 9 | +# and specify the fully qualified class name to the JavaScript interface | ||
| 10 | +# class: | ||
| 11 | +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { | ||
| 12 | +# public *; | ||
| 13 | +#} | ||
| 14 | + | ||
| 15 | +# Uncomment this to preserve the line number information for | ||
| 16 | +# debugging stack traces. | ||
| 17 | +#-keepattributes SourceFile,LineNumberTable | ||
| 18 | + | ||
| 19 | +# If you keep the line number information, uncomment this to | ||
| 20 | +# hide the original source file name. | ||
| 21 | +#-renamesourcefileattribute SourceFile |
app/release/app-release.apk
0 → 100644
No preview for this file type
app/release/output-metadata.json
0 → 100644
| 1 | +{ | ||
| 2 | + "version": 3, | ||
| 3 | + "artifactType": { | ||
| 4 | + "type": "APK", | ||
| 5 | + "kind": "Directory" | ||
| 6 | + }, | ||
| 7 | + "applicationId": "com.apaas.bigscreem", | ||
| 8 | + "variantName": "release", | ||
| 9 | + "elements": [ | ||
| 10 | + { | ||
| 11 | + "type": "SINGLE", | ||
| 12 | + "filters": [], | ||
| 13 | + "attributes": [], | ||
| 14 | + "versionCode": 1, | ||
| 15 | + "versionName": "1.0", | ||
| 16 | + "outputFile": "app-release.apk" | ||
| 17 | + } | ||
| 18 | + ], | ||
| 19 | + "elementType": "File" | ||
| 20 | +} |
app/src/main/AndroidManifest.xml
0 → 100644
| 1 | +<?xml version="1.0" encoding="utf-8"?> | ||
| 2 | +<manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||
| 3 | + xmlns:tools="http://schemas.android.com/tools" | ||
| 4 | + package="com.apaas.bigscreem" | ||
| 5 | + tools:ignore="MissingLeanbackLauncher"> | ||
| 6 | + | ||
| 7 | + <uses-permission android:name="android.permission.INTERNET" /> | ||
| 8 | + | ||
| 9 | + <uses-feature | ||
| 10 | + android:name="android.hardware.touchscreen" | ||
| 11 | + android:required="false" /> | ||
| 12 | + <uses-feature | ||
| 13 | + android:name="android.software.leanback" | ||
| 14 | + android:required="true" /> | ||
| 15 | + | ||
| 16 | + <application | ||
| 17 | + android:allowBackup="true" | ||
| 18 | + android:icon="@mipmap/ic_launcher" | ||
| 19 | + android:label="@string/app_name" | ||
| 20 | + android:supportsRtl="true" | ||
| 21 | + android:theme="@style/Theme.BigScreem"> | ||
| 22 | + <activity | ||
| 23 | + android:name=".MainActivity" | ||
| 24 | + android:banner="@drawable/app_icon_your_company" | ||
| 25 | + android:exported="true" | ||
| 26 | + android:icon="@drawable/app_icon_your_company" | ||
| 27 | + android:label="@string/app_name" | ||
| 28 | + android:logo="@drawable/app_icon_your_company" | ||
| 29 | + android:screenOrientation="landscape"> | ||
| 30 | + <intent-filter> | ||
| 31 | + <action android:name="android.intent.action.MAIN" /> | ||
| 32 | + | ||
| 33 | + <category android:name="android.intent.category.LAUNCHER" /> | ||
| 34 | + </intent-filter> | ||
| 35 | + </activity> | ||
| 36 | + <activity | ||
| 37 | + android:name=".DetailsActivity" | ||
| 38 | + android:screenOrientation="landscape" | ||
| 39 | + android:exported="false" /> | ||
| 40 | + <activity | ||
| 41 | + android:name=".PlaybackActivity" | ||
| 42 | + android:exported="false" /> | ||
| 43 | + <activity | ||
| 44 | + android:name=".BrowseErrorActivity" | ||
| 45 | + android:exported="false" /> | ||
| 46 | + </application> | ||
| 47 | + | ||
| 48 | +</manifest> |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.os.Bundle | ||
| 4 | +import android.os.Handler | ||
| 5 | +import android.os.Looper | ||
| 6 | +import android.view.Gravity | ||
| 7 | +import android.view.LayoutInflater | ||
| 8 | +import android.view.View | ||
| 9 | +import android.view.ViewGroup | ||
| 10 | +import android.widget.FrameLayout | ||
| 11 | +import android.widget.ProgressBar | ||
| 12 | +import androidx.fragment.app.Fragment | ||
| 13 | +import androidx.fragment.app.FragmentActivity | ||
| 14 | + | ||
| 15 | +/** | ||
| 16 | + * BrowseErrorActivity shows how to use ErrorFragment. | ||
| 17 | + */ | ||
| 18 | +class BrowseErrorActivity : FragmentActivity() { | ||
| 19 | + | ||
| 20 | + private lateinit var mErrorFragment: ErrorFragment | ||
| 21 | + private lateinit var mSpinnerFragment: SpinnerFragment | ||
| 22 | + | ||
| 23 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 24 | + super.onCreate(savedInstanceState) | ||
| 25 | + setContentView(R.layout.activity_main) | ||
| 26 | + if (savedInstanceState == null) { | ||
| 27 | + getSupportFragmentManager().beginTransaction() | ||
| 28 | + .replace(R.id.main_browse_fragment, MainFragment()) | ||
| 29 | + .commitNow() | ||
| 30 | + } | ||
| 31 | + testError() | ||
| 32 | + } | ||
| 33 | + | ||
| 34 | + private fun testError() { | ||
| 35 | + mErrorFragment = ErrorFragment() | ||
| 36 | + supportFragmentManager | ||
| 37 | + .beginTransaction() | ||
| 38 | + .add(R.id.main_browse_fragment, mErrorFragment) | ||
| 39 | + .commit() | ||
| 40 | + | ||
| 41 | + mSpinnerFragment = SpinnerFragment() | ||
| 42 | + supportFragmentManager | ||
| 43 | + .beginTransaction() | ||
| 44 | + .add(R.id.main_browse_fragment, mSpinnerFragment) | ||
| 45 | + .commit() | ||
| 46 | + | ||
| 47 | + val handler = Handler(Looper.myLooper()!!) | ||
| 48 | + handler.postDelayed({ | ||
| 49 | + supportFragmentManager | ||
| 50 | + .beginTransaction() | ||
| 51 | + .remove(mSpinnerFragment) | ||
| 52 | + .commit() | ||
| 53 | + mErrorFragment.setErrorContent() | ||
| 54 | + }, TIMER_DELAY) | ||
| 55 | + } | ||
| 56 | + | ||
| 57 | + class SpinnerFragment : Fragment() { | ||
| 58 | + override fun onCreateView( | ||
| 59 | + inflater: LayoutInflater, container: ViewGroup?, | ||
| 60 | + savedInstanceState: Bundle? | ||
| 61 | + ): View? { | ||
| 62 | + val progressBar = ProgressBar(container?.context) | ||
| 63 | + if (container is FrameLayout) { | ||
| 64 | + val layoutParams = | ||
| 65 | + FrameLayout.LayoutParams(SPINNER_WIDTH, SPINNER_HEIGHT, Gravity.CENTER) | ||
| 66 | + progressBar.layoutParams = layoutParams | ||
| 67 | + } | ||
| 68 | + return progressBar | ||
| 69 | + } | ||
| 70 | + } | ||
| 71 | + | ||
| 72 | + companion object { | ||
| 73 | + private val TIMER_DELAY = 3000L | ||
| 74 | + private val SPINNER_WIDTH = 100 | ||
| 75 | + private val SPINNER_HEIGHT = 100 | ||
| 76 | + } | ||
| 77 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.graphics.drawable.Drawable | ||
| 4 | +import androidx.leanback.widget.ImageCardView | ||
| 5 | +import androidx.leanback.widget.Presenter | ||
| 6 | +import androidx.core.content.ContextCompat | ||
| 7 | +import android.util.Log | ||
| 8 | +import android.view.ViewGroup | ||
| 9 | + | ||
| 10 | +import com.bumptech.glide.Glide | ||
| 11 | +import kotlin.properties.Delegates | ||
| 12 | + | ||
| 13 | +/** | ||
| 14 | + * A CardPresenter is used to generate Views and bind Objects to them on demand. | ||
| 15 | + * It contains an ImageCardView. | ||
| 16 | + */ | ||
| 17 | +class CardPresenter : Presenter() { | ||
| 18 | + private var mDefaultCardImage: Drawable? = null | ||
| 19 | + private var sSelectedBackgroundColor: Int by Delegates.notNull() | ||
| 20 | + private var sDefaultBackgroundColor: Int by Delegates.notNull() | ||
| 21 | + | ||
| 22 | + override fun onCreateViewHolder(parent: ViewGroup): Presenter.ViewHolder { | ||
| 23 | + Log.d(TAG, "onCreateViewHolder") | ||
| 24 | + | ||
| 25 | + sDefaultBackgroundColor = ContextCompat.getColor(parent.context, R.color.default_background) | ||
| 26 | + sSelectedBackgroundColor = | ||
| 27 | + ContextCompat.getColor(parent.context, R.color.selected_background) | ||
| 28 | + mDefaultCardImage = ContextCompat.getDrawable(parent.context, R.drawable.movie) | ||
| 29 | + | ||
| 30 | + val cardView = object : ImageCardView(parent.context) { | ||
| 31 | + override fun setSelected(selected: Boolean) { | ||
| 32 | + updateCardBackgroundColor(this, selected) | ||
| 33 | + super.setSelected(selected) | ||
| 34 | + } | ||
| 35 | + } | ||
| 36 | + | ||
| 37 | + cardView.isFocusable = true | ||
| 38 | + cardView.isFocusableInTouchMode = true | ||
| 39 | + updateCardBackgroundColor(cardView, false) | ||
| 40 | + return Presenter.ViewHolder(cardView) | ||
| 41 | + } | ||
| 42 | + | ||
| 43 | + override fun onBindViewHolder(viewHolder: Presenter.ViewHolder, item: Any) { | ||
| 44 | + val movie = item as Movie | ||
| 45 | + val cardView = viewHolder.view as ImageCardView | ||
| 46 | + | ||
| 47 | + Log.d(TAG, "onBindViewHolder") | ||
| 48 | + if (movie.cardImageUrl != null) { | ||
| 49 | + cardView.titleText = movie.title | ||
| 50 | + cardView.contentText = movie.studio | ||
| 51 | + cardView.setMainImageDimensions(CARD_WIDTH, CARD_HEIGHT) | ||
| 52 | + Glide.with(viewHolder.view.context) | ||
| 53 | + .load(movie.cardImageUrl) | ||
| 54 | + .centerCrop() | ||
| 55 | + .error(mDefaultCardImage) | ||
| 56 | + .into(cardView.mainImageView) | ||
| 57 | + } | ||
| 58 | + } | ||
| 59 | + | ||
| 60 | + override fun onUnbindViewHolder(viewHolder: Presenter.ViewHolder) { | ||
| 61 | + Log.d(TAG, "onUnbindViewHolder") | ||
| 62 | + val cardView = viewHolder.view as ImageCardView | ||
| 63 | + // Remove references to images so that the garbage collector can free up memory | ||
| 64 | + cardView.badgeImage = null | ||
| 65 | + cardView.mainImage = null | ||
| 66 | + } | ||
| 67 | + | ||
| 68 | + private fun updateCardBackgroundColor(view: ImageCardView, selected: Boolean) { | ||
| 69 | + val color = if (selected) sSelectedBackgroundColor else sDefaultBackgroundColor | ||
| 70 | + // Both background colors should be set because the view"s background is temporarily visible | ||
| 71 | + // during animations. | ||
| 72 | + view.setBackgroundColor(color) | ||
| 73 | + view.setInfoAreaBackgroundColor(color) | ||
| 74 | + } | ||
| 75 | + | ||
| 76 | + companion object { | ||
| 77 | + private val TAG = "CardPresenter" | ||
| 78 | + | ||
| 79 | + private val CARD_WIDTH = 313 | ||
| 80 | + private val CARD_HEIGHT = 176 | ||
| 81 | + } | ||
| 82 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.os.Build | ||
| 4 | +import android.os.Bundle | ||
| 5 | +import android.os.Handler | ||
| 6 | +import android.view.View | ||
| 7 | +import android.view.WindowInsets | ||
| 8 | +import android.view.WindowInsetsController | ||
| 9 | +import android.webkit.WebView | ||
| 10 | +import android.webkit.WebViewClient | ||
| 11 | +import androidx.fragment.app.FragmentActivity | ||
| 12 | + | ||
| 13 | + | ||
| 14 | +/** | ||
| 15 | + * Details activity class that loads [VideoDetailsFragment] class. | ||
| 16 | + */ | ||
| 17 | +class DetailsActivity : FragmentActivity() { | ||
| 18 | + | ||
| 19 | + private var mSelectedMovie: Movie? = null | ||
| 20 | + private val handler = Handler() | ||
| 21 | + private lateinit var mWebView: WebView | ||
| 22 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 23 | + super.onCreate(savedInstanceState) | ||
| 24 | + setContentView(R.layout.activity_details) | ||
| 25 | + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { | ||
| 26 | + window.setDecorFitsSystemWindows(false) | ||
| 27 | + window.insetsController?.let { | ||
| 28 | + it.hide(WindowInsets.Type.statusBars() or WindowInsets.Type.navigationBars()) | ||
| 29 | + it.systemBarsBehavior = WindowInsetsController.BEHAVIOR_SHOW_TRANSIENT_BARS_BY_SWIPE | ||
| 30 | + } | ||
| 31 | + } else { | ||
| 32 | + @Suppress("DEPRECATION") | ||
| 33 | + window.decorView.systemUiVisibility = (View.SYSTEM_UI_FLAG_FULLSCREEN | ||
| 34 | + or View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | ||
| 35 | + or View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY | ||
| 36 | + or View.SYSTEM_UI_FLAG_LAYOUT_STABLE | ||
| 37 | + or View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | ||
| 38 | + or View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION) | ||
| 39 | + } | ||
| 40 | + | ||
| 41 | + mSelectedMovie = intent.getSerializableExtra(MOVIE) as Movie | ||
| 42 | + if (mSelectedMovie != null) { | ||
| 43 | + mWebView = findViewById(R.id.webView) | ||
| 44 | + // 启用JavaScript | ||
| 45 | + | ||
| 46 | + mWebView.settings.javaScriptEnabled = true // 启用JavaScript | ||
| 47 | + | ||
| 48 | + mWebView.settings.setUserAgentString("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/51.0.2704.79 Safari/537.36") // 设置用户代理为桌面浏览器的样子 | ||
| 49 | + | ||
| 50 | + mWebView.webViewClient = WebViewClient() // 处理点击链接的方式 | ||
| 51 | + | ||
| 52 | + // 设置WebChromeClient来处理JavaScript的对话框、网页标题等 | ||
| 53 | + | ||
| 54 | + mSelectedMovie!!.webUrl?.let { mWebView.loadUrl(it) } | ||
| 55 | + } | ||
| 56 | + | ||
| 57 | + val refreshInterval = 43200000L | ||
| 58 | +// val refreshInterval = 5000L | ||
| 59 | + // 使用Handler定时刷新WebView | ||
| 60 | + handler.postDelayed(object : Runnable { | ||
| 61 | + override fun run() { | ||
| 62 | + mWebView.reload() | ||
| 63 | + // 递归调用自身实现定时刷新 | ||
| 64 | + handler.postDelayed(this, refreshInterval) | ||
| 65 | + } | ||
| 66 | + }, refreshInterval) | ||
| 67 | + } | ||
| 68 | + | ||
| 69 | + companion object { | ||
| 70 | + const val SHARED_ELEMENT_NAME = "hero" | ||
| 71 | + const val MOVIE = "Movie" | ||
| 72 | + } | ||
| 73 | + | ||
| 74 | + override fun onDestroy() { | ||
| 75 | + super.onDestroy() | ||
| 76 | + // 清理资源,移除所有回调 | ||
| 77 | + handler.removeCallbacksAndMessages(null) | ||
| 78 | + } | ||
| 79 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import androidx.leanback.widget.AbstractDetailsDescriptionPresenter | ||
| 4 | + | ||
| 5 | +class DetailsDescriptionPresenter : AbstractDetailsDescriptionPresenter() { | ||
| 6 | + | ||
| 7 | + override fun onBindDescription( | ||
| 8 | + viewHolder: AbstractDetailsDescriptionPresenter.ViewHolder, | ||
| 9 | + item: Any | ||
| 10 | + ) { | ||
| 11 | + val movie = item as Movie | ||
| 12 | + | ||
| 13 | + viewHolder.title.text = movie.title | ||
| 14 | + viewHolder.subtitle.text = movie.studio | ||
| 15 | + viewHolder.body.text = movie.description | ||
| 16 | + } | ||
| 17 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.os.Bundle | ||
| 4 | +import android.view.View | ||
| 5 | + | ||
| 6 | +import androidx.core.content.ContextCompat | ||
| 7 | +import androidx.leanback.app.ErrorSupportFragment | ||
| 8 | + | ||
| 9 | +/** | ||
| 10 | + * This class demonstrates how to extend [ErrorSupportFragment]. | ||
| 11 | + */ | ||
| 12 | +class ErrorFragment : ErrorSupportFragment() { | ||
| 13 | + | ||
| 14 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 15 | + super.onCreate(savedInstanceState) | ||
| 16 | + title = resources.getString(R.string.app_name) | ||
| 17 | + } | ||
| 18 | + | ||
| 19 | + internal fun setErrorContent() { | ||
| 20 | + imageDrawable = | ||
| 21 | + ContextCompat.getDrawable(context!!, androidx.leanback.R.drawable.lb_ic_sad_cloud) | ||
| 22 | + message = resources.getString(R.string.error_fragment_message) | ||
| 23 | + setDefaultBackground(TRANSLUCENT) | ||
| 24 | + | ||
| 25 | + buttonText = resources.getString(R.string.dismiss_error) | ||
| 26 | + buttonClickListener = View.OnClickListener { | ||
| 27 | + fragmentManager!!.beginTransaction().remove(this@ErrorFragment).commit() | ||
| 28 | + } | ||
| 29 | + } | ||
| 30 | + | ||
| 31 | + companion object { | ||
| 32 | + private val TRANSLUCENT = true | ||
| 33 | + } | ||
| 34 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.os.Bundle | ||
| 4 | +import androidx.fragment.app.FragmentActivity | ||
| 5 | + | ||
| 6 | +/** | ||
| 7 | + * Loads [MainFragment]. | ||
| 8 | + */ | ||
| 9 | +class MainActivity : FragmentActivity() { | ||
| 10 | + | ||
| 11 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 12 | + super.onCreate(savedInstanceState) | ||
| 13 | + setContentView(R.layout.activity_main) | ||
| 14 | + if (savedInstanceState == null) { | ||
| 15 | + getSupportFragmentManager().beginTransaction() | ||
| 16 | + .replace(R.id.main_browse_fragment, MainFragment()) | ||
| 17 | + .commitNow() | ||
| 18 | + } | ||
| 19 | + } | ||
| 20 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import java.util.Collections | ||
| 4 | +import java.util.Timer | ||
| 5 | +import java.util.TimerTask | ||
| 6 | + | ||
| 7 | +import android.content.Intent | ||
| 8 | +import android.graphics.Color | ||
| 9 | +import android.graphics.drawable.Drawable | ||
| 10 | +import android.os.Bundle | ||
| 11 | +import android.os.Handler | ||
| 12 | +import android.os.Looper | ||
| 13 | +import androidx.leanback.app.BackgroundManager | ||
| 14 | +import androidx.leanback.app.BrowseSupportFragment | ||
| 15 | +import androidx.leanback.widget.ArrayObjectAdapter | ||
| 16 | +import androidx.leanback.widget.HeaderItem | ||
| 17 | +import androidx.leanback.widget.ImageCardView | ||
| 18 | +import androidx.leanback.widget.ListRow | ||
| 19 | +import androidx.leanback.widget.ListRowPresenter | ||
| 20 | +import androidx.leanback.widget.OnItemViewClickedListener | ||
| 21 | +import androidx.leanback.widget.OnItemViewSelectedListener | ||
| 22 | +import androidx.leanback.widget.Presenter | ||
| 23 | +import androidx.leanback.widget.Row | ||
| 24 | +import androidx.leanback.widget.RowPresenter | ||
| 25 | +import androidx.core.app.ActivityOptionsCompat | ||
| 26 | +import androidx.core.content.ContextCompat | ||
| 27 | +import android.util.DisplayMetrics | ||
| 28 | +import android.util.Log | ||
| 29 | +import android.view.Gravity | ||
| 30 | +import android.view.ViewGroup | ||
| 31 | +import android.widget.TextView | ||
| 32 | +import android.widget.Toast | ||
| 33 | + | ||
| 34 | +import com.bumptech.glide.Glide | ||
| 35 | +import com.bumptech.glide.request.target.SimpleTarget | ||
| 36 | +import com.bumptech.glide.request.transition.Transition | ||
| 37 | + | ||
| 38 | +/** | ||
| 39 | + * Loads a grid of cards with movies to browse. | ||
| 40 | + */ | ||
| 41 | +class MainFragment : BrowseSupportFragment() { | ||
| 42 | + | ||
| 43 | + private val mHandler = Handler(Looper.myLooper()!!) | ||
| 44 | + private lateinit var mBackgroundManager: BackgroundManager | ||
| 45 | + private var mDefaultBackground: Drawable? = null | ||
| 46 | + private lateinit var mMetrics: DisplayMetrics | ||
| 47 | + private var mBackgroundTimer: Timer? = null | ||
| 48 | + private var mBackgroundUri: String? = null | ||
| 49 | + | ||
| 50 | + override fun onActivityCreated(savedInstanceState: Bundle?) { | ||
| 51 | + Log.i(TAG, "onCreate") | ||
| 52 | + super.onActivityCreated(savedInstanceState) | ||
| 53 | + | ||
| 54 | + prepareBackgroundManager() | ||
| 55 | + | ||
| 56 | + setupUIElements() | ||
| 57 | + | ||
| 58 | + loadRows() | ||
| 59 | + | ||
| 60 | + setupEventListeners() | ||
| 61 | + } | ||
| 62 | + | ||
| 63 | + override fun onDestroy() { | ||
| 64 | + super.onDestroy() | ||
| 65 | + Log.d(TAG, "onDestroy: " + mBackgroundTimer?.toString()) | ||
| 66 | + mBackgroundTimer?.cancel() | ||
| 67 | + } | ||
| 68 | + | ||
| 69 | + private fun prepareBackgroundManager() { | ||
| 70 | + | ||
| 71 | + mBackgroundManager = BackgroundManager.getInstance(activity) | ||
| 72 | + mBackgroundManager.attach(activity!!.window) | ||
| 73 | + mDefaultBackground = ContextCompat.getDrawable(context!!, R.drawable.default_background) | ||
| 74 | + mMetrics = DisplayMetrics() | ||
| 75 | + activity!!.windowManager.defaultDisplay.getMetrics(mMetrics) | ||
| 76 | + } | ||
| 77 | + | ||
| 78 | + private fun setupUIElements() { | ||
| 79 | + title = getString(R.string.browse_title) | ||
| 80 | + // over title | ||
| 81 | + headersState = BrowseSupportFragment.HEADERS_ENABLED | ||
| 82 | + isHeadersTransitionOnBackEnabled = true | ||
| 83 | + | ||
| 84 | + // set fastLane (or headers) background color | ||
| 85 | + brandColor = ContextCompat.getColor(context!!, R.color.fastlane_background) | ||
| 86 | + // set search icon color | ||
| 87 | + searchAffordanceColor = ContextCompat.getColor(context!!, R.color.search_opaque) | ||
| 88 | + } | ||
| 89 | + | ||
| 90 | + private fun loadRows() { | ||
| 91 | + val list = MovieList.list | ||
| 92 | + | ||
| 93 | + val rowsAdapter = ArrayObjectAdapter(ListRowPresenter()) | ||
| 94 | + val cardPresenter = CardPresenter() | ||
| 95 | + | ||
| 96 | + for (i in 0 until NUM_ROWS) { | ||
| 97 | + if (i != 0) { | ||
| 98 | + Collections.shuffle(list) | ||
| 99 | + } | ||
| 100 | + val listRowAdapter = ArrayObjectAdapter(cardPresenter) | ||
| 101 | + for (j in 0 until NUM_COLS) { | ||
| 102 | + listRowAdapter.add(list[j]) | ||
| 103 | + } | ||
| 104 | + val header = HeaderItem(i.toLong(), MovieList.MOVIE_CATEGORY[i]) | ||
| 105 | + rowsAdapter.add(ListRow(header, listRowAdapter)) | ||
| 106 | + } | ||
| 107 | + | ||
| 108 | + val gridHeader = HeaderItem(NUM_ROWS.toLong(), "其他") | ||
| 109 | + | ||
| 110 | + val mGridPresenter = GridItemPresenter() | ||
| 111 | + val gridRowAdapter = ArrayObjectAdapter(mGridPresenter) | ||
| 112 | + gridRowAdapter.add(resources.getString(R.string.grid_view)) | ||
| 113 | + gridRowAdapter.add(getString(R.string.error_fragment)) | ||
| 114 | + gridRowAdapter.add(resources.getString(R.string.personal_settings)) | ||
| 115 | + rowsAdapter.add(ListRow(gridHeader, gridRowAdapter)) | ||
| 116 | + | ||
| 117 | + adapter = rowsAdapter | ||
| 118 | + } | ||
| 119 | + | ||
| 120 | + private fun setupEventListeners() { | ||
| 121 | + setOnSearchClickedListener { | ||
| 122 | + Toast.makeText(context!!, "Implement your own in-app search", Toast.LENGTH_LONG) | ||
| 123 | + .show() | ||
| 124 | + } | ||
| 125 | + | ||
| 126 | + onItemViewClickedListener = ItemViewClickedListener() | ||
| 127 | + onItemViewSelectedListener = ItemViewSelectedListener() | ||
| 128 | + } | ||
| 129 | + | ||
| 130 | + private inner class ItemViewClickedListener : OnItemViewClickedListener { | ||
| 131 | + override fun onItemClicked( | ||
| 132 | + itemViewHolder: Presenter.ViewHolder, | ||
| 133 | + item: Any, | ||
| 134 | + rowViewHolder: RowPresenter.ViewHolder, | ||
| 135 | + row: Row | ||
| 136 | + ) { | ||
| 137 | + | ||
| 138 | + if (item is Movie) { | ||
| 139 | + Log.d(TAG, "Item: " + item.toString()) | ||
| 140 | + val intent = Intent(context!!, DetailsActivity::class.java) | ||
| 141 | + intent.putExtra(DetailsActivity.MOVIE, item) | ||
| 142 | + | ||
| 143 | + val bundle = ActivityOptionsCompat.makeSceneTransitionAnimation( | ||
| 144 | + activity!!, | ||
| 145 | + (itemViewHolder.view as ImageCardView).mainImageView, | ||
| 146 | + DetailsActivity.SHARED_ELEMENT_NAME | ||
| 147 | + ) | ||
| 148 | + .toBundle() | ||
| 149 | + startActivity(intent, bundle) | ||
| 150 | + } else if (item is String) { | ||
| 151 | + if (item.contains(getString(R.string.error_fragment))) { | ||
| 152 | + val intent = Intent(context!!, BrowseErrorActivity::class.java) | ||
| 153 | + startActivity(intent) | ||
| 154 | + } else { | ||
| 155 | + Toast.makeText(context!!, item, Toast.LENGTH_SHORT).show() | ||
| 156 | + } | ||
| 157 | + } | ||
| 158 | + } | ||
| 159 | + } | ||
| 160 | + | ||
| 161 | + private inner class ItemViewSelectedListener : OnItemViewSelectedListener { | ||
| 162 | + override fun onItemSelected( | ||
| 163 | + itemViewHolder: Presenter.ViewHolder?, item: Any?, | ||
| 164 | + rowViewHolder: RowPresenter.ViewHolder, row: Row | ||
| 165 | + ) { | ||
| 166 | + if (item is Movie) { | ||
| 167 | + mBackgroundUri = item.backgroundImageUrl | ||
| 168 | + startBackgroundTimer() | ||
| 169 | + } | ||
| 170 | + } | ||
| 171 | + } | ||
| 172 | + | ||
| 173 | + private fun updateBackground(uri: String?) { | ||
| 174 | + val width = mMetrics.widthPixels | ||
| 175 | + val height = mMetrics.heightPixels | ||
| 176 | + Glide.with(context!!) | ||
| 177 | + .load(uri) | ||
| 178 | + .centerCrop() | ||
| 179 | + .error(mDefaultBackground) | ||
| 180 | + .into<SimpleTarget<Drawable>>( | ||
| 181 | + object : SimpleTarget<Drawable>(width, height) { | ||
| 182 | + override fun onResourceReady( | ||
| 183 | + drawable: Drawable, | ||
| 184 | + transition: Transition<in Drawable>? | ||
| 185 | + ) { | ||
| 186 | + mBackgroundManager.drawable = drawable | ||
| 187 | + } | ||
| 188 | + }) | ||
| 189 | + mBackgroundTimer?.cancel() | ||
| 190 | + } | ||
| 191 | + | ||
| 192 | + private fun startBackgroundTimer() { | ||
| 193 | + mBackgroundTimer?.cancel() | ||
| 194 | + mBackgroundTimer = Timer() | ||
| 195 | + mBackgroundTimer?.schedule(UpdateBackgroundTask(), BACKGROUND_UPDATE_DELAY.toLong()) | ||
| 196 | + } | ||
| 197 | + | ||
| 198 | + private inner class UpdateBackgroundTask : TimerTask() { | ||
| 199 | + | ||
| 200 | + override fun run() { | ||
| 201 | + mHandler.post { updateBackground(mBackgroundUri) } | ||
| 202 | + } | ||
| 203 | + } | ||
| 204 | + | ||
| 205 | + private inner class GridItemPresenter : Presenter() { | ||
| 206 | + override fun onCreateViewHolder(parent: ViewGroup): Presenter.ViewHolder { | ||
| 207 | + val view = TextView(parent.context) | ||
| 208 | + view.layoutParams = ViewGroup.LayoutParams(GRID_ITEM_WIDTH, GRID_ITEM_HEIGHT) | ||
| 209 | + view.isFocusable = true | ||
| 210 | + view.isFocusableInTouchMode = true | ||
| 211 | + view.setBackgroundColor(ContextCompat.getColor(context!!, R.color.default_background)) | ||
| 212 | + view.setTextColor(Color.WHITE) | ||
| 213 | + view.gravity = Gravity.CENTER | ||
| 214 | + return Presenter.ViewHolder(view) | ||
| 215 | + } | ||
| 216 | + | ||
| 217 | + override fun onBindViewHolder(viewHolder: Presenter.ViewHolder, item: Any) { | ||
| 218 | + (viewHolder.view as TextView).text = item as String | ||
| 219 | + } | ||
| 220 | + | ||
| 221 | + override fun onUnbindViewHolder(viewHolder: Presenter.ViewHolder) {} | ||
| 222 | + } | ||
| 223 | + | ||
| 224 | + companion object { | ||
| 225 | + private val TAG = "MainFragment" | ||
| 226 | + | ||
| 227 | + private val BACKGROUND_UPDATE_DELAY = 300 | ||
| 228 | + private val GRID_ITEM_WIDTH = 200 | ||
| 229 | + private val GRID_ITEM_HEIGHT = 200 | ||
| 230 | + private val NUM_ROWS = 1 | ||
| 231 | + private val NUM_COLS = 11 | ||
| 232 | + } | ||
| 233 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import java.io.Serializable | ||
| 4 | + | ||
| 5 | +/** | ||
| 6 | + * Movie class represents video entity with title, description, image thumbs and video url. | ||
| 7 | + */ | ||
| 8 | +data class Movie( | ||
| 9 | + var id: Long = 0, | ||
| 10 | + var title: String? = null, | ||
| 11 | + var description: String? = null, | ||
| 12 | + var backgroundImageUrl: String? = null, | ||
| 13 | + var cardImageUrl: String? = null, | ||
| 14 | + var videoUrl: String? = null, | ||
| 15 | + var studio: String? = null, | ||
| 16 | + var webUrl: String? = null | ||
| 17 | +) : Serializable { | ||
| 18 | + | ||
| 19 | + override fun toString(): String { | ||
| 20 | + return "Movie{" + | ||
| 21 | + "id=" + id + | ||
| 22 | + ", title='" + title + '\'' + | ||
| 23 | + ", webUrl='" + webUrl + '\'' + | ||
| 24 | + ", backgroundImageUrl='" + backgroundImageUrl + '\'' + | ||
| 25 | + ", cardImageUrl='" + cardImageUrl + '\'' + | ||
| 26 | + '}' | ||
| 27 | + } | ||
| 28 | + | ||
| 29 | + companion object { | ||
| 30 | + internal const val serialVersionUID = 727566175075960653L | ||
| 31 | + } | ||
| 32 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +object MovieList { | ||
| 4 | + val MOVIE_CATEGORY = arrayOf( | ||
| 5 | + "华贸" | ||
| 6 | + ) | ||
| 7 | + | ||
| 8 | + val list: List<Movie> by lazy { | ||
| 9 | + setupMovies() | ||
| 10 | + } | ||
| 11 | + private var count: Long = 0 | ||
| 12 | + | ||
| 13 | + private fun setupMovies(): List<Movie> { | ||
| 14 | + val title = arrayOf( | ||
| 15 | + "党建看板", | ||
| 16 | + "公司介绍", | ||
| 17 | + "人员看板", | ||
| 18 | + "生产入库看板", | ||
| 19 | + "设备状态", | ||
| 20 | + "在制品看板", | ||
| 21 | + "生产进度看板", | ||
| 22 | + "质量看板", | ||
| 23 | + "生产计划看板", | ||
| 24 | + "订单看板", | ||
| 25 | + "综合看板" | ||
| 26 | + ) | ||
| 27 | + | ||
| 28 | + val description = "" | ||
| 29 | + val studio = arrayOf( | ||
| 30 | + "党建看板", | ||
| 31 | + "公司介绍", | ||
| 32 | + "人员看板", | ||
| 33 | + "生产入库看板", | ||
| 34 | + "设备状态", | ||
| 35 | + "在制品看板", | ||
| 36 | + "生产进度看板", | ||
| 37 | + "质量看板", | ||
| 38 | + "生产计划看板", | ||
| 39 | + "订单看板", | ||
| 40 | + "综合看板" | ||
| 41 | + ) | ||
| 42 | + | ||
| 43 | + val webUrl = arrayOf( | ||
| 44 | + "https://mi.qgutech.com/sc/hx-hm-zhdj.html", | ||
| 45 | + "https://mi.qgutech.com/sc/lx-hm-qyxc.html", | ||
| 46 | + "https://mi.qgutech.com/sc/lx-hm-ryyxqk.html", | ||
| 47 | + "https://mi.qgutech.com/sc/hx-hm-kcsjfx.html", | ||
| 48 | + "https://mi.qgutech.com/sc/hx-hm-sbgkzx.html", | ||
| 49 | + "https://mi.qgutech.com/sc/hx-hm-zjpkb.html", | ||
| 50 | + "https://mi.qgutech.com/sc/hx-hm-scjdkb.html", | ||
| 51 | + "https://mi.qgutech.com/sc/hx-hm-zlgkkb.html", | ||
| 52 | + "https://mi.qgutech.com/sc/hx-hm-scjhkb.html", | ||
| 53 | + "https://mi.qgutech.com/sc/hx-hm-xsjdkb.html", | ||
| 54 | + "https://mi.qgutech.com/sc/hx-hm-kcyjkb.html", | ||
| 55 | + ) | ||
| 56 | + val bgImageUrl = arrayOf( | ||
| 57 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/Zeitgeist/Zeitgeist%202010_%20Year%20in%20Review/bg.jpg", | ||
| 58 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/Demo%20Slam/Google%20Demo%20Slam_%2020ft%20Search/bg.jpg", | ||
| 59 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Gmail%20Blue/bg.jpg", | ||
| 60 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Fiber%20to%20the%20Pole/bg.jpg", | ||
| 61 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 62 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 63 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 64 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 65 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 66 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 67 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg" | ||
| 68 | + ) | ||
| 69 | + val cardImageUrl = arrayOf( | ||
| 70 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/Zeitgeist/Zeitgeist%202010_%20Year%20in%20Review/card.jpg", | ||
| 71 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/Demo%20Slam/Google%20Demo%20Slam_%2020ft%20Search/card.jpg", | ||
| 72 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Gmail%20Blue/card.jpg", | ||
| 73 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Fiber%20to%20the%20Pole/card.jpg", | ||
| 74 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/card.jpg", | ||
| 75 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 76 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 77 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 78 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 79 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg", | ||
| 80 | + "https://commondatastorage.googleapis.com/android-tv/Sample%20videos/April%20Fool's%202013/Introducing%20Google%20Nose/bg.jpg" | ||
| 81 | + ) | ||
| 82 | + | ||
| 83 | + val list = title.indices.map { | ||
| 84 | + buildMovieInfo( | ||
| 85 | + title[it], | ||
| 86 | + description, | ||
| 87 | + studio[it], | ||
| 88 | + cardImageUrl[it], | ||
| 89 | + bgImageUrl[it], | ||
| 90 | + webUrl[it], | ||
| 91 | + ) | ||
| 92 | + } | ||
| 93 | + | ||
| 94 | + return list | ||
| 95 | + } | ||
| 96 | + | ||
| 97 | + private fun buildMovieInfo( | ||
| 98 | + title: String, | ||
| 99 | + description: String, | ||
| 100 | + studio: String, | ||
| 101 | + cardImageUrl: String, | ||
| 102 | + backgroundImageUrl: String, | ||
| 103 | + webUrl: String | ||
| 104 | + ): Movie { | ||
| 105 | + val movie = Movie() | ||
| 106 | + movie.id = count++ | ||
| 107 | + movie.title = title | ||
| 108 | + movie.description = description | ||
| 109 | + movie.studio = studio | ||
| 110 | + movie.cardImageUrl = cardImageUrl | ||
| 111 | + movie.backgroundImageUrl = backgroundImageUrl | ||
| 112 | + movie.webUrl = webUrl | ||
| 113 | + return movie | ||
| 114 | + } | ||
| 115 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.os.Bundle | ||
| 4 | +import androidx.fragment.app.FragmentActivity | ||
| 5 | + | ||
| 6 | +/** Loads [PlaybackVideoFragment]. */ | ||
| 7 | +class PlaybackActivity : FragmentActivity() { | ||
| 8 | + | ||
| 9 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 10 | + super.onCreate(savedInstanceState) | ||
| 11 | + if (savedInstanceState == null) { | ||
| 12 | + supportFragmentManager.beginTransaction() | ||
| 13 | + .replace(android.R.id.content, PlaybackVideoFragment()) | ||
| 14 | + .commit() | ||
| 15 | + } | ||
| 16 | + } | ||
| 17 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.net.Uri | ||
| 4 | +import android.os.Bundle | ||
| 5 | +import androidx.leanback.app.VideoSupportFragment | ||
| 6 | +import androidx.leanback.app.VideoSupportFragmentGlueHost | ||
| 7 | +import androidx.leanback.media.MediaPlayerAdapter | ||
| 8 | +import androidx.leanback.media.PlaybackTransportControlGlue | ||
| 9 | +import androidx.leanback.widget.PlaybackControlsRow | ||
| 10 | + | ||
| 11 | +/** Handles video playback with media controls. */ | ||
| 12 | +class PlaybackVideoFragment : VideoSupportFragment() { | ||
| 13 | + | ||
| 14 | + private lateinit var mTransportControlGlue: PlaybackTransportControlGlue<MediaPlayerAdapter> | ||
| 15 | + | ||
| 16 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 17 | + super.onCreate(savedInstanceState) | ||
| 18 | + | ||
| 19 | + val (_, title, description, _, _, videoUrl) = | ||
| 20 | + activity?.intent?.getSerializableExtra(DetailsActivity.MOVIE) as Movie | ||
| 21 | + | ||
| 22 | + val glueHost = VideoSupportFragmentGlueHost(this@PlaybackVideoFragment) | ||
| 23 | + val playerAdapter = MediaPlayerAdapter(context) | ||
| 24 | + playerAdapter.setRepeatAction(PlaybackControlsRow.RepeatAction.INDEX_NONE) | ||
| 25 | + | ||
| 26 | + mTransportControlGlue = PlaybackTransportControlGlue(getActivity(), playerAdapter) | ||
| 27 | + mTransportControlGlue.host = glueHost | ||
| 28 | + mTransportControlGlue.title = title | ||
| 29 | + mTransportControlGlue.subtitle = description | ||
| 30 | + mTransportControlGlue.playWhenPrepared() | ||
| 31 | + | ||
| 32 | + playerAdapter.setDataSource(Uri.parse(videoUrl)) | ||
| 33 | + } | ||
| 34 | + | ||
| 35 | + override fun onPause() { | ||
| 36 | + super.onPause() | ||
| 37 | + mTransportControlGlue.pause() | ||
| 38 | + } | ||
| 39 | +} |
| 1 | +package com.apaas.bigscreem | ||
| 2 | + | ||
| 3 | +import android.content.Intent | ||
| 4 | +import android.os.Bundle | ||
| 5 | +import android.net.Uri | ||
| 6 | +import androidx.leanback.app.DetailsSupportFragment | ||
| 7 | +import androidx.leanback.widget.RowPresenter | ||
| 8 | +import androidx.core.app.ActivityOptionsCompat | ||
| 9 | +import androidx.core.content.ContextCompat | ||
| 10 | +import android.util.Log | ||
| 11 | +import android.webkit.WebView | ||
| 12 | +import android.widget.Toast | ||
| 13 | +import androidx.browser.customtabs.CustomTabsIntent | ||
| 14 | + | ||
| 15 | +import com.bumptech.glide.Glide | ||
| 16 | +import com.bumptech.glide.request.target.SimpleTarget | ||
| 17 | +import com.bumptech.glide.request.transition.Transition | ||
| 18 | + | ||
| 19 | +import java.util.Collections | ||
| 20 | + | ||
| 21 | +/** | ||
| 22 | + * A wrapper fragment for leanback details screens. | ||
| 23 | + * It shows a detailed view of video and its metadata plus related videos. | ||
| 24 | + */ | ||
| 25 | +class VideoDetailsFragment : DetailsSupportFragment() { | ||
| 26 | + | ||
| 27 | + private var mSelectedMovie: Movie? = null | ||
| 28 | + private lateinit var mWebView: WebView | ||
| 29 | + | ||
| 30 | + override fun onCreate(savedInstanceState: Bundle?) { | ||
| 31 | + super.onCreate(savedInstanceState) | ||
| 32 | + mSelectedMovie = activity!!.intent.getSerializableExtra(DetailsActivity.MOVIE) as Movie | ||
| 33 | + if (mSelectedMovie != null) { | ||
| 34 | + mSelectedMovie!!.webUrl?.let { openWebPage(it) }; | ||
| 35 | + } else { | ||
| 36 | + val intent = Intent(context!!, MainActivity::class.java) | ||
| 37 | + startActivity(intent) | ||
| 38 | + } | ||
| 39 | + } | ||
| 40 | + | ||
| 41 | + fun openWebPage(url: String) { | ||
| 42 | + val customTabsIntent = CustomTabsIntent.Builder().build() | ||
| 43 | + activity?.let { customTabsIntent.launchUrl(it, Uri.parse(url)) } | ||
| 44 | + } | ||
| 45 | +} |
12.1 KB
| 1 | +<?xml version="1.0" encoding="utf-8"?> | ||
| 2 | + | ||
| 3 | +<shape xmlns:android="http://schemas.android.com/apk/res/android" | ||
| 4 | + android:shape="rectangle"> | ||
| 5 | + <gradient | ||
| 6 | + android:angle="-270" | ||
| 7 | + android:endColor="@color/background_gradient_end" | ||
| 8 | + android:startColor="@color/background_gradient_start" /> | ||
| 9 | +</shape> |
app/src/main/res/drawable/movie.png
0 → 100644
12.7 KB
app/src/main/res/layout/activity_details.xml
0 → 100644
| 1 | +<?xml version="1.0" encoding="utf-8"?> | ||
| 2 | +<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||
| 3 | + xmlns:tools="http://schemas.android.com/tools" | ||
| 4 | + android:id="@+id/details_fragment" | ||
| 5 | + android:layout_width="match_parent" | ||
| 6 | + android:layout_height="match_parent" | ||
| 7 | + tools:context=".DetailsActivity" | ||
| 8 | + tools:deviceIds="tv" > | ||
| 9 | + | ||
| 10 | + <WebView | ||
| 11 | + android:id="@+id/webView" | ||
| 12 | + android:layout_width="match_parent" | ||
| 13 | + android:layout_height="match_parent" /> | ||
| 14 | +</FrameLayout> |
app/src/main/res/layout/activity_main.xml
0 → 100644
| 1 | +<?xml version="1.0" encoding="utf-8"?> | ||
| 2 | +<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||
| 3 | + xmlns:tools="http://schemas.android.com/tools" | ||
| 4 | + android:id="@+id/main_browse_fragment" | ||
| 5 | + android:layout_width="match_parent" | ||
| 6 | + android:layout_height="match_parent" | ||
| 7 | + tools:context=".MainActivity" | ||
| 8 | + tools:deviceIds="tv" | ||
| 9 | + tools:ignore="MergeRootFrame" > | ||
| 10 | +</FrameLayout> |
No preview for this file type
No preview for this file type
No preview for this file type
No preview for this file type
No preview for this file type
app/src/main/res/values/colors.xml
0 → 100644
| 1 | +<resources> | ||
| 2 | + <color name="background_gradient_start">#000000</color> | ||
| 3 | + <color name="background_gradient_end">#DDDDDD</color> | ||
| 4 | + <color name="fastlane_background">#0096a6</color> | ||
| 5 | + <color name="search_opaque">#ffaa3f</color> | ||
| 6 | + <color name="selected_background">#ffaa3f</color> | ||
| 7 | + <color name="default_background">#3d3d3d</color> | ||
| 8 | +</resources> |
app/src/main/res/values/strings.xml
0 → 100644
| 1 | +<resources> | ||
| 2 | + <string name="app_name">bigScreem</string> | ||
| 3 | + <string name="browse_title">华贸大屏展示</string> | ||
| 4 | + <string name="related_movies">Related Videos</string> | ||
| 5 | + <string name="grid_view">返回</string> | ||
| 6 | + <string name="error_fragment">错误</string> | ||
| 7 | + <string name="personal_settings">设置中心</string> | ||
| 8 | + <string name="watch_trailer_1">Watch trailer</string> | ||
| 9 | + <string name="watch_trailer_2">FREE</string> | ||
| 10 | + <string name="rent_1">Rent By Day</string> | ||
| 11 | + <string name="rent_2">From $1.99</string> | ||
| 12 | + <string name="buy_1">Buy and Own</string> | ||
| 13 | + <string name="buy_2">AT $9.99</string> | ||
| 14 | + <string name="movie">Movie</string> | ||
| 15 | + | ||
| 16 | + <!-- Error messages --> | ||
| 17 | + <string name="error_fragment_message">An error occurred</string> | ||
| 18 | + <string name="dismiss_error">Dismiss</string> | ||
| 19 | +</resources> |
app/src/main/res/values/themes.xml
0 → 100644
build.gradle
0 → 100644
| 1 | +// Top-level build file where you can add configuration options common to all sub-projects/modules. | ||
| 2 | +plugins { | ||
| 3 | + id 'com.android.application' version '7.2.1' apply false | ||
| 4 | + id 'com.android.library' version '7.2.1' apply false | ||
| 5 | + id 'org.jetbrains.kotlin.android' version '1.6.10' apply false | ||
| 6 | +} | ||
| 7 | + | ||
| 8 | +task clean(type: Delete) { | ||
| 9 | + delete rootProject.buildDir | ||
| 10 | +} |
gradle.properties
0 → 100644
| 1 | +# Project-wide Gradle settings. | ||
| 2 | +# IDE (e.g. Android Studio) users: | ||
| 3 | +# Gradle settings configured through the IDE *will override* | ||
| 4 | +# any settings specified in this file. | ||
| 5 | +# For more details on how to configure your build environment visit | ||
| 6 | +# http://www.gradle.org/docs/current/userguide/build_environment.html | ||
| 7 | +# Specifies the JVM arguments used for the daemon process. | ||
| 8 | +# The setting is particularly useful for tweaking memory settings. | ||
| 9 | +org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8 | ||
| 10 | +# When configured, Gradle will run in incubating parallel mode. | ||
| 11 | +# This option should only be used with decoupled projects. More details, visit | ||
| 12 | +# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects | ||
| 13 | +# org.gradle.parallel=true | ||
| 14 | +# AndroidX package structure to make it clearer which packages are bundled with the | ||
| 15 | +# Android operating system, and which are packaged with your app"s APK | ||
| 16 | +# https://developer.android.com/topic/libraries/support-library/androidx-rn | ||
| 17 | +android.useAndroidX=true | ||
| 18 | +# Kotlin code style for this project: "official" or "obsolete": | ||
| 19 | +kotlin.code.style=official | ||
| 20 | +# Enables namespacing of each library's R class so that its R class includes only the | ||
| 21 | +# resources declared in the library itself and none from the library's dependencies, | ||
| 22 | +# thereby reducing the size of the R class for that library | ||
| 23 | +android.nonTransitiveRClass=true |
gradle/wrapper/gradle-wrapper.jar
0 → 100644
No preview for this file type
gradle/wrapper/gradle-wrapper.properties
0 → 100644
gradlew
0 → 100644
| 1 | +#!/usr/bin/env sh | ||
| 2 | + | ||
| 3 | +# | ||
| 4 | +# Copyright 2015 the original author or authors. | ||
| 5 | +# | ||
| 6 | +# Licensed under the Apache License, Version 2.0 (the "License"); | ||
| 7 | +# you may not use this file except in compliance with the License. | ||
| 8 | +# You may obtain a copy of the License at | ||
| 9 | +# | ||
| 10 | +# https://www.apache.org/licenses/LICENSE-2.0 | ||
| 11 | +# | ||
| 12 | +# Unless required by applicable law or agreed to in writing, software | ||
| 13 | +# distributed under the License is distributed on an "AS IS" BASIS, | ||
| 14 | +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
| 15 | +# See the License for the specific language governing permissions and | ||
| 16 | +# limitations under the License. | ||
| 17 | +# | ||
| 18 | + | ||
| 19 | +############################################################################## | ||
| 20 | +## | ||
| 21 | +## Gradle start up script for UN*X | ||
| 22 | +## | ||
| 23 | +############################################################################## | ||
| 24 | + | ||
| 25 | +# Attempt to set APP_HOME | ||
| 26 | +# Resolve links: $0 may be a link | ||
| 27 | +PRG="$0" | ||
| 28 | +# Need this for relative symlinks. | ||
| 29 | +while [ -h "$PRG" ] ; do | ||
| 30 | + ls=`ls -ld "$PRG"` | ||
| 31 | + link=`expr "$ls" : '.*-> \(.*\)$'` | ||
| 32 | + if expr "$link" : '/.*' > /dev/null; then | ||
| 33 | + PRG="$link" | ||
| 34 | + else | ||
| 35 | + PRG=`dirname "$PRG"`"/$link" | ||
| 36 | + fi | ||
| 37 | +done | ||
| 38 | +SAVED="`pwd`" | ||
| 39 | +cd "`dirname \"$PRG\"`/" >/dev/null | ||
| 40 | +APP_HOME="`pwd -P`" | ||
| 41 | +cd "$SAVED" >/dev/null | ||
| 42 | + | ||
| 43 | +APP_NAME="Gradle" | ||
| 44 | +APP_BASE_NAME=`basename "$0"` | ||
| 45 | + | ||
| 46 | +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||
| 47 | +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | ||
| 48 | + | ||
| 49 | +# Use the maximum available, or set MAX_FD != -1 to use that value. | ||
| 50 | +MAX_FD="maximum" | ||
| 51 | + | ||
| 52 | +warn () { | ||
| 53 | + echo "$*" | ||
| 54 | +} | ||
| 55 | + | ||
| 56 | +die () { | ||
| 57 | + echo | ||
| 58 | + echo "$*" | ||
| 59 | + echo | ||
| 60 | + exit 1 | ||
| 61 | +} | ||
| 62 | + | ||
| 63 | +# OS specific support (must be 'true' or 'false'). | ||
| 64 | +cygwin=false | ||
| 65 | +msys=false | ||
| 66 | +darwin=false | ||
| 67 | +nonstop=false | ||
| 68 | +case "`uname`" in | ||
| 69 | + CYGWIN* ) | ||
| 70 | + cygwin=true | ||
| 71 | + ;; | ||
| 72 | + Darwin* ) | ||
| 73 | + darwin=true | ||
| 74 | + ;; | ||
| 75 | + MINGW* ) | ||
| 76 | + msys=true | ||
| 77 | + ;; | ||
| 78 | + NONSTOP* ) | ||
| 79 | + nonstop=true | ||
| 80 | + ;; | ||
| 81 | +esac | ||
| 82 | + | ||
| 83 | +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar | ||
| 84 | + | ||
| 85 | + | ||
| 86 | +# Determine the Java command to use to start the JVM. | ||
| 87 | +if [ -n "$JAVA_HOME" ] ; then | ||
| 88 | + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then | ||
| 89 | + # IBM's JDK on AIX uses strange locations for the executables | ||
| 90 | + JAVACMD="$JAVA_HOME/jre/sh/java" | ||
| 91 | + else | ||
| 92 | + JAVACMD="$JAVA_HOME/bin/java" | ||
| 93 | + fi | ||
| 94 | + if [ ! -x "$JAVACMD" ] ; then | ||
| 95 | + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME | ||
| 96 | + | ||
| 97 | +Please set the JAVA_HOME variable in your environment to match the | ||
| 98 | +location of your Java installation." | ||
| 99 | + fi | ||
| 100 | +else | ||
| 101 | + JAVACMD="java" | ||
| 102 | + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||
| 103 | + | ||
| 104 | +Please set the JAVA_HOME variable in your environment to match the | ||
| 105 | +location of your Java installation." | ||
| 106 | +fi | ||
| 107 | + | ||
| 108 | +# Increase the maximum file descriptors if we can. | ||
| 109 | +if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then | ||
| 110 | + MAX_FD_LIMIT=`ulimit -H -n` | ||
| 111 | + if [ $? -eq 0 ] ; then | ||
| 112 | + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then | ||
| 113 | + MAX_FD="$MAX_FD_LIMIT" | ||
| 114 | + fi | ||
| 115 | + ulimit -n $MAX_FD | ||
| 116 | + if [ $? -ne 0 ] ; then | ||
| 117 | + warn "Could not set maximum file descriptor limit: $MAX_FD" | ||
| 118 | + fi | ||
| 119 | + else | ||
| 120 | + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" | ||
| 121 | + fi | ||
| 122 | +fi | ||
| 123 | + | ||
| 124 | +# For Darwin, add options to specify how the application appears in the dock | ||
| 125 | +if $darwin; then | ||
| 126 | + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" | ||
| 127 | +fi | ||
| 128 | + | ||
| 129 | +# For Cygwin or MSYS, switch paths to Windows format before running java | ||
| 130 | +if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then | ||
| 131 | + APP_HOME=`cygpath --path --mixed "$APP_HOME"` | ||
| 132 | + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` | ||
| 133 | + | ||
| 134 | + JAVACMD=`cygpath --unix "$JAVACMD"` | ||
| 135 | + | ||
| 136 | + # We build the pattern for arguments to be converted via cygpath | ||
| 137 | + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` | ||
| 138 | + SEP="" | ||
| 139 | + for dir in $ROOTDIRSRAW ; do | ||
| 140 | + ROOTDIRS="$ROOTDIRS$SEP$dir" | ||
| 141 | + SEP="|" | ||
| 142 | + done | ||
| 143 | + OURCYGPATTERN="(^($ROOTDIRS))" | ||
| 144 | + # Add a user-defined pattern to the cygpath arguments | ||
| 145 | + if [ "$GRADLE_CYGPATTERN" != "" ] ; then | ||
| 146 | + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" | ||
| 147 | + fi | ||
| 148 | + # Now convert the arguments - kludge to limit ourselves to /bin/sh | ||
| 149 | + i=0 | ||
| 150 | + for arg in "$@" ; do | ||
| 151 | + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` | ||
| 152 | + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option | ||
| 153 | + | ||
| 154 | + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition | ||
| 155 | + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` | ||
| 156 | + else | ||
| 157 | + eval `echo args$i`="\"$arg\"" | ||
| 158 | + fi | ||
| 159 | + i=`expr $i + 1` | ||
| 160 | + done | ||
| 161 | + case $i in | ||
| 162 | + 0) set -- ;; | ||
| 163 | + 1) set -- "$args0" ;; | ||
| 164 | + 2) set -- "$args0" "$args1" ;; | ||
| 165 | + 3) set -- "$args0" "$args1" "$args2" ;; | ||
| 166 | + 4) set -- "$args0" "$args1" "$args2" "$args3" ;; | ||
| 167 | + 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; | ||
| 168 | + 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; | ||
| 169 | + 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; | ||
| 170 | + 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; | ||
| 171 | + 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; | ||
| 172 | + esac | ||
| 173 | +fi | ||
| 174 | + | ||
| 175 | +# Escape application args | ||
| 176 | +save () { | ||
| 177 | + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done | ||
| 178 | + echo " " | ||
| 179 | +} | ||
| 180 | +APP_ARGS=`save "$@"` | ||
| 181 | + | ||
| 182 | +# Collect all arguments for the java command, following the shell quoting and substitution rules | ||
| 183 | +eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" | ||
| 184 | + | ||
| 185 | +exec "$JAVACMD" "$@" |
gradlew.bat
0 → 100644
| 1 | +@rem | ||
| 2 | +@rem Copyright 2015 the original author or authors. | ||
| 3 | +@rem | ||
| 4 | +@rem Licensed under the Apache License, Version 2.0 (the "License"); | ||
| 5 | +@rem you may not use this file except in compliance with the License. | ||
| 6 | +@rem You may obtain a copy of the License at | ||
| 7 | +@rem | ||
| 8 | +@rem https://www.apache.org/licenses/LICENSE-2.0 | ||
| 9 | +@rem | ||
| 10 | +@rem Unless required by applicable law or agreed to in writing, software | ||
| 11 | +@rem distributed under the License is distributed on an "AS IS" BASIS, | ||
| 12 | +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
| 13 | +@rem See the License for the specific language governing permissions and | ||
| 14 | +@rem limitations under the License. | ||
| 15 | +@rem | ||
| 16 | + | ||
| 17 | +@if "%DEBUG%" == "" @echo off | ||
| 18 | +@rem ########################################################################## | ||
| 19 | +@rem | ||
| 20 | +@rem Gradle startup script for Windows | ||
| 21 | +@rem | ||
| 22 | +@rem ########################################################################## | ||
| 23 | + | ||
| 24 | +@rem Set local scope for the variables with windows NT shell | ||
| 25 | +if "%OS%"=="Windows_NT" setlocal | ||
| 26 | + | ||
| 27 | +set DIRNAME=%~dp0 | ||
| 28 | +if "%DIRNAME%" == "" set DIRNAME=. | ||
| 29 | +set APP_BASE_NAME=%~n0 | ||
| 30 | +set APP_HOME=%DIRNAME% | ||
| 31 | + | ||
| 32 | +@rem Resolve any "." and ".." in APP_HOME to make it shorter. | ||
| 33 | +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi | ||
| 34 | + | ||
| 35 | +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||
| 36 | +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" | ||
| 37 | + | ||
| 38 | +@rem Find java.exe | ||
| 39 | +if defined JAVA_HOME goto findJavaFromJavaHome | ||
| 40 | + | ||
| 41 | +set JAVA_EXE=java.exe | ||
| 42 | +%JAVA_EXE% -version >NUL 2>&1 | ||
| 43 | +if "%ERRORLEVEL%" == "0" goto execute | ||
| 44 | + | ||
| 45 | +echo. | ||
| 46 | +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||
| 47 | +echo. | ||
| 48 | +echo Please set the JAVA_HOME variable in your environment to match the | ||
| 49 | +echo location of your Java installation. | ||
| 50 | + | ||
| 51 | +goto fail | ||
| 52 | + | ||
| 53 | +:findJavaFromJavaHome | ||
| 54 | +set JAVA_HOME=%JAVA_HOME:"=% | ||
| 55 | +set JAVA_EXE=%JAVA_HOME%/bin/java.exe | ||
| 56 | + | ||
| 57 | +if exist "%JAVA_EXE%" goto execute | ||
| 58 | + | ||
| 59 | +echo. | ||
| 60 | +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% | ||
| 61 | +echo. | ||
| 62 | +echo Please set the JAVA_HOME variable in your environment to match the | ||
| 63 | +echo location of your Java installation. | ||
| 64 | + | ||
| 65 | +goto fail | ||
| 66 | + | ||
| 67 | +:execute | ||
| 68 | +@rem Setup the command line | ||
| 69 | + | ||
| 70 | +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar | ||
| 71 | + | ||
| 72 | + | ||
| 73 | +@rem Execute Gradle | ||
| 74 | +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* | ||
| 75 | + | ||
| 76 | +:end | ||
| 77 | +@rem End local scope for the variables with windows NT shell | ||
| 78 | +if "%ERRORLEVEL%"=="0" goto mainEnd | ||
| 79 | + | ||
| 80 | +:fail | ||
| 81 | +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of | ||
| 82 | +rem the _cmd.exe /c_ return code! | ||
| 83 | +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 | ||
| 84 | +exit /b 1 | ||
| 85 | + | ||
| 86 | +:mainEnd | ||
| 87 | +if "%OS%"=="Windows_NT" endlocal | ||
| 88 | + | ||
| 89 | +:omega |
settings.gradle
0 → 100644
| 1 | +pluginManagement { | ||
| 2 | + repositories { | ||
| 3 | + gradlePluginPortal() | ||
| 4 | + google() | ||
| 5 | + mavenCentral() | ||
| 6 | + } | ||
| 7 | +} | ||
| 8 | +dependencyResolutionManagement { | ||
| 9 | + repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) | ||
| 10 | + repositories { | ||
| 11 | + google() | ||
| 12 | + mavenCentral() | ||
| 13 | + } | ||
| 14 | +} | ||
| 15 | +rootProject.name = "bigScreem" | ||
| 16 | +include ':app' |