{"id":12172,"date":"2026-05-02T09:00:05","date_gmt":"2026-05-02T00:00:05","guid":{"rendered":"https:\/\/www.moonmile.net\/blog\/?p=12172"},"modified":"2026-04-30T18:58:34","modified_gmt":"2026-04-30T09:58:34","slug":"ble-chat-%e5%ae%9f%e6%a9%9f%e3%81%ae%e7%96%8e%e9%80%9a%e7%a2%ba%e8%aa%8d%e3%82%b3%e3%83%bc%e3%83%89%e3%82%92%e8%bf%bd%e5%8a%a0%e3%81%99%e3%82%8b","status":"publish","type":"post","link":"http:\/\/www.moonmile.net\/blog\/archives\/12172","title":{"rendered":"BLE-chat \u5b9f\u6a5f\u306e\u758e\u901a\u78ba\u8a8d\u30b3\u30fc\u30c9\u3092\u8ffd\u52a0\u3059\u308b"},"content":{"rendered":"\n<p>BLE \u3092\u4f7f\u3063\u305f Android \u5b9f\u6a5f\u306e\u81ea\u52d5\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u8003\u3048\u307e\u3059\u3002AI \u3092\u542b\u3081\u305f\u30b9\u30d1\u30a4\u30e9\u30eb\u958b\u767a\u3067\u306f\u3001Web API \u3092\u4f7f\u3063\u305f\u307b\u3046\u304c\u5b9f\u7528\u7684\u306a\u3088\u3046\u306a\u6c17\u304c\u3059\u308b\u306e\u3067\u3059\u304c&#8230;\u3072\u3068\u307e\u305a BLE \u306e\u307b\u3046\u304b\u3089\u3084\u3063\u3066\u307f\u307e\u3059\u3002Web API \u5229\u7528\u306e\u307b\u3046\u306f\u3001\u5148\u65e5\u57f7\u7b46\u3057\u305f PHP &amp; Laravel \u958b\u767a\u306e\u30b5\u30f3\u30d7\u30eb\u30b3\u30fc\u30c9\u3092\u4f7f\u3063\u3066\u3044\u305a\u308c\u3084\u3063\u3066\u307f\u307e\u3059\u3002<br>\u30c6\u30b9\u30c8\u306e\u57fa\u6e96\u306b\u3082\u3044\u308d\u3044\u308d\u3042\u308b\u306e\u3067\u3059\u304c\u3001<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u52d5\u4f5c\u78ba\u8a8d\u3092\u81ea\u52d5\u5316\u3059\u308b\u305f\u3081\u306e\u81ea\u52d5\u5b9f\u884c\u30b3\u30fc\u30c9<\/li>\n\n\n\n<li>\u8a2d\u8a08\u304b\u3089\u751f\u6210\u3055\u308c\u305f\u30b3\u30fc\u30c9\u3092\u78ba\u8a8d\u3059\u308b\u305f\u3081\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9<\/li>\n\n\n\n<li>\u5b8c\u6210\u54c1\u306e\u54c1\u8cea\u3092\u78ba\u8a8d\u3059\u308b\u305f\u3081\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9<\/li>\n<\/ul>\n\n\n\n<p>\u3042\u305f\u308a\u304b\u3089\u624b\u3092\u3064\u3051\u3066\u3044\u304d\u307e\u3057\u3087\u3046\u3002\u3053\u308c\u3082\u3001\u3042\u308c\u3053\u308c\u8003\u3048\u3053\u3080\u3088\u308a\u3082\u3001\u3072\u3068\u307e\u305a\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u3044\u3066\u52d5\u304b\u3057\u3066\u307f\u3066\u304b\u3089\u826f\u3044\u65b9\u6cd5\u3092\u8003\u3048\u308b\u3068\u3044\u3046\u30a2\u30b8\u30e3\u30a4\u30eb\u65b9\u5f0f\u3067\u3059\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u52d5\u4f5c\u78ba\u8a8d\u3092\u3059\u308b\u305f\u3081\u306e\u758e\u901a\u78ba\u8a8d\u30b3\u30fc\u30c9<\/strong><\/h2>\n\n\n\n<p>BLE \u3092\u4f7f\u3063\u305f Android \u30a2\u30d7\u30ea\u306e\u5834\u5408\u3001\u3044\u3061\u3070\u3093\u9762\u5012\u81ed\u3044\u306e\u304c\u758e\u901a\u78ba\u8a8d\u3067\u3059\u3002\u96fb\u6ce2\u306e\u72b6\u614b\u304c\u5c4a\u3044\u3066\u3044\u308b\u306e\u304b\u3001\u30e9\u30a4\u30d6\u30e9\u30ea\u304c\u304d\u3061\u3093\u3068\u52d5\u4f5c\u3057\u3066\u3044\u308b\u306e\u304b\u3001\u9001\u53d7\u4fe1\u304c\u3046\u307e\u304f\u3044\u3063\u3066\u3044\u308b\u306e\u304b\u3001\u3068\u3044\u3046\u8af8\u3005\u306e\u7406\u7531\u304b\u3089\u3001\u30d3\u30eb\u30c9\u3057\u3066\u65b0\u3057\u3044\u30a2\u30d7\u30ea\u3092\u5165\u308c\u76f4\u3057\u305f\u3068\u304d\u306b\u758e\u901a\u78ba\u8a8d\u306f\u5fc5\u9808\u3067\u3059\u3002\u306a\u3093\u3068\u3044\u3063\u3066\u3082\u3001BLE \u304c\u7e4b\u304c\u3063\u3066\u3044\u306a\u3044\u307e\u307e\u306b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u52d5\u304b\u3057\u3066\u3082\u300c\u5168\u7136\u7e4b\u304c\u3089\u306a\u3044\u300d\u305f\u3081\u306b\u3001\u539f\u56e0\u3092\u7279\u5b9a\u3059\u308b\u306e\u305f\u3044\u3078\u3093\u3067\u3059\u3002\u30a2\u30d7\u30ea\u306e\u3042\u308c\u3053\u308c\u306e\u52d5\u4f5c\u3092\u78ba\u8a8d\u3059\u308b\u305f\u3081\u306b\u300c\u78ba\u5b9f\u306b2\u53f0\u3067\u7e4b\u304c\u3063\u3066\u3044\u308b\u300d\u3042\u308b\u3044\u306f\u300c\u78ba\u5b9f\u306b\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u9001\u53d7\u4fe1\u3055\u308c\u3066\u3044\u308b\u300d\u3068\u3044\u3046\u72b6\u614b\u3092\u78ba\u4fdd\u3057\u306a\u3051\u308c\u3070\u3044\u3051\u307e\u305b\u3093\u3002\u30cd\u30c3\u30c8\u30ef\u30fc\u30af\u7cfb\u306e\u30c6\u30b9\u30c8\u3092\u3059\u308b\u3068\u304d\u306b\u306f\u3053\u3053\u304c\u6700\u91cd\u8981\u3067\u3059\u306d\u3002<\/p>\n\n\n\n<p>\u3044\u304f\u3064\u304b\u65b9\u6cd5\u304c\u3042\u308b\u3068\u601d\u3046\u306e\u3067\u3059\u304c\u3001\u3072\u3068\u307e\u305a Claude Code \u306b\u805e\u3044\u3066\u307f\u307e\u3057\u3087\u3046\u3002<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">BLE \u306e\u758e\u901a\u78ba\u8a8d\u304c\u81ea\u52d5\u5316\u3067\u304d\u308b\u3088\u3046\u306b\u3057\u305f\u3044\u3002\u3069\u3093\u306a\u65b9\u6cd5\u304c\u3042\u308b\uff1f<\/pre>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-43.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"721\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-43-1024x721.png\" alt=\"\" class=\"wp-image-12173\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-43-1024x721.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-43-300x211.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-43-768x540.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-43.png 1198w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u57fa\u672c\u306f BLE \u3092\u4f7f\u308f\u306a\u3044\u3067\u5358\u4f53\u30c6\u30b9\u30c8\u3067\u52d5\u4f5c\u3059\u308b\u65b9\u6cd5\u306a\u306e\u3067\u3059\u304c\u3001\u3053\u308c\u306f\u5f8c\u3067\u8ffd\u52a0\u3059\u308b\u3068\u3057\u3066&#8230; Test Uiautomator <a href=\"https:\/\/developer.android.google.cn\/jetpack\/androidx\/releases\/test-uiautomator?hl=ja#2.4.0\">https:\/\/developer.android.google.cn\/jetpack\/androidx\/releases\/test-uiautomator?hl=ja#2.4.0<\/a> \u304c\u4f7f\u3048\u308b\u305d\u3046\u306a\u306e\u3067\u3001\u3053\u308c\u3092\u8a66\u3057\u3066\u307f\u307e\u3059\u3002<\/p>\n\n\n\n<p>\u5148\u306e\u30b9\u30d1\u30a4\u30e9\u30eb\u958b\u767a\u306b\u5f93\u3063\u3066\u300c\u6570\u884c\u306e\u30d7\u30ed\u30f3\u30d7\u30c8\u300d\u3092\u5165\u308c\u3066\u30c6\u30b9\u30c8\u306e\u52d5\u4f5c\u78ba\u8a8d\u30b3\u30fc\u30c9\u3092\u3044\u308c\u3066\u3044\u304d\u307e\u3059\u3002\u3044\u304d\u306a\u308a\u30c6\u30b9\u30c8\u8a2d\u8a08\u66f8\u30ec\u30d9\u30eb\u306e\u3082\u306e\u306f\u4f5c\u308a\u307e\u305b\u3093\u3002<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">\u5b9f\u6a5f2\u53f0\u3068 Uiautomator \u3092\u4f7f\u3063\u3066\u3001\u4ee5\u4e0b\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u3044\u3066<br><br>- \u300cHello\u300d\u30921\u56de\u3060\u3051\u9001\u4fe1\u3059\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9<br>- \u300cHello\u300d+ \u901a\u756a \u30921\u79d2\u304a\u304d\u306b10\u56de\u9001\u4fe1\u3059\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9<br>- \u300cHello\u300d+ \u901a\u756a \u30925\u79d2\u304a\u304d\u306b5\u56de\u9001\u4fe1\u3059\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9<br>- \u53d7\u4fe1\u3057\u305f\u30e1\u30c3\u30bb\u30fc\u30b8 adb \u30b3\u30de\u30f3\u30c9\u3067\u51fa\u529b\u3059\u308b\u30b3\u30fc\u30c9<\/pre>\n\n\n\n<p>adb \u30b3\u30de\u30f3\u30c9\u3092\u4f7f\u3063\u3066\u3001androidTest \u3092\u547c\u3073\u51fa\u3059\u3053\u3068\u304c\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"970\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44-1024x970.png\" alt=\"\" class=\"wp-image-12174\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44-1024x970.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44-300x284.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44-768x727.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44-1536x1454.png 1536w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-44.png 1804w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>BleTestBase.kt \u5185\u3067 uiautomator \u3092\u4f7f\u3063\u3066\u64cd\u4f5c<\/p>\n\n\n<div class=\"wp-block-syntaxhighlighter-code \"><pre class=\"brush: java; title: ; notranslate\" title=\"\">\npackage net.moonmile.ble5_chat.claude\n\nimport android.content.Intent\nimport android.util.Log\nimport androidx.test.platform.app.InstrumentationRegistry\nimport androidx.test.uiautomator.By\nimport androidx.test.uiautomator.UiDevice\nimport androidx.test.uiautomator.Until\nimport org.junit.Before\n\n\/**\n * BLE \u30c1\u30e3\u30c3\u30c8\u30c6\u30b9\u30c8\u306e\u5171\u901a\u57fa\u5e95\u30af\u30e9\u30b9\u3002\n *\n * \u8d77\u52d5\u30e2\u30fc\u30c9\u304c 2 \u7a2e\u985e\u3042\u308b:\n *   - setUp()              \u2026 \u30a2\u30d7\u30ea\u3092\u5b8c\u5168\u306b\u518d\u8d77\u52d5\uff08BLE \u521d\u671f\u5316\u304b\u3089\u3084\u308a\u76f4\u3059\uff09\n *   - attachToRunningApp() \u2026 \u65e2\u306b\u8d77\u52d5\u3057\u3066\u3044\u308b\u30a2\u30d7\u30ea\u306b\u30a2\u30bf\u30c3\u30c1\uff08BLE \u72b6\u614b\u3092\u7dad\u6301\uff09\n *\n * attachToRunningApp() \u3092\u4f7f\u3046\u5834\u5408\u306f\u3001\u30b5\u30d6\u30af\u30e9\u30b9\u3067 setUp() \u3092\u30aa\u30fc\u30d0\u30fc\u30e9\u30a4\u30c9\u3057\u3066\u547c\u3076\u3002\n *\/\nabstract class BleTestBase {\n\n    protected lateinit var device: UiDevice\n\n    companion object {\n        const val PACKAGE        = &quot;net.moonmile.ble5_chat.claude&quot;\n        const val LAUNCH_TIMEOUT = 8_000L   \/\/ \u30a2\u30d7\u30ea\u8d77\u52d5\u5f85\u6a5f (ms)\n        const val BLE_INIT_WAIT  = 2_000L   \/\/ BLE \u521d\u671f\u5316\u5f85\u6a5f (ms)  \u203b\u30d5\u30eb\u8d77\u52d5\u6642\u306e\u307f\u4f7f\u7528\n        const val ATTACH_WAIT    = 500L     \/\/ \u30a2\u30bf\u30c3\u30c1\u5f8c\u306e UI \u5b89\u5b9a\u5f85\u6a5f (ms)\n        const val SEND_SETTLE    = 500L     \/\/ \u9001\u4fe1\u5f8c\u306e UI \u5b89\u5b9a\u5f85\u6a5f (ms)\n        const val TAG            = &quot;BleChatTest&quot;\n    }\n\n    \/\/ \u2500\u2500 \u30d5\u30eb\u8d77\u52d5\u30e2\u30fc\u30c9 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    \/**\n     * \u30a2\u30d7\u30ea\u3092\u5b8c\u5168\u306b\u518d\u8d77\u52d5\u3059\u308b\u3002\n     * \u30bf\u30b9\u30af\u3092\u30af\u30ea\u30a2\u3057\u3066 MainActivity \u3092\u65b0\u898f\u8d77\u52d5\u3057\u3001BLE \u521d\u671f\u5316\u307e\u3067\u5f85\u6a5f\u3059\u308b\u3002\n     *\/\n    @Before\n    open fun setUp() {\n        val instrumentation = InstrumentationRegistry.getInstrumentation()\n        device = UiDevice.getInstance(instrumentation)\n\n        \/\/ \u30db\u30fc\u30e0\u306b\u623b\u3057\u3066\u304b\u3089\u8d77\u52d5\uff08\u524d\u56de\u306e\u72b6\u614b\u3092\u30ea\u30bb\u30c3\u30c8\uff09\n        device.pressHome()\n\n        val intent = instrumentation.context.packageManager\n            .getLaunchIntentForPackage(PACKAGE)\n            ?.apply { addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK) }\n            ?: error(&quot;\u30d1\u30c3\u30b1\u30fc\u30b8\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093: $PACKAGE&quot;)\n\n        instrumentation.context.startActivity(intent)\n\n        \/\/ \u30a2\u30d7\u30ea\u30a6\u30a3\u30f3\u30c9\u30a6\u304c\u8868\u793a\u3055\u308c\u308b\u307e\u3067\u5f85\u6a5f\n        device.wait(Until.hasObject(By.pkg(PACKAGE).depth(0)), LAUNCH_TIMEOUT)\n\n        \/\/ BLE \u521d\u671f\u5316\uff08\u30b9\u30ad\u30e3\u30f3\u958b\u59cb\uff09\u3092\u5f85\u6a5f\n        Thread.sleep(BLE_INIT_WAIT)\n\n        Log.i(TAG, &quot;setUp: \u30a2\u30d7\u30ea\u8d77\u52d5\u5b8c\u4e86\uff08\u30d5\u30eb\u8d77\u52d5\uff09&quot;)\n    }\n\n    \/\/ \u2500\u2500 \u30a2\u30bf\u30c3\u30c1\u30e2\u30fc\u30c9 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    \/**\n     * \u65e2\u306b\u8d77\u52d5\u3057\u3066\u3044\u308b\u30a2\u30d7\u30ea\u306b\u30a2\u30bf\u30c3\u30c1\u3059\u308b\u3002\n     *\n     * FLAG_ACTIVITY_CLEAR_TASK \u3092\u4ed8\u3051\u305a\u306b startActivity() \u3092\u547c\u3076\u3053\u3068\u3067\u3001\n     * \u65e2\u5b58\u306e\u30bf\u30b9\u30af\u3092\u305d\u306e\u307e\u307e\u30d5\u30a9\u30a2\u30b0\u30e9\u30a6\u30f3\u30c9\u306b\u623b\u3059\u3002\n     * BLE \u306f\u3059\u3067\u306b\u521d\u671f\u5316\u30fb\u30b9\u30ad\u30e3\u30f3\u4e2d\u306e\u72b6\u614b\u304c\u7dad\u6301\u3055\u308c\u308b\u3002\n     *\n     * \u30b5\u30d6\u30af\u30e9\u30b9\u3067 setUp() \u3092\u30aa\u30fc\u30d0\u30fc\u30e9\u30a4\u30c9\u3057\u3066\u547c\u3076:\n     *\n     *   @Before\n     *   override fun setUp() = attachToRunningApp()\n     *\/\n    protected fun attachToRunningApp() {\n        val instrumentation = InstrumentationRegistry.getInstrumentation()\n        device = UiDevice.getInstance(instrumentation)\n\n        \/\/ \u30d5\u30e9\u30b0\u306a\u3057 = \u65e2\u5b58\u30bf\u30b9\u30af\u3092\u30d5\u30a9\u30a2\u30b0\u30e9\u30a6\u30f3\u30c9\u306b\u623b\u3059\u3060\u3051\uff08Activity \u306f\u518d\u751f\u6210\u3055\u308c\u306a\u3044\uff09\n        val intent = instrumentation.context.packageManager\n            .getLaunchIntentForPackage(PACKAGE)\n            ?: error(&quot;\u30d1\u30c3\u30b1\u30fc\u30b8\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093: $PACKAGE&quot;)\n        \/\/ FLAG_ACTIVITY_CLEAR_TASK \u3092\u610f\u56f3\u7684\u306b\u4ed8\u3051\u306a\u3044\n\n        instrumentation.context.startActivity(intent)\n\n        \/\/ \u30a6\u30a3\u30f3\u30c9\u30a6\u304c\u30d5\u30a9\u30a2\u30b0\u30e9\u30a6\u30f3\u30c9\u306b\u6765\u308b\u307e\u3067\u5f85\u6a5f\n        device.wait(Until.hasObject(By.pkg(PACKAGE).depth(0)), LAUNCH_TIMEOUT)\n\n        \/\/ UI \u5b89\u5b9a\u5f85\u6a5f\u306e\u307f\uff08BLE \u518d\u521d\u671f\u5316\u306f\u4e0d\u8981\uff09\n        Thread.sleep(ATTACH_WAIT)\n\n        Log.i(TAG, &quot;attachToRunningApp: \u65e2\u5b58\u30a2\u30d7\u30ea\u306b\u30a2\u30bf\u30c3\u30c1\u5b8c\u4e86&quot;)\n    }\n\n    \/**\n     * \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u5165\u529b\u3057\u3066\u9001\u4fe1\u3059\u308b\u3002\n     *\n     * @param text \u9001\u4fe1\u3059\u308b\u30c6\u30ad\u30b9\u30c8\n     * @throws IllegalStateException \u5165\u529b\u6b04\u307e\u305f\u306f\u9001\u4fe1\u30dc\u30bf\u30f3\u304c\u898b\u3064\u304b\u3089\u306a\u3044\u5834\u5408\n     *\/\n    protected fun sendMessage(text: String) {\n        \/\/ \u30c6\u30ad\u30b9\u30c8\u5165\u529b\u6b04\u3092\u53d6\u5f97\uff08Compose \u306e OutlinedTextField \u306f EditText \u3068\u3057\u3066\u898b\u3048\u308b\uff09\n        val inputField = device.findObject(By.clazz(&quot;android.widget.EditText&quot;))\n            ?: error(&quot;\u5165\u529b\u30d5\u30a3\u30fc\u30eb\u30c9\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3002\u753b\u9762\u304c\u6b63\u3057\u304f\u8868\u793a\u3055\u308c\u3066\u3044\u308b\u304b\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&quot;)\n\n        inputField.clear()\n        inputField.setText(text)\n\n        \/\/ \u9001\u4fe1\u30dc\u30bf\u30f3\u3092\u30bf\u30c3\u30d7\uff08contentDescription = &quot;\u9001\u4fe1&quot;\uff09\n        val sendButton = device.findObject(By.desc(&quot;\u9001\u4fe1&quot;))\n            ?: error(&quot;\u9001\u4fe1\u30dc\u30bf\u30f3\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3002BLE \u304c ON \u306b\u306a\u3063\u3066\u3044\u308b\u304b\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&quot;)\n\n        sendButton.click()\n\n        \/\/ UI \u304c\u5b89\u5b9a\u3059\u308b\u307e\u3067\u5f85\u6a5f\n        Thread.sleep(SEND_SETTLE)\n    }\n}\n<\/pre><\/div>\n\n\n<p>\u9023\u7d9a\u9001\u4fe1\u3092\u8a66\u3059 BleChatSendTest.kt<\/p>\n\n\n<div class=\"wp-block-syntaxhighlighter-code \"><pre class=\"brush: java; title: ; notranslate\" title=\"\">\n@RunWith(AndroidJUnit4::class)\nclass BleChatSendTest : BleTestBase() {\n\n    \/**\n     * \u30a2\u30d7\u30ea\u3092\u518d\u8d77\u52d5\u305b\u305a\u3001\u65e2\u306b\u8d77\u52d5\u3057\u3066\u3044\u308b BLE \u30bb\u30c3\u30b7\u30e7\u30f3\u306b\u30a2\u30bf\u30c3\u30c1\u3059\u308b\u3002\n     * BLE \u306e\u521d\u671f\u5316\u30fb\u30b9\u30ad\u30e3\u30f3\u72b6\u614b\u306f\u7dad\u6301\u3055\u308c\u308b\u305f\u3081\u3001\u9001\u4fe1\u53ef\u80fd\u306b\u306a\u308b\u307e\u3067\u306e\u5f85\u6a5f\u304c\u4e0d\u8981\u3002\n     *\/\n    @Before\n    override fun setUp() = attachToRunningApp()\n\n    \/\/ \u2500\u2500 \u30c6\u30b9\u30c81: \u300cHello\u300d\u30921\u56de\u9001\u4fe1 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    @Test\n    fun sendHelloOnce() {\n        Log.i(TAG, &quot;=== sendHelloOnce \u958b\u59cb ===&quot;)\n\n        sendMessage(&quot;Hello&quot;)\n        Log.i(TAG, &quot;&#x5B;SEND] Hello&quot;)\n\n        Log.i(TAG, &quot;=== sendHelloOnce \u5b8c\u4e86 ===&quot;)\n    }\n\n    \/\/ \u2500\u2500 \u30c6\u30b9\u30c82: \u300cHello + \u901a\u756a\u300d\u30921\u79d2\u304a\u304d\u306b10\u56de\u9001\u4fe1 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    @Test\n    fun sendHelloTenTimesEverySecond() {\n        val total    = 10\n        val interval = 1_000L   \/\/ 1\u79d2\n\n        Log.i(TAG, &quot;=== sendHelloTenTimesEverySecond \u958b\u59cb\uff08${interval}ms \u9593\u9694 \u00d7 $total \u56de\uff09===&quot;)\n\n        repeat(total) { index -&gt;\n            val seq = index + 1\n            val msg = &quot;Hello $seq&quot;\n\n            sendMessage(msg)\n            Log.i(TAG, &quot;&#x5B;SEND]&#x5B;$seq\/$total] $msg&quot;)\n\n            if (seq &lt; total) {\n                Thread.sleep(interval)\n            }\n        }\n\n        Log.i(TAG, &quot;=== sendHelloTenTimesEverySecond \u5b8c\u4e86 ===&quot;)\n    }\n\n    \/\/ \u2500\u2500 \u30c6\u30b9\u30c83: \u300cHello + \u901a\u756a\u300d\u30925\u79d2\u304a\u304d\u306b5\u56de\u9001\u4fe1 \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    @Test\n    fun sendHelloFiveTimesEveryFiveSeconds() {\n        val total    = 5\n        val interval = 5_000L   \/\/ 5\u79d2\n\n        Log.i(TAG, &quot;=== sendHelloFiveTimesEveryFiveSeconds \u958b\u59cb\uff08${interval}ms \u9593\u9694 \u00d7 $total \u56de\uff09===&quot;)\n\n        repeat(total) { index -&gt;\n            val seq = index + 1\n            val msg = &quot;Hello $seq&quot;\n\n            sendMessage(msg)\n            Log.i(TAG, &quot;&#x5B;SEND]&#x5B;$seq\/$total] $msg&quot;)\n\n            if (seq &lt; total) {\n                Thread.sleep(interval)\n            }\n        }\n\n        Log.i(TAG, &quot;=== sendHelloFiveTimesEveryFiveSeconds \u5b8c\u4e86 ===&quot;)\n    }\n}\n<\/pre><\/div>\n\n\n<p>Uiautomator \u306e\u4f7f\u3044\u65b9\u304c\u3084\u3084\u3053\u3057\u3044\u306e\u3067\u3001\u4e8c\u306e\u8db3\u3092\u8e0f\u3093\u3067\u3044\u305f\u306e\u3067\u3059\u304c\u3001AI \u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3067\u3053\u3053\u307e\u3067\u51fa\u6765\u4e0a\u304c\u308b\u306e\u3067\u3042\u308c\u3070\u3001AI \u306b\u4efb\u305b\u3066\u3057\u307e\u3063\u305f\u307b\u3046\u304c\u697d\u3067\u3059\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u758e\u901a\u78ba\u8a8d\u306e\u305f\u3081\u306e\u30c7\u30d0\u30c3\u30b0\u753b\u9762\u3092\u4f5c\u6210\u3059\u308b<\/strong><\/h2>\n\n\n\n<p>\u305f\u3060\u3057\u3001\u3053\u306e\u307e\u307e\u3060\u3068\u3044\u3061\u3044\u3061 adb \u30b3\u30de\u30f3\u30c9\u3092\u53e9\u304b\u306a\u3044\u3068\u3044\u3051\u306a\u3044\u306e\u3067\u3001\u758e\u901a\u78ba\u8a8d\u7528\u306e\u30c7\u30d0\u30c3\u30b0\u753b\u9762\u3092\u8ffd\u52a0\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">\u5b9f\u6a5f\u3067\u758e\u901a\u78ba\u8a8d\u304c\u3067\u304d\u308b\u3088\u3046\u306b\u3001\u30c7\u30d0\u30c3\u30b0\u7528\u306e\u753b\u9762\u3092\u300c\u8a2d\u5b9a\u300d\u2192\u300c\u758e\u901a\u78ba\u8a8d\u300d\u3068\u3044\u3046\u5f62\u3067\u8ffd\u52a0\n\n- \u300cHello\u300d\u30921\u56de\u3060\u3051\u9001\u4fe1\u3059\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\n- \u300cHello\u300d+ \u901a\u756a \u30921\u79d2\u304a\u304d\u306b10\u56de\u9001\u4fe1\u3059\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\n- \u300cHello\u300d+ \u901a\u756a \u30925\u79d2\u304a\u304d\u306b5\u56de\u9001\u4fe1\u3059\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\n- \u53d7\u4fe1\u3057\u305f\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u30ea\u30b9\u30c8\u306b\u8868\u793a<\/pre>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-45.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"721\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-45-1024x721.png\" alt=\"\" class=\"wp-image-12175\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-45-1024x721.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-45-300x211.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-45-768x540.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-45.png 1198w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u3067\u304d\u3042\u304c\u3063\u305f\u3089\u3001Android Studio \u3067\u30d3\u30eb\u30c9\u3057\u3066\u5b9f\u6a5f\u3067\u52d5\u4f5c\u78ba\u8a8d\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"711\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46-1024x711.png\" alt=\"\" class=\"wp-image-12176\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46-1024x711.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46-300x208.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46-768x533.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46-1536x1066.png 1536w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-46.png 1831w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u5b9a\u756a\u306e\u30dc\u30bf\u30f3\u3092\u62bc\u3057\u3066\u3001\u758e\u901a\u3092\u3059\u308b\u7c21\u5358\u306a\u753b\u9762\u306a\u306e\u3067\u3059\u304c\u3001\u3053\u308c\u304c\u3042\u308b\u3068\u7d50\u69cb\u4fbf\u5229\u3067\u3059\u3002\u30b7\u30b9\u30c6\u30e0\u3092\u4f5c\u308b\u3068\u304d\u306b\u3001\u3053\u3093\u306a\u611f\u3058\u3067\u958b\u767a\u7528\u306e\u30c7\u30d0\u30c3\u30b0\u753b\u9762\u3092\u7528\u610f\u3057\u3066\u304a\u304f\u3068\u624b\u8efd\u306b\u30dc\u30bf\u30f3\u3084\u30ea\u30b9\u30c8\u8868\u793a\u306a\u3069\u304c\u3067\u304d\u3066\u4fbf\u5229\u3067\u3059\u3002\u3044\u307e\u307e\u3067\u306f\u3001\u753b\u9762\u3092\u4f5c\u308b\u306e\u306b\u3061\u3087\u3063\u3068\u624b\u9593\u304c\u304b\u304b\u3063\u3066\u5104\u52ab\u306a\u611f\u3058\u3060\u3063\u305f\u306e\u3067\u3059\u304c\u3001AI \u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3067\u3053\u3053\u307e\u3067\u51fa\u6765\u3066\u304f\u308c\u308c\u3070\u3053\u308c\u3067\u5341\u5206\u3067\u3059\u3002<\/p>\n\n\n\n<p>\u3053\u306e\u753b\u9762\u81ea\u4f53\u306f\u3001\u30ec\u30a4\u30a2\u30a6\u30c8\u306f\u7279\u306b\u3053\u3060\u308f\u3089\u306a\u3044\u306e\u3067 AI \u30b3\u30fc\u30c9\u30c7\u30a3\u30f3\u30b0\u306e\u307e\u307e\u3067\u6e08\u307e\u305b\u3066\u304a\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<p>\u3042\u3068\u3001\uff12\u53f0\u3060\u3068\u758e\u901a\u3060\u3051\u3067\u89e3\u308a\u8f9b\u3044\u306e\u3067\u3059\u304c\u3001Android \u304c 3\u53f0\u4ee5\u4e0a\u3042\u308b\u3068\u3001\u540c\u6642\u306b\u8907\u6570\u53f0\u306b BLE \u30a2\u30c9\u30d0\u30bf\u30a4\u30ba\u3067\u9001\u4fe1\u3057\u3066\u3044\u308b\u3053\u3068\u304c\u308f\u304b\u308a\u307e\u3059\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-47.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"768\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-47.png\" alt=\"\" class=\"wp-image-12177\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-47.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-47-300x225.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-47-768x576.png 768w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u3053\u308c\u3001\u5b9f\u969b\u306b\u52d5\u4f5c\u3059\u308b\u307e\u3067\u308f\u304b\u3089\u306a\u304b\u3063\u305f\u306e\u3067\u3059\u304c\u3001\u610f\u5916\u3068 BLE5 \u306e\u62e1\u5f35\u30a2\u30c9\u30d0\u30bf\u30a4\u30ba\u3060\u3068\u629c\u3051\u304c\u3067\u307e\u3059\u306d\u3002\u53f3\u306e\u7aef\u672b\u304b\u3089\u30a2\u30c9\u30d0\u30bf\u30a4\u30ba\u767a\u4fe1\u30921\u79d2\u6bce\u306b\u5207\u308a\u66ff\u3048\u3066\u767a\u4fe1\u3057\u3066\u3044\u308b\u306e\u3067\u3059\u304c\u3001\u771f\u3093\u4e2d\u306e\u7aef\u672b\u3067\u306f10\u4ef6\u3059\u3079\u3066\u53d7\u4fe1\u3057\u3066\u3044\u307e\u3059\u304c\u3001\u5de6\u306e\u7aef\u672b\u3067\u306f\u3068\u3073\u3068\u3073\u30675\u4ef6\u3057\u304b\u53d7\u4fe1\u3067\u304d\u3066\u3044\u307e\u305b\u3093\u3002<\/p>\n\n\n\n<p>\u3053\u308c\u306f\u3001BLE \u30a2\u30c9\u30d0\u30bf\u30a4\u30ba\u3092\u53d7\u4fe1\u3059\u308b\u3068\u304d\u306b\u3042\u308b\u7a0b\u5ea6\u9045\u5ef6\u3059\u308b\u306e\u3067\u3001\u305d\u3053\u3067\u53d6\u308a\u3053\u307c\u3057\u304c\u767a\u751f\u3057\u3066\u3044\u308b\u3068\u8003\u3048\u3089\u308c\u307e\u3059\u3002\u62e1\u5f35\u30a2\u30c9\u30d0\u30bf\u30a4\u30ba\u306e\u5834\u5408\u3001UDP \u901a\u4fe1\u306e\u3088\u3046\u306b\u901a\u4fe1\u30c7\u30fc\u30bf\u304c\u5fc5\u305a\u5c4a\u304f\u3068\u306f\u9650\u3089\u306a\u3044\u306e\u3067\u3001\u3053\u308c\u306f\u3053\u308c\u3067\u6b63\u3057\u3044\u52d5\u4f5c\u3067\u3059\u3002<\/p>\n\n\n\n<p>\u305f\u3060\u3001\u306a\u306b\u304b\u5909\u306a\u30bf\u30a4\u30df\u30f3\u30b0\u3067\u843d\u3061\u3066\u3044\u308b\u306e\u304c\u6c17\u306b\u306a\u308b\u3068\u3053\u308d&#8230;<\/p>\n\n\n<div class=\"wp-block-syntaxhighlighter-code \"><pre class=\"brush: plain; title: ; notranslate\" title=\"\">\n2026-04-30 10:08:18.601 14575-14575 BLE5Chat\/B...nerManager net.moonmile.ble5_chat.claude        D  Duplicate filtered: 3d71fb74-69da-44ad-ab9d-15205c8f437f\n2026-04-30 10:08:18.726 14575-14575 BLE5Chat\/B...nerManager net.moonmile.ble5_chat.claude        D  Duplicate filtered: d44b0b62-7253-4a11-bb7d-052e0c82b652\n2026-04-30 10:08:18.868 14575-14575 BLE5Chat\/B...nerManager net.moonmile.ble5_chat.claude        D  Received: 870038ea-059e-43b7-a4e5-611339da2d92 from 11112222\n2026-04-30 10:08:18.869 14575-14575 BLE5Chat\/MainActivity   net.moonmile.ble5_chat.claude        D  Received: 870038ea-059e-43b7-a4e5-611339da2d92 from 11112222\n2026-04-30 10:08:18.943 14575-14575 AndroidRuntime          net.moonmile.ble5_chat.claude        D  Shutting down VM\n2026-04-30 10:08:18.948 14575-14575 AndroidRuntime          net.moonmile.ble5_chat.claude        E  FATAL EXCEPTION: main (Fix with AI)\n                                                                                                    Process: net.moonmile.ble5_chat.claude, PID: 14575\n                                                                                                    java.lang.IllegalArgumentException: Key &quot;870038ea-059e-43b7-a4e5-611339da2d92&quot; was already used. If you are using LazyColumn\/Row please make sure you provide a unique key for each item.\n                                                                                                    \tat androidx.compose.ui.internal.InlineClassHelperKt.throwIllegalArgumentException(InlineClassHelper.kt:36)\n                                                                                                    \tat androidx.compose.ui.layout.LayoutNodeSubcompositionsState.subcompose(SubcomposeLayout.kt:1366)\n                                                                                                    \tat androidx.compose.ui.layout.LayoutNodeSubcompositionsState$Scope.subcompose(SubcomposeLayout.kt:1231)\n                                                                                                    \tat androidx.compose.foundation.lazy.layout.LazyLayoutMeasureScopeImpl.compose(LazyLayoutMeasureScope.kt:94)\n                                                                                                    \tat androidx.compose.foundation.lazy.layout.LazyLayoutMeasuredItemProvider.getPlaceables-3p2s80s(LazyLayoutMeasuredItem.kt:60)\n                                                                                                    \tat androidx.compose.foundation.lazy.LazyListMeasuredItemProvider.getAndMeasure-0kLqBqw(LazyListMeasuredItemProvider.kt:53)\n                                                                                                    \tat androidx.compose.foundation.lazy.LazyListMeasuredItemProvider.getAndMeasure-0kLqBqw$default(LazyListMeasuredItemProvider.kt:47)\n                                                                                                    \tat androidx.compose.foundation.lazy.LazyListMeasureKt.measureLazyList-LCrQqZ4(LazyListMeasure.kt:224)\n<\/pre><\/div>\n\n\n<p>\u304a\u305d\u3089\u304f\u3001\u30c0\u30d6\u308a\u3067\u53d7\u4fe1\u300cReceived: 870038ea-059e-43b7-a4e5-611339da2d92 from 11112222\u300d\u3057\u305f\u3068\u304d\u306b\u3001\u30ea\u30b9\u30c8\u306b\u8ffd\u52a0\u3059\u308b\u969b\u306b\u30ad\u30fc\u304c\u91cd\u8907\u3057\u3066\u3044\u308b\u3088\u3046\u306a\u6c17\u304c\u3057\u307e\u3059\u3002\u3053\u3053\u306f\u5f8c\u3067\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u7c21\u5358\u306a\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u3044\u3066\u304a\u304f<\/strong><\/h2>\n\n\n\n<p>BLE \u30c1\u30e3\u30fc\u30c3\u3068\u30c4\u30fc\u30eb\u306b\u306f\u3001\u660e\u78ba\u306a\u30ed\u30b8\u30c3\u30af\u30af\u30e9\u30b9\u304c\u3042\u308b\u308f\u3051\u3067\u306f\u306a\u3044\u306e\u3067\u3001\u308f\u3056\u308f\u3056\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u304f\u5fc5\u8981\u3082\u306a\u3044\u306e\u3067\u3059\u304c\u3001\u300c\u8a2d\u8a08\u304b\u3089\u751f\u6210\u3055\u308c\u305f\u30b3\u30fc\u30c9\u3092\u78ba\u8a8d\u3059\u308b\u305f\u3081\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u300d\u3068\u3044\u3046\u89b3\u70b9\u304b\u3089\u3001\u7c21\u6613\u7684\u306a\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u8ffd\u52a0\u3057\u3066\u304a\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<p>\u3044\u308f\u3086\u308b\u30af\u30e9\u30b9\u8a2d\u8a08\u306e\u901a\u308a\u306b\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3055\u308c\u3066\u3044\u308b\u5426\u304b\u3001\u3092\u30c1\u30a7\u30c3\u30af\u3057\u307e\u3059\u3002<\/p>\n\n\n<div class=\"wp-block-syntaxhighlighter-code \"><pre class=\"brush: plain; title: ; notranslate\" title=\"\">\n### 1. \u30af\u30e9\u30b9\u4e00\u89a7\n\n| \u30af\u30e9\u30b9\u540d | \u7a2e\u5225 | \u4e3b\u306a\u8cac\u52d9 |\n|---|---|---|\n| MainActivity | UI Host | Compose \u30a8\u30f3\u30c8\u30ea\u30dd\u30a4\u30f3\u30c8\u3001`setContent`\u3001UI \u72b6\u614b\u7ba1\u7406\u3001\u30e9\u30a4\u30d5\u30b5\u30a4\u30af\u30eb\u9023\u643a |\n| ChatScreen | UI Compose | \u7d14\u7c8b\u306a\u63cf\u753b\uff08State \u3092\u53d7\u3051\u53d6\u308a\u30b3\u30fc\u30eb\u30d0\u30c3\u30af\u3092\u547c\u3076\uff09 |\n| ChatRepository | Domain\/Application | \u9001\u53d7\u4fe1\u30e6\u30fc\u30b9\u30b1\u30fc\u30b9\u7d71\u5408\u3001\u5c65\u6b74\u306e\u4ef2\u4ecb\uff08IF\uff09 |\n| ChatRepositoryImpl | Data | `BleChatService` \u3092\u4f7f\u3063\u305f Repository \u5b9f\u88c5 |\n| BleChatService | BLE Facade | BLE \u521d\u671f\u5316\u3001Advertiser\/Scanner \u306e\u958b\u59cb\u505c\u6b62\u7ba1\u7406 |\n| BleAdvertiserManager | BLE Tx | Extended Advertising \u306b\u3088\u308b\u30e1\u30c3\u30bb\u30fc\u30b8\u9001\u51fa |\n| BleScannerManager | BLE Rx | \u5e83\u544a\u30d1\u30b1\u30c3\u30c8\u53d7\u4fe1\u3001\u91cd\u8907\u6392\u9664 |\n| MessageCodec | Protocol | \u6587\u5b57\u5217\u3068\u30d0\u30a4\u30c8\u5217\u306e\u5909\u63db\uff08\u6700\u5927 100 \u6587\u5b57\u5236\u7d04\u542b\u3080\uff09 |\n| DispatcherProvider | Concurrency | Coroutine Dispatcher \u306e\u62bd\u8c61\u5316\uff08\u30c6\u30b9\u30c8\u5dee\u3057\u66ff\u3048\u7528\uff09 |\n| DuplicateFilter | Reliability | MessageId \u306b\u3088\u308b\u91cd\u8907\u53d7\u4fe1\u6291\u6b62 |\n| PeerRegistry | Session | \u53c2\u52a0\u7aef\u672b\u306e\u6700\u7d42\u53d7\u4fe1\u6642\u523b\u7ba1\u7406\uff08\u53c2\u52a0\u8005\u628a\u63e1\uff09 |\n| FavoriteRepository | Data | \u304a\u6c17\u306b\u5165\u308a\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u8ffd\u52a0\u30fb\u524a\u9664\u30fb\u6c38\u7d9a\u5316\uff08SharedPreferences + JSON\uff09\u3001StateFlow \u3067\u5909\u66f4\u3092\u901a\u77e5 |\n| ChatMessage | Model | \u30c1\u30e3\u30c3\u30c8\u672c\u6587\u3001\u9001\u4fe1\u8005 ID\u3001\u6642\u523b\u3001MessageId |\n| AdvPacket | Model | 1\u5e83\u544a\u5358\u4f4d\u30c7\u30fc\u30bf\uff08MessageId\u3001Seq\u3001Total\u3001Payload\uff09 |\n| ChatUiState | UI Model | Compose \u63cf\u753b\u7528\u72b6\u614b\uff08\u30e1\u30c3\u30bb\u30fc\u30b8\u3001\u5165\u529b\u5024\u3001\u9001\u4fe1\u53ef\u5426\u3001\u30a8\u30e9\u30fc\uff09 |\n| ChatUiEvent | UI Model | UI \u64cd\u4f5c\u30a4\u30d9\u30f3\u30c8\uff08\u9001\u4fe1\u3001\u5165\u529b\u66f4\u65b0\u3001\u958b\u59cb\u3001\u505c\u6b62\uff09 |\n| ChatUiEffect | UI Model | 1\u56de\u6027\u30a4\u30d9\u30f3\u30c8\uff08Toast\u3001\u6a29\u9650\u8981\u6c42\u8a98\u5c0e\uff09 |\n| AppLogger | Cross-cutting | \u30ed\u30b0\u53ce\u96c6\uff08\u30c6\u30b9\u30c8\u30fb\u969c\u5bb3\u89e3\u6790\u7528\uff09 |\n| ErrorHandler | Cross-cutting | BLE \u4f8b\u5916\u3001\u6a29\u9650\u30a8\u30e9\u30fc\u3001BLE OFF \u691c\u77e5\u3001\u5fa9\u65e7\u30ea\u30c8\u30e9\u30a4\u5224\u65ad |\n\n<\/pre><\/div>\n\n\n<p>\u4eca\u56de\u306e\u30b9\u30d1\u30a4\u30e9\u30eb\u958b\u767a\u3067\u306f\u57fa\u672c\u7684\u306b\u3001\u30b3\u30fc\u30c9\u304b\u3089 \u6982\u8981\u8a2d\u8a08.md \u3092\u518d\u4f5c\u6210\u3057\u3066\u3044\u308b\u306e\u3067\u3001\u3053\u3053\u306e\u30af\u30e9\u30b9\u4e00\u89a7\u306f\u6b63\u3057\u3044\u306f\u305a\u3067\u3059\u3002\u4eba\u9593\u306e\u958b\u767a\u8005\u306a\u3089\u3070\u3001\u8a2d\u8a08\u2192\u5b9f\u88c5\u3068\u3044\u3046\u30d7\u30ed\u30bb\u30b9\u306e\u4e2d\u3067\u3001\u8a2d\u8a08\u901a\u308a\u306b\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u304c\u3055\u308c\u3066\u3044\u308b\u306e\u304b\uff1f\u3092\u300c\u5b9f\u88c5\u5de5\u7a0b\u300d\u3068\u3057\u3066\u30c1\u30a7\u30c3\u30af\u3057\u3066\u3044\u304d\u307e\u3059\u3002\u3053\u3053\u306f\u3001\u30b3\u30fc\u30c9\u30ec\u30d3\u30e5\u30fc\u3068\u304b\u3001\u30d2\u30a2\u30ea\u30f3\u30b0\u3068\u304b\u3001\u3042\u308b\u3044\u306f\u30c6\u30b9\u30c8\u5de5\u7a0b\u3067\u306e\u30c1\u30a7\u30c3\u30af\u304c\u5165\u308a\u307e\u3059\u3002<\/p>\n\n\n\n<p>\u540c\u3058\u30d1\u30bf\u30fc\u30f3\u3092 AI \u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u306e\u30d7\u30ed\u30bb\u30b9\u3067\u3082\u4f7f\u3048\u306a\u3044\u304b\u3001\u3068\u3044\u3046\u8a66\u307f\u3067\u3059\u3002<\/p>\n\n\n\n<p>\u304a\u305d\u3089\u304f\u3001\u30af\u30e9\u30b9\u540d\u3084\u30e1\u30bd\u30c3\u30c9\u540d\u3092\u30b3\u30fc\u30c9\u3092\u691c\u7d22\u3057\u3066\u78ba\u8a8d\u3059\u308b\u30c4\u30fc\u30eb\u304c\u3042\u308c\u3070\u5341\u5206\u3060\u3068\u601d\u3046\u306e\u3067\u3059\u304c\u3001\u3053\u3053\u3067\u306f\u7c21\u6613\u7684\u306a\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u3044\u3066\u78ba\u8a8d\u3057\u307e\u3059\u3002\u3044\u308f\u3086\u308b\u3001\u30c6\u30b9\u30c8\u99c6\u52d5\u304b\u3089\u306e\u501f\u7528\u3067\u3059\u304c\u3001\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u5148\u306b\u66f8\u3044\u3066\u5b9f\u88c5\u3068\u3044\u3046\u6d41\u308c\u3067\u306f\u306a\u304f\u3001\u5b9f\u88c5\u30b3\u30fc\u30c9\u306b\u5408\u308f\u305b\u3066\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u304d\u307e\u3059\u3002\u5f53\u305f\u308a\u524d\u3067\u3059\u304c\u3001\u5fc5\u305a\u901a\u308b\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306b\u306a\u308a\u3001\u3053\u306e\u307e\u307e\u3067\u306f\u610f\u5473\u304c\u3042\u308a\u307e\u305b\u3093\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"546\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48-1024x546.png\" alt=\"\" class=\"wp-image-12178\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48-1024x546.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48-300x160.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48-768x410.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48-1536x819.png 1536w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-48.png 1916w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u30b9\u30d1\u30a4\u30e9\u30eb\u958b\u767a\u3067\u3001\u6b21\u306e\u300c\u8a2d\u8a08\u751f\u6210\u300d\u304c\u884c\u308f\u308c\u3066\u300c\u30b3\u30fc\u30c9\u751f\u6210\u300d\u304c\u5b9f\u884c\u3055\u308c\u305f\u3068\u304d\u306b\u3001\u4e0d\u610f\u306b\u30af\u30e9\u30b9\u3084\u30e1\u30bd\u30c3\u30c9\u304c\u6d88\u3048\u3066\u3044\u306a\u3044\u3053\u3068\u3092\u78ba\u8a8d\u3059\u308b\u305f\u3081\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3067\u3059\u3002\u3082\u3061\u308d\u3093\u3001\u8a2d\u8a08\u81ea\u4f53\u304c\u5909\u308f\u3063\u3066\u3057\u307e\u3048\u3070\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306e\u5909\u66f4\u3082\u5fc5\u9808\u306a\u306e\u3067\u3059\u304c\u3001\u73fe\u6642\u70b9\u3067\u306f\u30b3\u30fc\u30c9\u306e\u5909\u66f4\u3060\u3051\u3092\u30c1\u30a7\u30c3\u30af\u3059\u308b\u7c21\u6613\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306b\u7d5e\u308a\u307e\u3059\u3002<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">\u5b9f\u6a5f\u306e BLE \u3092\u4f7f\u308f\u306a\u3044\u5358\u4f53\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u8ffd\u52a0\u3059\u308b<br><br>- net.moonmile.ble5_chat.claude.ble.MessageCodec<br>- net.moonmile.ble5_chat.claude.ble.DuplicateFilter<br>- net.moonmile.ble5_chat.claude.ble.PeerRegistry<br>- net.moonmile.ble5_chat.claude.util.ErrorHandler<br>- net.moonmile.ble5_chat.claude.util.DispatcherProvider<br>- net.moonmile.ble5_chat.claude.util.DefaultDispatcherProvider<br><br>\u30c7\u30fc\u30bf\u30af\u30e9\u30b9\u306e\u7c21\u6613\u30c6\u30b9\u30c8\u3082\u8ffd\u52a0\u3059\u308b<br><br>- net.moonmile.ble5_chat.claude.model.ChatMessage<br>- net.moonmile.ble5_chat.claude.model.AdvPacket<br>- net.moonmile.ble5_chat.claude.model.ChatUiState<br>- net.moonmile.ble5_chat.claude.model.ChatUiEffect<br>- net.moonmile.ble5_chat.claude.model.ChatUiEvent<\/pre>\n\n\n\n<p>\u5f62\u5f0f\u7684\u306a\u578b\u30c1\u30a7\u30c3\u30af\u306a\u306e\u3067\u3001\u5b9f\u6a5f\u3092\u4f7f\u308f\u306a\u3044\u3067\u30c6\u30b9\u30c8\u304c\u3067\u304d\u308b\u30af\u30e9\u30b9\u3060\u3051\u3092\u9078\u3093\u3067\u3044\u307e\u3059\u3002<\/p>\n\n\n\n<p>![](images\/20260430_06.jpg)<\/p>\n\n\n\n<p>\u30b3\u30de\u30f3\u30c9\u30e9\u30a4\u30f3\u3067<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">.\/gradlew test<\/pre>\n\n\n\n<p>\u3068\u3059\u308b\u3068\u3001JUnit \u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u304c\u52d5\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<p>![](images\/20260430_07.jpg)<\/p>\n\n\n\n<p>\u57fa\u672c\u7684\u306b Claude Code \u304c\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u66f8\u3044\u305f\u3042\u3068\u306b\u3001\u81ea\u3089\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u304c\u3059\u3079\u3066\u6210\u529f\u3059\u308b\u307e\u3067\u66f8\u304d\u76f4\u3057\u3066\u304f\u308c\u308b\u306e\u3067\u3001\u521d\u56de\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306f\u3059\u3079\u3066\u6210\u529f\u3059\u308b\u306b\u6c7a\u307e\u3063\u3066\u3044\u307e\u3059\u3002<\/p>\n\n\n\n<p>\u4e2d\u8eab\u306f\u7c21\u5358\u306a\u3082\u306e\u3070\u304b\u308a\u2252\u30c6\u30b9\u30c8\u304c\u901a\u308b\u3082\u306e\u3070\u304b\u308a\u306a\u306e\u3067\u3001\u5185\u5bb9\u306f\u3042\u307e\u308a\u95a2\u4fc2\u3042\u308a\u307e\u305b\u3093\u3002\u6b21\u56de\u4ee5\u964d<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u6570\u884c\u306e\u30d7\u30ed\u30f3\u30d7\u30c8\u3067\u30b3\u30fc\u30c9\u751f\u6210\u3092\u3057\u305f\u3068\u304d<\/li>\n\n\n\n<li>\u8a2d\u8a08\u66f8\u304b\u3089\u30b3\u30fc\u30c9\u751f\u6210\u3092\u3057\u305f\u3068\u304d<\/li>\n<\/ul>\n\n\n\n<p>\u3053\u306e\u5f8c\u3067\u3001\u5358\u4f53\u30c6\u30b9\u30c8\u306e\u30b3\u30fc\u30c9\u3092\u52d5\u304b\u3057\u3066\u300c\u65e2\u5b58\u306e\u30b3\u30fc\u30c9\u304c\u58ca\u308c\u3066\u3044\u306a\u3044\u4e8b\u300d\u3092\u78ba\u8a8d\u3059\u308c\u3070 ok \u3067\u3059\u3002<br>\u3042\u308b\u3044\u306f\u3001\u30ea\u30d5\u30a1\u30af\u30bf\u30ea\u30f3\u30b0\u76ee\u7684\u3084\u975e\u4e92\u63db\u306e\u30b3\u30fc\u30c9\u3092\u542b\u3081\u3088\u3046\u3068\u3057\u305f\u3068\u304d\u306f\u3001\u518d\u3073\u52d5\u4f5c\u78ba\u8a8d\u7528\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306b AI \u304c\u624b\u3092\u5165\u308c\u308b\u3001\u3068\u3044\u3046\u3053\u3068\u306b\u306a\u308a\u307e\u3059\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>mock \u3092\u4f7f\u3063\u305f\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u7d44\u307f\u3053\u3080<\/strong><\/h2>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30c1\u30e3\u30c3\u30c8\u306e\u4fdd\u6301 ChatRepositoryImpl<\/li>\n\n\n\n<li>\u304a\u6c17\u306b\u5165\u308a\u306e\u4fdd\u6301 FavoriteRepository<\/li>\n<\/ul>\n\n\n\n<p>\u3053\u308c\u3089\u306f\u3001\u5185\u90e8\u30c7\u30fc\u30bf\u306b BleChatService \u3084 SharedPreferences \u3092\u4f7f\u3063\u3066\u3044\u308b\u305f\u3081\u3001\u5358\u4f53\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3067\u306f\u52d5\u304b\u305b\u307e\u305b\u3093\u3002\u3053\u308c\u3089\u306b\u5bfe\u3057\u3066\u306f\u30e2\u30c3\u30af\u5316\u3092\u3057\u3066\u3001\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u52d5\u304b\u305b\u308b\u3088\u3046\u306b\u3057\u307e\u3059\u3002<br>SharedPreferences \u306b\u3088\u308b\u6c38\u7d9a\u5316\u306b\u95a2\u3057\u3066\u306f androidTest \u3067\u52d5\u304b\u3059\u307b\u3046\u304c\u3044\u3044\u306e\u3067\u3059\u304c\u3001\u3044\u3063\u305f\u3093\u3001\u3053\u3053\u306f\u30e2\u30c3\u30af\u5316\u306b\u3057\u3066\u304a\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">\u30e2\u30c3\u30af\u5316\u3092\u4f7f\u3063\u3066\u4ee5\u4e0b\u306e\u30af\u30e9\u30b9\u306e\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u8ffd\u52a0<br><br>- ChatRepositoryImpl<br>- FavoriteRepository<\/pre>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-49.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"721\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-49-1024x721.png\" alt=\"\" class=\"wp-image-12179\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-49-1024x721.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-49-300x211.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-49-768x540.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-49.png 1198w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u9014\u4e2d\u304c\u3088\u304f\u308f\u304b\u3089\u306a\u3044\u306e\u3067\u3059\u304c\u3001\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u4f5c\u3063\u305f\u5f8c\u306b\u5931\u6557\u3057\u3066\u30b3\u30fc\u30c9\u3092\u3084\u308a\u76f4\u3057\u3057\u3066\u3044\u305f\u308a\u3057\u307e\u3059\u306d\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><a href=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-50.png\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"721\" src=\"https:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-50-1024x721.png\" alt=\"\" class=\"wp-image-12180\" srcset=\"http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-50-1024x721.png 1024w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-50-300x211.png 300w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-50-768x540.png 768w, http:\/\/www.moonmile.net\/blog\/wp-content\/uploads\/2026\/04\/image-50.png 1198w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/a><\/figure>\n\n\n\n<p>\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306e\u5185\u5bb9\u306f\u3055\u3066\u304a\u304d\u3001\u304a\u304a\u307e\u304b\u306a\u30af\u30e9\u30b9\u3084\u30e1\u30bd\u30c3\u30c9\u3092\u56fa\u5b9a\u5316\u3059\u308b\u3053\u3068\u304c\u3067\u304d\u307e\u3057\u305f\u3002\u3053\u308c\u3067\u3001\u6b21\u56de\u4ee5\u964d\u30b3\u30fc\u30c9\u306b\u4f55\u304b\u3092\u4ed8\u3051\u52a0\u3048\u308b\u5834\u5408\u306f\u3001\u300c\u65e2\u5b58\u306e\u30a4\u30f3\u30bf\u30fc\u30d5\u30a7\u30fc\u30b9\u3092\u5909\u3048\u306a\u3044\u3088\u3046\u306b\u614e\u91cd\u306b\u30b3\u30fc\u30c9\u3092\u5909\u66f4\u3059\u308b\u304b\u300d\u3042\u308b\u3044\u306f\u300c\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306e\u4fee\u6b63\u3082\u542b\u3081\u3066\u4ee3\u66ff\u306b\u8a2d\u8a08\/\u30b3\u30fc\u30c9\u5909\u66f4\u3092\u884c\u3046\u304b\u300d\u3092\u9078\u3076\u3053\u3068\u304c\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<p>\u5927\u80c6\u306a\u5909\u66f4\u3068\u3057\u3066\u306f\u3001<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>ChatMessage \u30c7\u30fc\u30bf\u30af\u30e9\u30b9\u3092\u5909\u3048\u3066 BLE \u30a2\u30c9\u30d0\u30bf\u30a4\u30ba\u306e\u30c7\u30fc\u30bf\u5f62\u5f0f\u3092\u5909\u66f4\u3059\u308b<\/li>\n\n\n\n<li>ChatRepository \u3092\u8907\u6570\u6301\u3066\u308b\u3088\u3046\u306b\u3057\u3066\u3001\u30eb\u30fc\u30e0\u306e\u5909\u66f4\u306a\u3069\u306b\u5bfe\u5fdc\u3059\u308b<\/li>\n\n\n\n<li>\u30c1\u30e3\u30c3\u30c8\u306e\u914d\u4fe1\u6a5f\u80fd\u3092\u3001\u6587\u5b57\u3067\u306f\u306a\u304f\u8272\u3084\u56f3\u5f62\u306a\u3069\u306b\u56fa\u5b9a\u3057\u3066\u3001\u30c1\u30e3\u30c3\u30c8\u753b\u9762\u306e UI \u3092\u5927\u5e45\u306b\u5909\u66f4\u3059\u308b<\/li>\n<\/ul>\n\n\n\n<p>\u306e\u3088\u3046\u306a\u5f62\u3067\u3059\u3002<\/p>\n\n\n\n<p>\u3053\u3053\u307e\u3067\u6765\u308b\u3068\u3001\u4eba\u9593\u304c\u958b\u767a\u3059\u308b\u5834\u5408\u306f\u300c\u65b0\u3057\u304f\u4f5c\u308a\u76f4\u3057\u305f\u307b\u3046\u304c\u624b\u3063\u53d6\u308a\u65e9\u3044\u300d\u3068\u3044\u3046\u5f62\u306b\u306a\u308a\u305d\u3046\u306a\u306e\u3067\u3059\u304c\u3001\u679c\u305f\u3057\u3066 AI \u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3092\u3059\u308b\u5834\u5408\u306b\u306f\u3069\u3046\u3059\u308c\u3044\u3044\u3067\u3057\u3087\u3046\u304b\uff1f\u3000\u3068\u3044\u3046\u8a71\u3067\u3059\u306d\u3002FolkBears \u306e\u5834\u5408\u306f\u3001BLE \u306e\u901a\u4fe1\u90e8\u5206\uff08GATT, Advertiser, Scanner\uff09\u3092\u30e9\u30a4\u30d6\u30e9\u30ea\u5316\u3057\u3066\u3001\u305d\u308c\u3092\u6d41\u7528\u3059\u308b\u8a2d\u8a08\u306b\u3057\u305f\u306e\u3067\u3059\u304c\u3001\u3053\u306e BLE \u30c1\u30e3\u30c3\u30c8\u30c4\u30fc\u30eb\u306e\u5834\u5408\u306f\u3069\u3046\u306a\u308b\u306e\u304b\uff1f<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u30b3\u30fc\u30c9<\/strong><\/h2>\n\n\n\n<p><a href=\"https:\/\/github.com\/moonmile\/BLE-chat\">https:\/\/github.com\/moonmile\/BLE-chat<\/a><\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>branch: dev\/test-network-cluade : \u5358\u4f53\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u306e\u8ffd\u52a0<\/li>\n<\/ul>\n","protected":false},"excerpt":{"rendered":"<p>BLE \u3092\u4f7f\u3063\u305f Android \u5b9f\u6a5f\u306e\u81ea\u52d5\u30c6\u30b9\u30c8\u30b3\u30fc\u30c9\u3092\u8003\u3048\u307e\u3059\u3002AI \u3092\u542b\u3081\u305f\u30b9\u30d1\u30a4\u30e9\u30eb\u958b\u767a\u3067\u306f\u3001Web API \u3092\u4f7f\u3063\u305f\u307b\u3046\u304c\u5b9f\u7528\u7684\u306a\u3088\u3046\u306a\u6c17\u304c\u3059\u308b\u306e\u3067\u3059\u304c&#8230;\u3072\u3068\u307e\u305a BLE \u306e\u307b\u3046\u304b\u3089\u3084\u3063\u3066\u307f\u307e\u3059\u3002W &hellip; <a href=\"http:\/\/www.moonmile.net\/blog\/archives\/12172\">\u7d9a\u304d\u3092\u8aad\u3080 <span class=\"meta-nav\">&rarr;<\/span><\/a><\/p>\n","protected":false},"author":2,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_jetpack_memberships_contains_paid_content":false,"footnotes":""},"categories":[3],"tags":[],"class_list":["post-12172","post","type-post","status-publish","format-standard","hentry","category-dev"],"jetpack_featured_media_url":"","jetpack_sharing_enabled":true,"_links":{"self":[{"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/posts\/12172","targetHints":{"allow":["GET"]}}],"collection":[{"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/users\/2"}],"replies":[{"embeddable":true,"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/comments?post=12172"}],"version-history":[{"count":1,"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/posts\/12172\/revisions"}],"predecessor-version":[{"id":12181,"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/posts\/12172\/revisions\/12181"}],"wp:attachment":[{"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/media?parent=12172"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/categories?post=12172"},{"taxonomy":"post_tag","embeddable":true,"href":"http:\/\/www.moonmile.net\/blog\/wp-json\/wp\/v2\/tags?post=12172"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}