commit 5a1ecc28e98541e8226e1e92b07bd2747e607bb3 Author: wlt233 <1486185683@qq.com> Date: Mon Apr 17 08:28:18 2023 +0800 init (v0.1) diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aa724b7 --- /dev/null +++ b/.gitignore @@ -0,0 +1,15 @@ +*.iml +.gradle +/local.properties +/.idea/caches +/.idea/libraries +/.idea/modules.xml +/.idea/workspace.xml +/.idea/navEditor.xml +/.idea/assetWizardSettings.xml +.DS_Store +/build +/captures +.externalNativeBuild +.cxx +local.properties diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..26d3352 --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,3 @@ +# Default ignored files +/shelf/ +/workspace.xml diff --git a/.idea/compiler.xml b/.idea/compiler.xml new file mode 100644 index 0000000..b589d56 --- /dev/null +++ b/.idea/compiler.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/gradle.xml b/.idea/gradle.xml new file mode 100644 index 0000000..a2d7c21 --- /dev/null +++ b/.idea/gradle.xml @@ -0,0 +1,19 @@ + + + + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000..6ba6396 --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,10 @@ + + + + + + + + + \ No newline at end of file diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 0000000..42afabf --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build \ No newline at end of file diff --git a/app/build.gradle b/app/build.gradle new file mode 100644 index 0000000..2825d5a --- /dev/null +++ b/app/build.gradle @@ -0,0 +1,49 @@ +plugins { + id 'com.android.application' + id 'org.jetbrains.kotlin.android' +} + +android { + namespace 'moe.tqlwsl.aicemu' + compileSdk 33 + + defaultConfig { + applicationId "moe.tqlwsl.aicemu" + minSdk 24 + targetSdk 33 + versionCode 1 + versionName "1.0" + + testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" + } + + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' + } + } + compileOptions { + sourceCompatibility JavaVersion.VERSION_1_8 + targetCompatibility JavaVersion.VERSION_1_8 + } + kotlinOptions { + jvmTarget = '1.8' + } + buildFeatures { + viewBinding true + } +} + +dependencies { + implementation 'androidx.core:core-ktx:1.7.0' + implementation 'androidx.appcompat:appcompat:1.4.1' + implementation 'com.google.android.material:material:1.5.0' + implementation 'androidx.constraintlayout:constraintlayout:2.1.3' + implementation 'androidx.navigation:navigation-fragment-ktx:2.4.1' + implementation 'androidx.navigation:navigation-ui-ktx:2.4.1' + testImplementation 'junit:junit:4.13.2' + androidTestImplementation 'androidx.test.ext:junit:1.1.3' + androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0' + implementation 'com.google.code.gson:gson:2.8.9' +} \ No newline at end of file diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro new file mode 100644 index 0000000..481bb43 --- /dev/null +++ b/app/proguard-rules.pro @@ -0,0 +1,21 @@ +# Add project specific ProGuard rules here. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile \ No newline at end of file diff --git a/app/src/androidTest/java/moe/tqlwsl/aicemu/ExampleInstrumentedTest.kt b/app/src/androidTest/java/moe/tqlwsl/aicemu/ExampleInstrumentedTest.kt new file mode 100644 index 0000000..4d5c1ec --- /dev/null +++ b/app/src/androidTest/java/moe/tqlwsl/aicemu/ExampleInstrumentedTest.kt @@ -0,0 +1,24 @@ +package moe.tqlwsl.aicemu + +import androidx.test.platform.app.InstrumentationRegistry +import androidx.test.ext.junit.runners.AndroidJUnit4 + +import org.junit.Test +import org.junit.runner.RunWith + +import org.junit.Assert.* + +/** + * Instrumented test, which will execute on an Android device. + * + * See [testing documentation](http://d.android.com/tools/testing). + */ +@RunWith(AndroidJUnit4::class) +class ExampleInstrumentedTest { + @Test + fun useAppContext() { + // Context of the app under test. + val appContext = InstrumentationRegistry.getInstrumentation().targetContext + assertEquals("moe.tqlwsl.aicemu", appContext.packageName) + } +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..a94d4a8 --- /dev/null +++ b/app/src/main/AndroidManifest.xml @@ -0,0 +1,50 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/assets/felica_template.json b/app/src/main/assets/felica_template.json new file mode 100644 index 0000000..98f53f3 --- /dev/null +++ b/app/src/main/assets/felica_template.json @@ -0,0 +1,29 @@ +{ + "00": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "01": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "02": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "03": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "04": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "05": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "06": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "07": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "08": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "09": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "0A": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "0B": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "0C": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "0D": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "0E": "FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF", + "80": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "81": "00 00 00 00 00 00 00 00", + "82": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "83": "00 00 00 00 00 00 00 00 00 F1 00 00 00 01 43 00", + "84": "00 00", + "85": "88 B4", + "86": "00 01", + "87": "00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00", + "88": "FE 7F 00 00 07 01 1E 00 FF 41 FF 41 01", + "90": "00 00 00", + "91": "", + "92": "00 00" +} \ No newline at end of file diff --git a/app/src/main/java/moe/tqlwsl/aicemu/EmuCard.kt b/app/src/main/java/moe/tqlwsl/aicemu/EmuCard.kt new file mode 100644 index 0000000..851c33d --- /dev/null +++ b/app/src/main/java/moe/tqlwsl/aicemu/EmuCard.kt @@ -0,0 +1,96 @@ +package moe.tqlwsl.aicemu + +import android.nfc.cardemulation.HostNfcFService +import android.os.Bundle +import android.util.Log +import android.widget.Toast + + +class EmuCard : HostNfcFService() { + private lateinit var card: FelicaCard + + + // byte utils + fun byteArrayOfInts(vararg ints: Int) = ByteArray(ints.size) { pos -> ints[pos].toByte() } + fun ByteArray.toHexString(hasSpace: Boolean = true) = this.joinToString("") { + (it.toInt() and 0xFF).toString(16).padStart(2, '0').uppercase() + if (hasSpace) " " else "" + } + + + // resp utils + fun packResponse(respType: Byte, nfcid2: ByteArray, payload: ByteArray): ByteArray { + var resp = ByteArray(1) + respType + nfcid2 + payload + resp[0] = resp.size.toByte() + return resp + } + + override fun processNfcFPacket(commandPacket: ByteArray, extras: Bundle?): ByteArray? { + val commandHexStr = commandPacket.toHexString() + Log.d("HCEFService", "processNfcFPacket NFCF") + Log.d("HCEFService", "received $commandHexStr") + //Toast.makeText(this, "received $commandHexStr", Toast.LENGTH_LONG).show() + + if (commandPacket.size < 1 + 1 + 8 || (commandPacket.size.toByte() != commandPacket[0])) { + Log.e("HCEFService", "processNfcFPacket: packet size error") + return null + } + + val nfcid2 = ByteArray(8) + System.arraycopy(commandPacket, 2, nfcid2, 0, 8) +// val myNfcid2 = +// byteArrayOfInts(0x02, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00) +// if (!Arrays.equals(myNfcid2, nfcid2)) { +// Log.e("HCEFService", "processNfcFPacket: nfcid2 error") +// return null +// } + + if (commandPacket[1] == 0x06.toByte()) { // READ BLK + val blockNum = commandPacket[13].toInt() + val payload = ByteArray(2 + 1 + 16 * blockNum) + payload[2] = blockNum.toByte() + for (i in 0 until blockNum) { + val id = commandPacket[13 + 2 + 2 * i] + card.readBlock(id)?.let { System.arraycopy(it, 0, payload, 3 + 16 * i, 16) } + } + val resp = packResponse(0x07.toByte(), nfcid2, payload) + val respHexStr = resp.toHexString() + Log.d("HCEFService", "send $respHexStr") + Toast.makeText(this, "Scanned", Toast.LENGTH_LONG).show() + //Toast.makeText(this, "received $commandHexStr\n\nsend $respHexStr", Toast.LENGTH_LONG).show() + return resp + } + else if (commandPacket[1] == 0x08.toByte()) { // WRITE BLK // not implemented + val payload = ByteArray(2) + val resp = packResponse(0x09.toByte(), nfcid2, payload) + val respHexStr = resp.toHexString() + Log.d("HCEFService", "send $respHexStr") + //Toast.makeText(this, "received $commandHexStr\n\nsend $respHexStr", Toast.LENGTH_LONG).show() + return resp + } + + + return byteArrayOfInts(0x04, 0x11, 0x45, 0x14) + // sendResponsePacket(byteArrayOfInts(0x04, 0x11, 0x45, 0x14)) + // return null + } + + + + override fun onCreate() { + Log.d("HCEFService", "onCreate NFCF") + super.onCreate() + val globalVar = this.applicationContext as GlobalVar + card = FelicaCard(globalVar.IDm) + // Toast.makeText(this, "onCreate", Toast.LENGTH_LONG).show() + } + + override fun onDestroy() { + Log.d("HCEFService", "onDestroy NFCF") + super.onDestroy() + // Toast.makeText(this, "onDestroy", Toast.LENGTH_LONG).show() + } + override fun onDeactivated(reason: Int) { + Log.d("HCEFService", "onDeactivated NFCF") + // Toast.makeText(this, "onDeactivated", Toast.LENGTH_LONG).show() + } +} diff --git a/app/src/main/java/moe/tqlwsl/aicemu/FelicaCard.kt b/app/src/main/java/moe/tqlwsl/aicemu/FelicaCard.kt new file mode 100644 index 0000000..9a9c2a9 --- /dev/null +++ b/app/src/main/java/moe/tqlwsl/aicemu/FelicaCard.kt @@ -0,0 +1,108 @@ +package moe.tqlwsl.aicemu + + +import android.app.Application +import android.content.Context +import android.util.Log +import com.google.gson.Gson +import com.google.gson.JsonSyntaxException +import com.google.gson.reflect.TypeToken +import java.io.BufferedReader +import java.io.File +import java.io.IOException +import java.io.InputStreamReader + +class FelicaCard(IDm: String) { + private var cardData = mutableMapOf() + private val gson = Gson() + + init { +// instance = this +// val fileName = "felica_template.json" +// Log.d("HCEFService", "Load Felica Data from assets/$fileName") +// val fileContent = readAssetFile(applicationContext, fileName) +// val mutableBlock = object : TypeToken>() {}.type +// try { +// val jsonData = gson.fromJson>(fileContent, mutableBlock) +// if (jsonData != null) { +// cardData = jsonData +// val block82 = ByteArray(16) +// val IDmBytes = IDm.decodeHex() +// System.arraycopy(IDmBytes, 0, block82, 0, 16) +// cardData["82"] = block82.toHexString() +// } + val block82 = ByteArray(16) + Log.d("HCEFService","Felica Card $IDm") + val IDmBytes = IDm.decodeHex() + System.arraycopy(IDmBytes, 0, block82, 0, 8) + cardData["82"] = block82.toHexString() + val block = ByteArray(16) + cardData["80"] = block.toHexString() + cardData["86"] = block.toHexString() + cardData["90"] = block.toHexString() + cardData["91"] = block.toHexString() + cardData["00"] = block.toHexString() + Log.d("HCEFService","Felica Card Data") + Log.d("HCEFService","=".repeat(53)) + for (en in cardData.entries) { + Log.d("HCEFService","[${en.key}]: ${en.value}") + } + Log.d("HCEFService","=".repeat(53)) +// } catch (e: IOException) { +// Log.e("Error", "File Read Error") +// } catch (e: JsonSyntaxException) { +// Log.e("Error", "File Syntax Error") +// } + } + + // utils + fun String.decodeHex(): ByteArray = + this.replace(" ", "").chunked(2).map { it.toInt(16).toByte() }.toByteArray() + fun ByteArray.toHexString(hasSpace: Boolean = true) = this.joinToString("") { + (it.toInt() and 0xFF).toString(16).padStart(2, '0').uppercase() + if (hasSpace) " " else "" + } + + fun readBlock(id: Byte): ByteArray? { + var idStr = (id.toInt() and 0xFF).toString(16).padStart(2, '0') + Log.d("HCEFService", "read block [$idStr]") + var blockStr = cardData[idStr] + if (blockStr == null) { + idStr = idStr.uppercase() + blockStr = cardData[idStr] + } + if (blockStr != null) { + val blockData = blockStr.decodeHex() + val resp = ByteArray(16) + System.arraycopy(blockData, 0, resp, 0, blockData.size) + return resp + } + + Log.e("Error", "Invalid Block") + return null + } + + fun writeBlock(id: Byte, data: ByteArray) { } + + fun readAssetFile(context: Context, fileName: String): String { + val stringBuilder = StringBuilder() + + // 获取AssetManager + val assetManager = context.assets + + // 通过AssetManager打开文件 + val inputStream = assetManager.open(fileName) + val bufferedReader = BufferedReader(InputStreamReader(inputStream)) + + // 逐行读取文件内容并将其添加到StringBuilder中 + bufferedReader.forEachLine { line -> + stringBuilder.append(line) + stringBuilder.append("\n") + } + + // 关闭流资源 + bufferedReader.close() + inputStream.close() + + return stringBuilder.toString() + } +} \ No newline at end of file diff --git a/app/src/main/java/moe/tqlwsl/aicemu/GlobalVar.kt b/app/src/main/java/moe/tqlwsl/aicemu/GlobalVar.kt new file mode 100644 index 0000000..29ddfbe --- /dev/null +++ b/app/src/main/java/moe/tqlwsl/aicemu/GlobalVar.kt @@ -0,0 +1,7 @@ +package moe.tqlwsl.aicemu + +import android.app.Application + +class GlobalVar : Application() { + var IDm: String = "" +} \ No newline at end of file diff --git a/app/src/main/java/moe/tqlwsl/aicemu/MainActivity.kt b/app/src/main/java/moe/tqlwsl/aicemu/MainActivity.kt new file mode 100644 index 0000000..e03db69 --- /dev/null +++ b/app/src/main/java/moe/tqlwsl/aicemu/MainActivity.kt @@ -0,0 +1,300 @@ +package moe.tqlwsl.aicemu + +import android.app.PendingIntent +import android.content.* +import android.nfc.NfcAdapter +import android.nfc.cardemulation.NfcFCardEmulation +import android.os.Bundle +import android.util.Log +import android.view.* +import android.widget.* +import androidx.appcompat.app.AppCompatActivity +import androidx.appcompat.widget.Toolbar +import androidx.cardview.widget.CardView +import androidx.core.view.WindowCompat +import androidx.navigation.ui.AppBarConfiguration +import com.google.android.material.floatingactionbutton.FloatingActionButton +import moe.tqlwsl.aicemu.databinding.ActivityMainBinding +import java.io.File +import com.google.gson.Gson +import com.google.gson.JsonSyntaxException +import com.google.gson.reflect.TypeToken +import java.io.IOException + + +internal data class Card(val name: String, val idm: String) + +class MainActivity : AppCompatActivity() { + + private lateinit var appBarConfiguration: AppBarConfiguration + private lateinit var binding: ActivityMainBinding + private lateinit var toolbar: Toolbar + private var readCardBroadcastReceiver: ReadCardBroadcastReceiver? = null + private lateinit var nfcAdapter: NfcAdapter + private lateinit var pendingIntent: PendingIntent + private val gson = Gson() + private var cards = mutableListOf() + private val jsonPath = "card.json" + private lateinit var jsonFile: File + private var showCardID: Boolean = false + private var nfcFCardEmulation: NfcFCardEmulation? = null + private var myComponentName: ComponentName? = null + + private val TAG = "AICEmu" + + + override fun onCreate(savedInstanceState: Bundle?) { + WindowCompat.setDecorFitsSystemWindows(window, false) + super.onCreate(savedInstanceState) + binding = ActivityMainBinding.inflate(layoutInflater) + setContentView(binding.root) + + // toolbar + toolbar = findViewById(R.id.toolbar) + setSupportActionBar(toolbar) + + // fab + findViewById(R.id.fab_add_card).setOnClickListener { + val readCardIntent = Intent(this, ReadCard::class.java) + startActivity(readCardIntent) + } + + // read card callback + readCardBroadcastReceiver = ReadCardBroadcastReceiver() + val intentFilter = IntentFilter("moe.tqlwsl.aicemu.READ_CARD") + registerReceiver(readCardBroadcastReceiver, intentFilter) + + // add intent in order not to read tag twice time + nfcAdapter = NfcAdapter.getDefaultAdapter(this) + val intent = Intent(this, javaClass).apply { + addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP) + } + pendingIntent = PendingIntent.getActivity(this, 0, intent, 0) + + // load json file + jsonFile = File(filesDir, jsonPath) + loadCards() + + // + nfcFCardEmulation = NfcFCardEmulation.getInstance(nfcAdapter) + myComponentName = ComponentName( + "moe.tqlwsl.aicemu", + "moe.tqlwsl.aicemu.EmuCard" + ) + } + + override fun onResume() { + super.onResume() + nfcAdapter.enableForegroundDispatch(this, pendingIntent, null, null) + val orgSys = nfcFCardEmulation?.getSystemCodeForService(myComponentName) + if (orgSys != null) { + setSys(orgSys) + } + nfcFCardEmulation?.enableService(this, myComponentName) + } + + override fun onPause() { + super.onPause() + nfcAdapter.disableForegroundDispatch(this) + nfcFCardEmulation?.disableService(this) + } + + override fun onNewIntent(intent: Intent?) { + super.onNewIntent(intent) + } + + override fun onDestroy() { + super.onDestroy() + unregisterReceiver(readCardBroadcastReceiver) + } + + override fun onCreateOptionsMenu(menu: Menu?): Boolean { + menuInflater.inflate(R.menu.toolbar_menu, menu) + return true + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + return when (item.itemId) { + R.id.toolbar_menu_hide_id -> { + val mainLayout: ViewGroup = findViewById(R.id.mainList) + if (showCardID) { + showCardID = false + for (i in 0 until mainLayout.childCount) { + val child = mainLayout.getChildAt(i) + if (child is CardView) { + val idView = child.findViewById(R.id.card_id) + val idShadowView = child.findViewById(R.id.card_id_shadow) + idView.visibility = View.GONE + idShadowView.visibility = View.VISIBLE + } + } + } + else { + showCardID = true + for (i in 0 until mainLayout.childCount) { + val child = mainLayout.getChildAt(i) + if (child is CardView) { + val idView = child.findViewById(R.id.card_id) + val idShadowView = child.findViewById(R.id.card_id_shadow) + idView.visibility = View.VISIBLE + idShadowView.visibility = View.GONE + } + } + } + true + } + R.id.toolbar_menu_settings -> { + Toast.makeText(applicationContext, "还没做()\nUnder constuction...", Toast.LENGTH_LONG).show() + true + } + else -> super.onOptionsItemSelected(item) + } + } + + + private fun setIDm(idm: String): Boolean { + nfcFCardEmulation?.disableService(this) + val resultIdm = nfcFCardEmulation?.setNfcid2ForService(myComponentName, idm) + nfcFCardEmulation?.enableService(this, myComponentName) + return resultIdm == true + } + + private fun setSys(sys: String): Boolean { + nfcFCardEmulation?.disableService(this) + val resultSys = nfcFCardEmulation?.registerSystemCodeForService(myComponentName, sys) + nfcFCardEmulation?.enableService(this, myComponentName) + return resultSys == true + } + + + + + private fun addCard(name: String, IDm: String?) { + val mainLayout: ViewGroup = findViewById(R.id.mainList) + val inflater = getSystemService(LAYOUT_INFLATER_SERVICE) as LayoutInflater + val cardView = inflater.inflate(R.layout.card, mainLayout, false) + mainLayout.addView(cardView) + val nameTextView = cardView.findViewById(R.id.card_name) + nameTextView.text = name + val IDmTextView = cardView.findViewById(R.id.card_id) + IDmTextView.text = IDm + val menuButton: ImageButton = cardView.findViewById(R.id.card_menu_button) + menuButton.setOnClickListener { + showPopupMenu(it, cardView) + } + cardView.setOnTouchListener { v, _ -> + + val globalVar = this.applicationContext as GlobalVar + val IDmTextView = v.findViewById(R.id.card_id) + globalVar.IDm = IDmTextView.text.toString() + + //setIDm(IDm) + val resultIdm = setIDm("02fe000000000000") + val resultSys = setSys("88B4") + + + val nameTextView = v.findViewById(R.id.card_name) + nameTextView.text = name + if (!resultIdm) { + Toast.makeText(applicationContext, "Error IDm", Toast.LENGTH_LONG ).show() + } + if (!resultSys) { + Toast.makeText(applicationContext, "Error Sys", Toast.LENGTH_LONG ).show() + } + if (resultIdm && resultSys) { + Toast.makeText(applicationContext, "正在模拟$name...", Toast.LENGTH_LONG).show() + } + v.performClick() + true + } + } + + private fun showPopupMenu(v: View, cardView: View) { + val popupMenu = PopupMenu(this, v) + popupMenu.inflate(R.menu.card_menu) // 加载菜单资源文件 + popupMenu.setOnMenuItemClickListener { item: MenuItem -> + when (item.itemId) { + R.id.card_menu_rename -> { + val textView = cardView.findViewById(R.id.card_name) + val editText = cardView.findViewById(R.id.card_name_edit) + textView.visibility = View.GONE + editText.visibility = View.VISIBLE + editText.setText(textView.text) + editText.requestFocus() + editText.setOnFocusChangeListener { v, hasFocus -> + if (!hasFocus) { + textView.text = editText.text + textView.visibility = View.VISIBLE + editText.visibility = View.GONE + saveCards() + } + } + cardView.setOnTouchListener { v, _ -> + editText.clearFocus() + v.performClick() + true + } + true + } + R.id.card_menu_delete -> { + val parentLayout = cardView.parent as? ViewGroup + parentLayout?.removeView(cardView) + saveCards() + true + } + else -> false + } + } + popupMenu.show() // 显示PopupMenu + } + + + private fun loadCards() { + val mutableListCard = object : TypeToken>() {}.type + try { + val jsonCards = gson.fromJson>(jsonFile.readText(), mutableListCard) + if (jsonCards != null) { + cards = jsonCards + } + } catch (e: IOException) { + Log.e("Error", "Save File Read Error") + } catch (e: JsonSyntaxException) { + Log.e("Error", "Save File Syntax Error") + } + val mainLayout: ViewGroup = findViewById(R.id.mainList) + mainLayout.removeAllViews() + for (card in cards) { + addCard(card.name, card.idm) + } + } + + + private fun saveCards() { + cards.clear() + val mainLayout: ViewGroup = findViewById(R.id.mainList) + for (i in 0 until mainLayout.childCount) { + val child = mainLayout.getChildAt(i) + if (child is CardView) { + val nameView = child.findViewById(R.id.card_name) + val idView = child.findViewById(R.id.card_id) + cards.add(Card(nameView.text.toString(), idView.text as String)) + } + } + try { + jsonFile.writeText(gson.toJson(cards).toString()) + } catch (e: IOException) { + Log.e("Error", "Save File Write Error") + } + } + + inner class ReadCardBroadcastReceiver : BroadcastReceiver() { + override fun onReceive(context: Context, intent: Intent) { + val idmString = intent.getStringExtra("Card_IDm") + addCard("AIC Card", idmString) + saveCards() + } + } + + + +} \ No newline at end of file diff --git a/app/src/main/java/moe/tqlwsl/aicemu/ReadCard.kt b/app/src/main/java/moe/tqlwsl/aicemu/ReadCard.kt new file mode 100644 index 0000000..c80e8e2 --- /dev/null +++ b/app/src/main/java/moe/tqlwsl/aicemu/ReadCard.kt @@ -0,0 +1,67 @@ +package moe.tqlwsl.aicemu; + +import android.app.Activity +import android.content.Intent +import android.nfc.NfcAdapter +import android.nfc.Tag +import android.nfc.tech.NfcF +import android.os.Bundle +import android.util.Log +import android.widget.ImageButton +import android.widget.Toast + + +class ReadCard : Activity(), NfcAdapter.ReaderCallback{ + + private var nfcAdapter: NfcAdapter? = null + private val TAG = "AICEmu" + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.add_card) + val closeButton = this.findViewById(R.id.pop_up_close_button) + closeButton.setOnClickListener { finish() } + nfcAdapter = NfcAdapter.getDefaultAdapter(this) + } + + override fun onResume() { + super.onResume() + + nfcAdapter?.let { + // 设置前台调度系统 + val options = Bundle() + options.putInt(NfcAdapter.EXTRA_READER_PRESENCE_CHECK_DELAY, 5000) + it.enableReaderMode(this, this, NfcAdapter.FLAG_READER_NFC_F, options) + } + } + + override fun onPause() { + super.onPause() + nfcAdapter?.disableReaderMode(this) + } + + override fun onTagDiscovered(tag: Tag) { + val nfcF = NfcF.get(tag) + nfcF?.let { + val idm = it.tag.id + val idmString = toHexString(idm) + + Log.d(TAG, "IDm: $idmString") + runOnUiThread { + Toast.makeText(this, "IDm: $idmString", Toast.LENGTH_LONG).show() + } + val intent = Intent("moe.tqlwsl.aicemu.READ_CARD") + intent.putExtra("Card_IDm", idmString) + sendBroadcast(intent) + finish() + } + } + + private fun toHexString(data: ByteArray): String { + val sb = StringBuilder() + for (b in data) { + sb.append(String.format("%02X", b)) + } + return sb.toString() + } +} diff --git a/app/src/main/res/drawable-v24/ic_launcher_foreground.xml b/app/src/main/res/drawable-v24/ic_launcher_foreground.xml new file mode 100644 index 0000000..2b068d1 --- /dev/null +++ b/app/src/main/res/drawable-v24/ic_launcher_foreground.xml @@ -0,0 +1,30 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_launcher_background.xml b/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 0000000..07d5da9 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,170 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml new file mode 100644 index 0000000..f7f5f8f --- /dev/null +++ b/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,49 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/add_card.xml b/app/src/main/res/layout/add_card.xml new file mode 100644 index 0000000..d4beada --- /dev/null +++ b/app/src/main/res/layout/add_card.xml @@ -0,0 +1,37 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card.xml b/app/src/main/res/layout/card.xml new file mode 100644 index 0000000..4a6f1c7 --- /dev/null +++ b/app/src/main/res/layout/card.xml @@ -0,0 +1,70 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/menu/card_menu.xml b/app/src/main/res/menu/card_menu.xml new file mode 100644 index 0000000..cb94c4c --- /dev/null +++ b/app/src/main/res/menu/card_menu.xml @@ -0,0 +1,8 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/menu/toolbar_menu.xml b/app/src/main/res/menu/toolbar_menu.xml new file mode 100644 index 0000000..3682ed2 --- /dev/null +++ b/app/src/main/res/menu/toolbar_menu.xml @@ -0,0 +1,11 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 0000000..eca70cf --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml new file mode 100644 index 0000000..eca70cf --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v33/ic_launcher.xml b/app/src/main/res/mipmap-anydpi-v33/ic_launcher.xml new file mode 100644 index 0000000..6f3b755 --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v33/ic_launcher.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher.webp b/app/src/main/res/mipmap-hdpi/ic_launcher.webp new file mode 100644 index 0000000..c209e78 Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp new file mode 100644 index 0000000..b2dfe3d Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher.webp b/app/src/main/res/mipmap-mdpi/ic_launcher.webp new file mode 100644 index 0000000..4f0f1d6 Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp new file mode 100644 index 0000000..62b611d Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xhdpi/ic_launcher.webp new file mode 100644 index 0000000..948a307 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..1b9a695 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp new file mode 100644 index 0000000..28d4b77 Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..9287f50 Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp new file mode 100644 index 0000000..aa7d642 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000..9126ae3 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp differ diff --git a/app/src/main/res/navigation/nav_graph.xml b/app/src/main/res/navigation/nav_graph.xml new file mode 100644 index 0000000..cf8a632 --- /dev/null +++ b/app/src/main/res/navigation/nav_graph.xml @@ -0,0 +1,5 @@ + + \ No newline at end of file diff --git a/app/src/main/res/values-night/themes.xml b/app/src/main/res/values-night/themes.xml new file mode 100644 index 0000000..c39eaae --- /dev/null +++ b/app/src/main/res/values-night/themes.xml @@ -0,0 +1,11 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml new file mode 100644 index 0000000..97d1dcf --- /dev/null +++ b/app/src/main/res/values/colors.xml @@ -0,0 +1,6 @@ + + + #FF000000 + #FFFFFFFF + #e86871 + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml new file mode 100644 index 0000000..477f4c7 --- /dev/null +++ b/app/src/main/res/values/strings.xml @@ -0,0 +1,14 @@ + + AICEmu + Rename + Delete + Hide/Show IDm + Settings + add card + edit card + close + 012EXXXXXXXXXXXX + AIC Card + 请与AIC卡贴贴! + EmuCard + \ No newline at end of file diff --git a/app/src/main/res/values/style.xml b/app/src/main/res/values/style.xml new file mode 100644 index 0000000..f9be74b --- /dev/null +++ b/app/src/main/res/values/style.xml @@ -0,0 +1,10 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/values/themes.xml b/app/src/main/res/values/themes.xml new file mode 100644 index 0000000..4cd3ae0 --- /dev/null +++ b/app/src/main/res/values/themes.xml @@ -0,0 +1,12 @@ + + + + +