Android Kotlin 从 Activity 中获取片段中的 Zxing 条形码结果
Android Kotlin Get the Zxing Barcode Results in Fragment from Activity
使用 Android Studio Kotlin,我在我的应用程序中使用了 Zxing 条码扫描器。我的应用程序使用片段架构。我可以从片段中扫描我的应用程序中的条形码,但扫描结果 returned 到 activity。我故意相信,因为签名需要 activity 作为输入。
我的问题是,
a) 如何从片段调用中获取扫描的 return 值以在片段而不是 activity 中执行 onActivityResult?
要么
b) 将 activity 收到的结果传回片段?
我正在使用以下教程Barcode Scanning in Kotlin
下面的 表示要覆盖该函数,但这似乎并没有真正覆盖父级,因为扫描仪函数签名似乎指的是 activity.
更新说明:
我更改了代码以使用 nav_graph。在不使用 nav_graph 的情况下,我收到了 fragment not found
警告。
Log.d 结果显示调用了 MainActivity 函数。
barcoderead D/MainActivity: Format: UPC_A
Contents: 630509790043
清单添加
<uses-feature android:name="android.hardware.camera.autoFocus" />
<uses-feature android:name="android.hardware.camera" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
build.gradle (app) 添加
implementation 'com.google.zxing:core:3.3.3'
implementation 'com.journeyapps:zxing-android-embedded:3.2.0'
主要Activity
import android.content.Intent
import android.os.Bundle
import android.util.Log
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import androidx.navigation.Navigation
import com.google.android.gms.ads.MobileAds
import com.google.zxing.integration.android.IntentIntegrator
import com.lightningbraingames.mytoybox.R
class MainActivity : AppCompatActivity() {
lateinit var btnBarcode: Button
lateinit var textView: TextView
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
Navigation.findNavController(this,
R.id.nav_host_fragment
)
}
}
片段
import android.content.Intent
import android.os.Bundle
import android.util.Log
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.Button
import android.widget.TextView
import android.widget.Toast
import androidx.fragment.app.Fragment
import com.google.zxing.integration.android.IntentIntegrator
class BarcodeRead : Fragment() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
}
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_barcode_read, container, false)
}
lateinit var btnBarcode: Button
lateinit var textView: TextView
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
btnBarcode = view.findViewById(R.id.button)
textView = view.findViewById(R.id.txtContent)
btnBarcode.setOnClickListener {
val intentIntegrator = IntentIntegrator(activity)
intentIntegrator.setBeepEnabled(false)
intentIntegrator.setCameraId(0)
intentIntegrator.setPrompt("SCAN")
intentIntegrator.setBarcodeImageEnabled(false)
intentIntegrator.initiateScan()
}
}
override fun onActivityResult(
requestCode: Int,
resultCode: Int,
data: Intent?
) {
val result = IntentIntegrator.parseActivityResult(requestCode, resultCode, data)
if (result != null) {
if (result.contents == null) {
Toast.makeText(context, "cancelled", Toast.LENGTH_SHORT).show()
} else {
Log.d("Fragment", "Scanned from Fragment")
Toast.makeText(context, "Scanned -> " + result.contents, Toast.LENGTH_SHORT)
.show()
textView.text = String.format("Scanned Result: %s", result)
Log.d("Fragment", "$result")
}
} else {
super.onActivityResult(requestCode, resultCode, data)
}
}
}
布局
主要 Activity
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity"
android:orientation="vertical">
<androidx.appcompat.widget.Toolbar
android:id="@+id/my_toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="?attr/colorPrimary"
android:elevation="4dp"
android:theme="@style/ThemeOverlay.AppCompat.ActionBar"
app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>
<fragment
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/nav_host_fragment"
android:name="androidx.navigation.fragment.NavHostFragment"
app:defaultNavHost="true"
app:navGraph="@navigation/nav_graph"/>
</LinearLayout>
片段
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/relativeLayout"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="4dp">
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_marginTop="50dp"
android:padding="8dp"
android:text="Tutorials Point"
android:textColor="@android:color/holo_green_dark"
android:textSize="48sp"
android:textStyle="bold" />
<TextView
android:id="@+id/txtContent"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/button"
android:layout_centerInParent="true"
android:layout_marginBottom="10dp"
android:text=" code reader"
android:textSize="16sp"
android:textStyle="bold" />
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:layout_marginTop="50dp"
android:text="Process" />
</RelativeLayout>
如果你的问题是你的片段没有得到结果,也许你从你的片段中遗漏了一些基于 ZXing Documentation:
的东西
你Activity应该
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
}
来自你的片段BarcodeRead
//val intentIntegrator = IntentIntegrator(activity)
val intentIntegrator = IntentIntegrator.forSupportFragment(this) // use this instead
使用 IntentIntegrator.forSupportFragment()
而不是 IntentIntegrator(getActivity())
很重要,因为如果使用后者,结果将仅在 activity 页面中可用。
使用 Android Studio Kotlin,我在我的应用程序中使用了 Zxing 条码扫描器。我的应用程序使用片段架构。我可以从片段中扫描我的应用程序中的条形码,但扫描结果 returned 到 activity。我故意相信,因为签名需要 activity 作为输入。
我的问题是, a) 如何从片段调用中获取扫描的 return 值以在片段而不是 activity 中执行 onActivityResult? 要么 b) 将 activity 收到的结果传回片段?
我正在使用以下教程Barcode Scanning in Kotlin
下面的
更新说明:
我更改了代码以使用 nav_graph。在不使用 nav_graph 的情况下,我收到了 fragment not found
警告。
Log.d 结果显示调用了 MainActivity 函数。
barcoderead D/MainActivity: Format: UPC_A
Contents: 630509790043
清单添加
<uses-feature android:name="android.hardware.camera.autoFocus" />
<uses-feature android:name="android.hardware.camera" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
build.gradle (app) 添加
implementation 'com.google.zxing:core:3.3.3'
implementation 'com.journeyapps:zxing-android-embedded:3.2.0'
主要Activity
import android.content.Intent
import android.os.Bundle
import android.util.Log
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import androidx.navigation.Navigation
import com.google.android.gms.ads.MobileAds
import com.google.zxing.integration.android.IntentIntegrator
import com.lightningbraingames.mytoybox.R
class MainActivity : AppCompatActivity() {
lateinit var btnBarcode: Button
lateinit var textView: TextView
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
Navigation.findNavController(this,
R.id.nav_host_fragment
)
}
}
片段
import android.content.Intent
import android.os.Bundle
import android.util.Log
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.Button
import android.widget.TextView
import android.widget.Toast
import androidx.fragment.app.Fragment
import com.google.zxing.integration.android.IntentIntegrator
class BarcodeRead : Fragment() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
}
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_barcode_read, container, false)
}
lateinit var btnBarcode: Button
lateinit var textView: TextView
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
btnBarcode = view.findViewById(R.id.button)
textView = view.findViewById(R.id.txtContent)
btnBarcode.setOnClickListener {
val intentIntegrator = IntentIntegrator(activity)
intentIntegrator.setBeepEnabled(false)
intentIntegrator.setCameraId(0)
intentIntegrator.setPrompt("SCAN")
intentIntegrator.setBarcodeImageEnabled(false)
intentIntegrator.initiateScan()
}
}
override fun onActivityResult(
requestCode: Int,
resultCode: Int,
data: Intent?
) {
val result = IntentIntegrator.parseActivityResult(requestCode, resultCode, data)
if (result != null) {
if (result.contents == null) {
Toast.makeText(context, "cancelled", Toast.LENGTH_SHORT).show()
} else {
Log.d("Fragment", "Scanned from Fragment")
Toast.makeText(context, "Scanned -> " + result.contents, Toast.LENGTH_SHORT)
.show()
textView.text = String.format("Scanned Result: %s", result)
Log.d("Fragment", "$result")
}
} else {
super.onActivityResult(requestCode, resultCode, data)
}
}
}
布局 主要 Activity
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity"
android:orientation="vertical">
<androidx.appcompat.widget.Toolbar
android:id="@+id/my_toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="?attr/colorPrimary"
android:elevation="4dp"
android:theme="@style/ThemeOverlay.AppCompat.ActionBar"
app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>
<fragment
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/nav_host_fragment"
android:name="androidx.navigation.fragment.NavHostFragment"
app:defaultNavHost="true"
app:navGraph="@navigation/nav_graph"/>
</LinearLayout>
片段
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/relativeLayout"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="4dp">
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_marginTop="50dp"
android:padding="8dp"
android:text="Tutorials Point"
android:textColor="@android:color/holo_green_dark"
android:textSize="48sp"
android:textStyle="bold" />
<TextView
android:id="@+id/txtContent"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/button"
android:layout_centerInParent="true"
android:layout_marginBottom="10dp"
android:text=" code reader"
android:textSize="16sp"
android:textStyle="bold" />
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:layout_marginTop="50dp"
android:text="Process" />
</RelativeLayout>
如果你的问题是你的片段没有得到结果,也许你从你的片段中遗漏了一些基于 ZXing Documentation:
的东西你Activity应该
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
}
来自你的片段BarcodeRead
//val intentIntegrator = IntentIntegrator(activity)
val intentIntegrator = IntentIntegrator.forSupportFragment(this) // use this instead
使用 IntentIntegrator.forSupportFragment()
而不是 IntentIntegrator(getActivity())
很重要,因为如果使用后者,结果将仅在 activity 页面中可用。