この文書の現在のバージョンと選択したバージョンの差分を表示します。
両方とも前のリビジョン 前のリビジョン 次のリビジョン | 前のリビジョン | ||
android:androidstudio2で始めるアプリ開発入門 [2019/10/17 06:14] ips |
android:androidstudio2で始めるアプリ開発入門 [2019/12/01 09:11] (現在) ips |
||
---|---|---|---|
ライン 379: | ライン 379: | ||
+ | ==== No Channel found for ==== | ||
+ | [[https://qiita.com/b_a_a_d_o/items/83c01942a348efd551d4|Android O(APIバージョン26)のPush通知]] | ||
+ | |||
+ | Android O(APIバージョン26)以上ではNotifcationの使用には「通知チャネル」の登録が必要。 | ||
+ | |||
+ | <code> | ||
+ | E/NotificationService: No Channel found for pkg=com.nekotype.ips.servicesample, channelId=null, id=1, tag=null, opPkg=com.nekotype.ips.servicesample, callingUid=10091, userId=0, incomingUserId=0, notificationUid=10091, notification=Notification(channel=null pri=0 contentView=null vibrate=null sound=null defaults=0x0 flags=0x0 color=0x00000000 vis=PRIVATE) | ||
+ | </code> | ||
+ | |||
+ | <code kotlin> | ||
+ | _player.setOnCompletionListener { | ||
+ | Log.d("service","setOnCompletionListener is called") | ||
+ | |||
+ | val manager:NotificationManager=getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager | ||
+ | |||
+ | if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){ | ||
+ | |||
+ | val channel = NotificationChannel("channel_id_sample","プッシュ通知",NotificationManager.IMPORTANCE_DEFAULT) | ||
+ | channel.apply { | ||
+ | enableLights(true) // 通知時にライトを有効にする | ||
+ | lightColor = Color.WHITE // 通知時のライトの色 | ||
+ | lockscreenVisibility = Notification.VISIBILITY_PUBLIC //ロック画面での表示レベル | ||
+ | } | ||
+ | manager.createNotificationChannel(channel) | ||
+ | } | ||
+ | |||
+ | val builder=NotificationCompat.Builder(this) | ||
+ | builder.apply { | ||
+ | setSmallIcon(android.R.drawable.ic_dialog_info) | ||
+ | setContentTitle("再生終了") | ||
+ | setContentText("音声ファイルの再生が終了しました") | ||
+ | } | ||
+ | |||
+ | if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){ | ||
+ | builder.setChannelId("channel_id_sample"); | ||
+ | } | ||
+ | |||
+ | val notification = builder.build() | ||
+ | manager.notify(1, notification) | ||
+ | stopSelf() | ||
+ | } | ||
+ | |||
+ | </code> | ||
+ | |||
+ | {{:kotlin:pasted:20191018-062650.png}} | ||
+ | |||
+ | ==== startForegroundService ==== | ||
+ | |||
+ | [[https://qiita.com/gksdyd88/items/30df1f220001fad69d9e|Android Oreoでサービスを使ってみる]] | ||
+ | [[https://qiita.com/naoi/items/03e76d10948fe0d45597|Foreground Serviceの基本]] | ||
+ | [[https://qiita.com/nukka123/items/791bc4f9043764789ee6|Android Oからのバックグラウンド・サービスの制限事項を実演する。]] | ||
+ | [[https://www.muaaru.com/2018/11/17/post-254/|[Android]バックグラウンドでセンサーなどのログを取得し続けるには]] | ||
+ | |||
+ | [[https://developer.android.com/about/versions/oreo/background|バックグラウンド実行制限]] | ||
+ | [[https://developer.android.com/about/versions/oreo/background-location-limits|バックグラウンド位置情報の制限]] | ||
+ | |||
+ | Foreground(=前)Service(バックグラウンド)と思っていたので矛盾を感じていた。 | ||
+ | しかし、Foregroundはユーザーが確認できること。 | ||
+ | Serivceは画面がないことを意味する。 | ||
+ | つまり画面はないが、ユーザーが確認できるのがForegroundService。 | ||
+ | ユーザーは通知で動作を確認できるが画面がない。 | ||
+ | |||
+ | 設定ポイントは下記3点 | ||
+ | |||
+ | - startForegroundService(intent)<呼出側> | ||
+ | - 5秒以内にstartForeground(1, notification)<呼ばれ側> | ||
+ | - <uses-permission android:name="android.permission.FOREGROUND_SERVICE" /> 追加 | ||
+ | |||
+ | 下記エラーは3つ目のuses-permissionの追加で解消した。 | ||
+ | <code> | ||
+ | 2019-10-18 23:51:22.643 4837-4837/com.nekotype.ips.servicesample E/AndroidRuntime: FATAL EXCEPTION: main | ||
+ | Process: com.nekotype.ips.servicesample, PID: 4837 | ||
+ | java.lang.SecurityException: Permission Denial: startForeground from pid=4837, uid=10091 requires android.permission.FOREGROUND_SERVICE | ||
+ | </code> | ||
+ | |||
+ | ===== 第14回.Android地図アプリとの連携とGPS機能の利用 ===== | ||
+ | |||
+ | ==== Locationのパーミッション ==== | ||
+ | |||
+ | [[https://developer.android.com/guide/topics/security/permissions.html?hl=ja|システム パーミッション]] | ||
+ | |||
+ | <code xml> | ||
+ | <!-- GPSから取得 --> | ||
+ | <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /> | ||
+ | |||
+ | <!-- Networkから取得 --> | ||
+ | <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" /> | ||
+ | </code> | ||
+ | GPS:高精度な位置情報を取得できるが、電力消費が激しい。屋内では使えない。 | ||
+ | Network:位置情報の精度は落ちるが、電力消費が少ない。屋内でも取得できる | ||
+ | |||
+ | ==== Parameter 'xxxx' is never used ==== | ||
+ | |||
+ | {{:kotlin:pasted:20191019-105034.png}} | ||
+ | ↓ | ||
+ | <code kotlin> | ||
+ | fun onMapShowCurrentButtonClick(@Suppress("UNUSED_PARAMETER")vieie: View){ | ||
+ | val uriStr="geo:${latitude},${longitude}" | ||
+ | val uri=Uri.parse(uriStr) | ||
+ | val intent = Intent(Intent.ACTION_VIEW,uri) | ||
+ | startActivity(intent) | ||
+ | } | ||
+ | </code> | ||
+ | |||
+ | ==== ユーザー許可を取得する ==== | ||
+ | |||
+ | <code> | ||
+ | override fun onCreate(savedInstanceState: Bundle?) { | ||
+ | super.onCreate(savedInstanceState) | ||
+ | setContentView(R.layout.activity_main) | ||
+ | |||
+ | val locationManager=getSystemService(Context.LOCATION_SERVICE)as LocationManager | ||
+ | |||
+ | // 位置情報のユーザー許可を確認 | ||
+ | if (ActivityCompat.checkSelfPermission(this,Manifest.permission.ACCESS_FINE_LOCATION ) | ||
+ | != PackageManager.PERMISSION_GRANTED | ||
+ | ) { | ||
+ | // 許可がなければ許可を求める | ||
+ | val permissions = arrayOf<String>(Manifest.permission.ACCESS_FINE_LOCATION) | ||
+ | ActivityCompat.requestPermissions(this, permissions, 1000) | ||
+ | // 求めた結果はonRequestPermissionsResult | ||
+ | return | ||
+ | } | ||
+ | |||
+ | locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,0L,0F,GPSLocationListener()) | ||
+ | |||
+ | } | ||
+ | |||
+ | // パーミッション許可を求めた結果 | ||
+ | override fun onRequestPermissionsResult( | ||
+ | requestCode: Int, | ||
+ | permissions: Array<out String>, | ||
+ | grantResults: IntArray | ||
+ | ) { | ||
+ | super.onRequestPermissionsResult(requestCode, permissions, grantResults) | ||
+ | |||
+ | if (requestCode==1000 && grantResults[0]==PackageManager.PERMISSION_GRANTED){ | ||
+ | |||
+ | //パーミッションが許可された場合 | ||
+ | val locationManager = getSystemService(Context.LOCATION_SERVICE) as LocationManager | ||
+ | val locationListener = GPSLocationListener() | ||
+ | |||
+ | // パーミッション許可を求めた結果を確認しているので不要に思えるが、 | ||
+ | // ↓ locationManager.requestLocationUpdates を呼ぶ前にかならず必要なだけ | ||
+ | if (ActivityCompat.checkSelfPermission(this,Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED){ | ||
+ | return | ||
+ | } | ||
+ | locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,0L,0F,locationListener) | ||
+ | |||
+ | } | ||
+ | |||
+ | } | ||
+ | |||
+ | </code> | ||
===== 第16回.Androidのマテリアルデザイン ~マテリアルデザインとツールバー~ ===== | ===== 第16回.Androidのマテリアルデザイン ~マテリアルデザインとツールバー~ ===== | ||