tv浏览网页工具

tv不方便安装浏览器,可能是政策不允许,但是需要浏览网页,需要一个浏览网页工具。

创建一个安卓工程,模板选tv的。

新增布局文件:

XML 复制代码
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <!-- 顶部工具栏:输入框 + 按钮 -->
    <LinearLayout
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        android:padding="10dp"
        android:background="#333333"
        android:visibility="visible">

        <EditText
            android:id="@+id/urlInput"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:hint="请输入完整 URL,如 http://192.168.1.1 或 https://www.doubao.com/chat"
            android:textSize="18sp"
            android:textColor="@android:color/black"
            android:textColorHint="@android:color/darker_gray"
            android:inputType="textUri"
            android:imeOptions="actionGo"
            android:selectAllOnFocus="true"
            android:nextFocusDown="@+id/goFullscreenBtn"
            android:nextFocusUp="@+id/goFullscreenBtn"
            android:background="@android:drawable/editbox_background" />

        <Button
            android:id="@+id/goFullscreenBtn"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="全屏显示"
            android:textSize="16sp" />
    </LinearLayout>

    <!-- WebView 全屏显示区域,初始隐藏 -->
    <WebView
        android:id="@+id/webView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:visibility="gone" />

</FrameLayout>

MainActivity代码如下:

Kotlin 复制代码
package com.example.tv_browser

import android.content.Context
import android.os.Bundle
import android.text.InputFilter
import android.view.KeyEvent
import android.view.View
import android.view.inputmethod.EditorInfo
import android.view.inputmethod.InputMethodManager
import android.webkit.WebChromeClient
import android.webkit.WebView
import android.webkit.WebViewClient
import android.widget.Button
import android.widget.EditText
import android.widget.LinearLayout
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import com.example.tv_browser.R

class MainActivity : AppCompatActivity() {

    private lateinit var urlInput: EditText
    private lateinit var goFullscreenBtn: Button
    private lateinit var webView: WebView
    private lateinit var toolbar: LinearLayout

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        urlInput = findViewById(R.id.urlInput)
        goFullscreenBtn = findViewById(R.id.goFullscreenBtn)
        webView = findViewById(R.id.webView)
        toolbar = findViewById(R.id.toolbar)

        setupWebView()

        // 实时将输入字母转为小写(大小写不敏感)
        urlInput.filters = arrayOf(InputFilter { source, start, end, dest, dstart, dend ->
            val builder = StringBuilder()
            for (i in start until end) {
                builder.append(source[i].lowercaseChar())
            }
            builder.toString()
        })

        // 全屏按钮点击事件
        goFullscreenBtn.setOnClickListener {
            // 先隐藏键盘
            hideKeyboard()

            // 让输入框失去焦点(有助于键盘关闭)
            urlInput.clearFocus()

            var url = urlInput.text.toString().trim()
            if (url.isEmpty()) {
                Toast.makeText(this, "请输入完整的网址(包含 http:// 或 https://)", Toast.LENGTH_SHORT).show()
                return@setOnClickListener
            }
            // 转为小写
            url = url.lowercase()
            // 不自动补全协议,直接加载
            loadUrlAndFullscreen(url)
        }

        // 软键盘"前往"按钮触发
        urlInput.setOnEditorActionListener { _, actionId, _ ->
            if (actionId == EditorInfo.IME_ACTION_GO) {
                goFullscreenBtn.performClick()
                true
            } else false
        }

        urlInput.requestFocus()
    }

    private fun hideKeyboard() {
        val imm = getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager
        imm.hideSoftInputFromWindow(urlInput.windowToken, 0)
    }

    private fun setupWebView() {
        webView.settings.apply {
            javaScriptEnabled = true
            domStorageEnabled = true
            loadWithOverviewMode = true
            useWideViewPort = true
            builtInZoomControls = false
            displayZoomControls = false
            allowFileAccess = true
            allowContentAccess = true
//            webView.settings.userAgentString = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36"
        }

        // 在 WebView 内部打开链接,不调起外部浏览器
        webView.webViewClient = object : WebViewClient() {
            override fun shouldOverrideUrlLoading(view: WebView?, request: android.webkit.WebResourceRequest?): Boolean {
                request?.url?.let { view?.loadUrl(it.toString()) }
                return true
            }
        }

        // 支持全屏视频等
        webView.webChromeClient = WebChromeClient()
    }

    private fun loadUrlAndFullscreen(url: String) {
        toolbar.visibility = View.GONE
        webView.visibility = View.VISIBLE
        webView.loadUrl(url)
    }

    // 处理返回键
    override fun onBackPressed() {
        when {
            webView.visibility == View.VISIBLE && toolbar.visibility == View.GONE -> {
                // 退出全屏,显示工具栏并清空页面
                toolbar.visibility = View.VISIBLE
                webView.visibility = View.GONE
                webView.stopLoading()
                webView.loadUrl("about:blank")
            }
            webView.canGoBack() -> webView.goBack()
            else -> super.onBackPressed()
        }
    }

    // 避免软键盘残留, 但应该不会
    override fun onPause() {
        super.onPause()
        val imm = getSystemService(INPUT_METHOD_SERVICE) as? android.view.inputmethod.InputMethodManager
        currentFocus?.let {
            imm?.hideSoftInputFromWindow(it.windowToken, 0)
        }
    }
}

AndroidManifest文件如下:

XML 复制代码
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.tv_browser">

    <!-- 网络权限 -->
    <uses-permission android:name="android.permission.INTERNET" />

    <!-- 声明 TV 专属特性,必须为 true(因为仅 TV 可用) -->
    <uses-feature
        android:name="android.software.leanback"
        android:required="true" />

    <!-- 声明不需要触摸屏(TV 没有触摸屏) -->
    <uses-feature
        android:name="android.hardware.touchscreen"
        android:required="false" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="TV Browser"
        android:usesCleartextTraffic="true"
        android:theme="@style/Theme.AppCompat.NoActionBar">

        <activity
            android:name=".MainActivity"
            android:exported="true"
            android:configChanges="orientation|screenSize">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
                <category android:name="android.intent.category.LEANBACK_LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>

libs.versions.toml :

Groovy 复制代码
[versions]
agp = "9.0.1"
coreKtx = "1.18.0"
appcompat = "1.7.1"
kotlin = "2.0.21"
composeBom = "2024.09.00"
tvFoundation = "1.0.0-alpha07"
tvMaterial = "1.0.0-alpha07"
lifecycleRuntimeKtx = "2.10.0"
activityCompose = "1.13.0"

[libraries]
androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" }
androidx-appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "appcompat" }
androidx-compose-bom = { group = "androidx.compose", name = "compose-bom", version.ref = "composeBom" }
androidx-compose-ui = { group = "androidx.compose.ui", name = "ui" }
androidx-compose-ui-graphics = { group = "androidx.compose.ui", name = "ui-graphics" }
androidx-compose-ui-tooling = { group = "androidx.compose.ui", name = "ui-tooling" }
androidx-compose-ui-tooling-preview = { group = "androidx.compose.ui", name = "ui-tooling-preview" }
androidx-compose-ui-test-manifest = { group = "androidx.compose.ui", name = "ui-test-manifest" }
androidx-compose-ui-test-junit4 = { group = "androidx.compose.ui", name = "ui-test-junit4" }
androidx-tv-foundation = { group = "androidx.tv", name = "tv-foundation", version.ref = "tvFoundation" }
androidx-tv-material = { group = "androidx.tv", name = "tv-material", version.ref = "tvMaterial" }
androidx-lifecycle-runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", version.ref = "lifecycleRuntimeKtx" }
androidx-activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activityCompose" }

[plugins]
android-application = { id = "com.android.application", version.ref = "agp" }
kotlin-compose = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kotlin" }

app/build.gradle.kts:

Groovy 复制代码
plugins {
    alias(libs.plugins.android.application)
    alias(libs.plugins.kotlin.compose)
}

android {
    namespace = "com.example.tv_browser"
    compileSdk {
        version = release(36)
    }

    defaultConfig {
        applicationId = "com.example.tv_browser"
        minSdk = 23
        targetSdk = 36
        versionCode = 1
        versionName = "1.0"

    }

    buildTypes {
        release {
            isMinifyEnabled = false
            proguardFiles(
                getDefaultProguardFile("proguard-android-optimize.txt"),
                "proguard-rules.pro"
            )
        }
    }
    compileOptions {
        sourceCompatibility = JavaVersion.VERSION_11
        targetCompatibility = JavaVersion.VERSION_11
    }
    buildFeatures {
        compose = true
    }
}

dependencies {
    implementation(libs.androidx.core.ktx)
    implementation(libs.androidx.appcompat)
    implementation(platform(libs.androidx.compose.bom))
    implementation(libs.androidx.compose.ui)
    implementation(libs.androidx.compose.ui.graphics)
    implementation(libs.androidx.compose.ui.tooling.preview)
    implementation(libs.androidx.tv.foundation)
    implementation(libs.androidx.tv.material)
    implementation(libs.androidx.lifecycle.runtime.ktx)
    implementation(libs.androidx.activity.compose)
    androidTestImplementation(platform(libs.androidx.compose.bom))
    androidTestImplementation(libs.androidx.compose.ui.test.junit4)
    debugImplementation(libs.androidx.compose.ui.tooling)
    debugImplementation(libs.androidx.compose.ui.test.manifest)

    implementation("androidx.leanback:leanback:1.0.0")
}

用模拟器运行:

下面那个键盘,用鼠标点击了没用,不知道为啥,直接用键盘可以输入。点击"全屏显示"按钮:

随便输入,提交:

ok.

相关推荐
GitLqr8 小时前
Flutter 3.44 插件内置 Kotlin (KGP) 双向兼容适配指南
android·flutter·dart
随遇丿而安12 小时前
第11周:Activity 跳转与传值 + 跳转优化
android
私人珍藏库13 小时前
[Android] BBLL 开源第三方B哩电视TV端
android·app·生活·工具·多功能
杉氧15 小时前
跨平台资源管理:一套代码如何搞定 Android、iOS 和 Web 的图片与多语言?
android·架构·android jetpack
安卓修改大师16 小时前
安卓修改大师实战:从反编译到定制的完整APK修改指南
android
柚鸥ASO优化17 小时前
安卓APP推广的“降本增效”密码:守好商店内,打好商店外
android·aso优化
我是一颗柠檬18 小时前
【Java项目技术亮点】EXPLAIN深度分析与慢查询治理
android·java·开发语言
Android-Flutter18 小时前
android compose shadow 阴影 使用
android·kotlin·compose
帅次18 小时前
Android 高级工程师面试:Java 多线程与并发 近1年高频追问 22 题
android·java·面试
2501_9437823519 小时前
【共创季稿事节】摩斯电码转换器:编码表与双向转换的实现
android·华为·鸿蒙·鸿蒙系统