From 1fea1f7a0d7226daac08bce578aa9e12cad2f72b Mon Sep 17 00:00:00 2001 From: Chang YiYu Date: Wed, 6 May 2015 18:10:47 +0800 Subject: [PATCH 01/14] Add --- examples/0601/MyAndroidTutorial/.gitignore | 6 + examples/0601/MyAndroidTutorial/.idea/.name | 1 + .../0601/MyAndroidTutorial/.idea/compiler.xml | 22 ++ .../.idea/copyright/profiles_settings.xml | 3 + .../0601/MyAndroidTutorial/.idea/gradle.xml | 19 + .../0601/MyAndroidTutorial/.idea/misc.xml | 22 ++ .../0601/MyAndroidTutorial/.idea/modules.xml | 9 + examples/0601/MyAndroidTutorial/.idea/vcs.xml | 6 + .../MyAndroidTutorial/MyAndroidTutorial.iml | 19 + .../0601/MyAndroidTutorial/app/.gitignore | 1 + examples/0601/MyAndroidTutorial/app/app.iml | 113 ++++++ .../0601/MyAndroidTutorial/app/build.gradle | 26 ++ .../MyAndroidTutorial/app/proguard-rules.pro | 17 + .../myandroidtutorial/ApplicationTest.java | 13 + .../src/debug/res/values/google_maps_api.xml | 18 + .../app/src/main/AndroidManifest.xml | 128 ++++++ .../myandroidtutorial/AboutActivity.java | 24 ++ .../myandroidtutorial/AlarmReceiver.java | 86 ++++ .../myandroidtutorial/ColorActivity.java | 75 ++++ .../net/macdidi/myandroidtutorial/Colors.java | 24 ++ .../macdidi/myandroidtutorial/FileUtil.java | 112 ++++++ .../myandroidtutorial/InitAlarmReceiver.java | 49 +++ .../net/macdidi/myandroidtutorial/Item.java | 156 ++++++++ .../myandroidtutorial/ItemActivity.java | 370 ++++++++++++++++++ .../myandroidtutorial/ItemAdapter.java | 80 ++++ .../myandroidtutorial/ItemAppWidget.java | 72 ++++ .../ItemAppWidgetConfigureActivity.java | 120 ++++++ .../macdidi/myandroidtutorial/ItemDAO.java | 214 ++++++++++ .../myandroidtutorial/MainActivity.java | 351 +++++++++++++++++ .../myandroidtutorial/MapsActivity.java | 338 ++++++++++++++++ .../macdidi/myandroidtutorial/MyDBHelper.java | 47 +++ .../myandroidtutorial/PlayActivity.java | 64 +++ .../myandroidtutorial/PrefActivity.java | 38 ++ .../myandroidtutorial/RecordActivity.java | 180 +++++++++ .../main/res/drawable-hdpi/ic_launcher.png | Bin 0 -> 9397 bytes .../main/res/drawable-mdpi/ic_launcher.png | Bin 0 -> 5237 bytes .../example_appwidget_preview.png | Bin 0 -> 3522 bytes .../main/res/drawable-xhdpi/ic_launcher.png | Bin 0 -> 14383 bytes .../main/res/drawable-xxhdpi/ic_launcher.png | Bin 0 -> 19388 bytes .../app/src/main/res/drawable/alarm_icon.png | Bin 0 -> 1605 bytes .../src/main/res/drawable/item_drawable.xml | 13 + .../src/main/res/drawable/location_icon.png | Bin 0 -> 1496 bytes .../app/src/main/res/drawable/pause_icon.png | Bin 0 -> 1107 bytes .../app/src/main/res/drawable/play_icon.png | Bin 0 -> 1248 bytes .../main/res/drawable/record_dark_icon.png | Bin 0 -> 1319 bytes .../src/main/res/drawable/record_red_icon.png | Bin 0 -> 3361 bytes .../main/res/drawable/record_sound_icon.png | Bin 0 -> 1318 bytes .../main/res/drawable/retangle_drawable.xml | 13 + .../main/res/drawable/select_color_icon.png | Bin 0 -> 1455 bytes .../src/main/res/drawable/selected_icon.png | Bin 0 -> 1546 bytes .../app/src/main/res/drawable/stop_icon.png | Bin 0 -> 1047 bytes .../main/res/drawable/take_picture_icon.png | Bin 0 -> 1460 bytes .../src/main/res/layout/activity_about.xml | 36 ++ .../src/main/res/layout/activity_color.xml | 16 + .../app/src/main/res/layout/activity_item.xml | 125 ++++++ .../app/src/main/res/layout/activity_main.xml | 30 ++ .../app/src/main/res/layout/activity_maps.xml | 7 + .../app/src/main/res/layout/activity_play.xml | 38 ++ .../src/main/res/layout/activity_record.xml | 51 +++ .../src/main/res/layout/item_app_widget.xml | 22 ++ .../app/src/main/res/layout/single_item.xml | 50 +++ .../app/src/main/res/menu/menu_main.xml | 56 +++ .../res/transition/explode_transition.xml | 8 + .../main/res/transition/fade_transition.xml | 8 + .../main/res/transition/slide_transition.xml | 8 + .../app/src/main/res/values-en/strings.xml | 12 + .../app/src/main/res/values-v14/dimens.xml | 10 + .../app/src/main/res/values-v21/styles.xml | 12 + .../app/src/main/res/values-w820dp/dimens.xml | 6 + .../app/src/main/res/values/colors.xml | 7 + .../app/src/main/res/values/dimens.xml | 15 + .../app/src/main/res/values/strings.xml | 59 +++ .../app/src/main/res/values/styles.xml | 8 + .../src/main/res/xml/item_app_widget_info.xml | 15 + .../app/src/main/res/xml/mypreference.xml | 21 + .../release/res/values/google_maps_api.xml | 18 + examples/0601/MyAndroidTutorial/build.gradle | 19 + .../0601/MyAndroidTutorial/gradle.properties | 18 + .../gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 49896 bytes .../gradle/wrapper/gradle-wrapper.properties | 6 + examples/0601/MyAndroidTutorial/gradlew | 164 ++++++++ examples/0601/MyAndroidTutorial/gradlew.bat | 90 +++++ .../0601/MyAndroidTutorial/settings.gradle | 1 + examples/0602/MyAndroidTutorial/.gitignore | 6 + examples/0602/MyAndroidTutorial/.idea/.name | 1 + .../0602/MyAndroidTutorial/.idea/compiler.xml | 22 ++ .../.idea/copyright/profiles_settings.xml | 3 + .../0602/MyAndroidTutorial/.idea/gradle.xml | 19 + .../0602/MyAndroidTutorial/.idea/misc.xml | 22 ++ .../0602/MyAndroidTutorial/.idea/modules.xml | 9 + examples/0602/MyAndroidTutorial/.idea/vcs.xml | 6 + .../MyAndroidTutorial/MyAndroidTutorial.iml | 19 + .../0602/MyAndroidTutorial/app/.gitignore | 1 + examples/0602/MyAndroidTutorial/app/app.iml | 114 ++++++ .../0602/MyAndroidTutorial/app/build.gradle | 27 ++ .../MyAndroidTutorial/app/proguard-rules.pro | 17 + .../myandroidtutorial/ApplicationTest.java | 13 + .../src/debug/res/values/google_maps_api.xml | 18 + .../app/src/main/AndroidManifest.xml | 128 ++++++ .../myandroidtutorial/AboutActivity.java | 24 ++ .../myandroidtutorial/AlarmReceiver.java | 86 ++++ .../myandroidtutorial/ColorActivity.java | 75 ++++ .../net/macdidi/myandroidtutorial/Colors.java | 24 ++ .../macdidi/myandroidtutorial/FileUtil.java | 112 ++++++ .../myandroidtutorial/InitAlarmReceiver.java | 49 +++ .../net/macdidi/myandroidtutorial/Item.java | 156 ++++++++ .../myandroidtutorial/ItemActivity.java | 370 ++++++++++++++++++ .../myandroidtutorial/ItemAdapter.java | 80 ++++ .../myandroidtutorial/ItemAdapterRV.java | 79 ++++ .../myandroidtutorial/ItemAppWidget.java | 72 ++++ .../ItemAppWidgetConfigureActivity.java | 120 ++++++ .../macdidi/myandroidtutorial/ItemDAO.java | 214 ++++++++++ .../myandroidtutorial/MainActivity.java | 333 ++++++++++++++++ .../myandroidtutorial/MapsActivity.java | 338 ++++++++++++++++ .../macdidi/myandroidtutorial/MyDBHelper.java | 47 +++ .../myandroidtutorial/PlayActivity.java | 64 +++ .../myandroidtutorial/PrefActivity.java | 38 ++ .../myandroidtutorial/RecordActivity.java | 180 +++++++++ .../main/res/drawable-hdpi/ic_launcher.png | Bin 0 -> 9397 bytes .../main/res/drawable-mdpi/ic_launcher.png | Bin 0 -> 5237 bytes .../example_appwidget_preview.png | Bin 0 -> 3522 bytes .../main/res/drawable-xhdpi/ic_launcher.png | Bin 0 -> 14383 bytes .../main/res/drawable-xxhdpi/ic_launcher.png | Bin 0 -> 19388 bytes .../app/src/main/res/drawable/alarm_icon.png | Bin 0 -> 1605 bytes .../src/main/res/drawable/item_drawable.xml | 13 + .../src/main/res/drawable/location_icon.png | Bin 0 -> 1496 bytes .../app/src/main/res/drawable/pause_icon.png | Bin 0 -> 1107 bytes .../app/src/main/res/drawable/play_icon.png | Bin 0 -> 1248 bytes .../main/res/drawable/record_dark_icon.png | Bin 0 -> 1319 bytes .../src/main/res/drawable/record_red_icon.png | Bin 0 -> 3361 bytes .../main/res/drawable/record_sound_icon.png | Bin 0 -> 1318 bytes .../main/res/drawable/retangle_drawable.xml | 13 + .../main/res/drawable/select_color_icon.png | Bin 0 -> 1455 bytes .../src/main/res/drawable/selected_icon.png | Bin 0 -> 1546 bytes .../app/src/main/res/drawable/stop_icon.png | Bin 0 -> 1047 bytes .../main/res/drawable/take_picture_icon.png | Bin 0 -> 1460 bytes .../src/main/res/layout/activity_about.xml | 36 ++ .../src/main/res/layout/activity_color.xml | 16 + .../app/src/main/res/layout/activity_item.xml | 125 ++++++ .../app/src/main/res/layout/activity_main.xml | 43 ++ .../app/src/main/res/layout/activity_maps.xml | 7 + .../app/src/main/res/layout/activity_play.xml | 38 ++ .../src/main/res/layout/activity_record.xml | 51 +++ .../src/main/res/layout/item_app_widget.xml | 22 ++ .../app/src/main/res/layout/single_item.xml | 61 +++ .../app/src/main/res/menu/menu_main.xml | 56 +++ .../res/transition/explode_transition.xml | 8 + .../main/res/transition/fade_transition.xml | 8 + .../main/res/transition/slide_transition.xml | 8 + .../app/src/main/res/values-en/strings.xml | 12 + .../app/src/main/res/values-v14/dimens.xml | 10 + .../app/src/main/res/values-v21/styles.xml | 12 + .../app/src/main/res/values-w820dp/dimens.xml | 6 + .../app/src/main/res/values/colors.xml | 7 + .../app/src/main/res/values/dimens.xml | 13 + .../app/src/main/res/values/strings.xml | 59 +++ .../app/src/main/res/values/styles.xml | 8 + .../src/main/res/xml/item_app_widget_info.xml | 15 + .../app/src/main/res/xml/mypreference.xml | 21 + .../release/res/values/google_maps_api.xml | 18 + examples/0602/MyAndroidTutorial/build.gradle | 19 + .../0602/MyAndroidTutorial/gradle.properties | 18 + .../gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 49896 bytes .../gradle/wrapper/gradle-wrapper.properties | 6 + examples/0602/MyAndroidTutorial/gradlew | 164 ++++++++ examples/0602/MyAndroidTutorial/gradlew.bat | 90 +++++ .../0602/MyAndroidTutorial/settings.gradle | 1 + 167 files changed, 7655 insertions(+) create mode 100644 examples/0601/MyAndroidTutorial/.gitignore create mode 100644 examples/0601/MyAndroidTutorial/.idea/.name create mode 100644 examples/0601/MyAndroidTutorial/.idea/compiler.xml create mode 100644 examples/0601/MyAndroidTutorial/.idea/copyright/profiles_settings.xml create mode 100644 examples/0601/MyAndroidTutorial/.idea/gradle.xml create mode 100644 examples/0601/MyAndroidTutorial/.idea/misc.xml create mode 100644 examples/0601/MyAndroidTutorial/.idea/modules.xml create mode 100644 examples/0601/MyAndroidTutorial/.idea/vcs.xml create mode 100644 examples/0601/MyAndroidTutorial/MyAndroidTutorial.iml create mode 100644 examples/0601/MyAndroidTutorial/app/.gitignore create mode 100644 examples/0601/MyAndroidTutorial/app/app.iml create mode 100644 examples/0601/MyAndroidTutorial/app/build.gradle create mode 100644 examples/0601/MyAndroidTutorial/app/proguard-rules.pro create mode 100644 examples/0601/MyAndroidTutorial/app/src/androidTest/java/net/macdidi/myandroidtutorial/ApplicationTest.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/debug/res/values/google_maps_api.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/AndroidManifest.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AboutActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AlarmReceiver.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ColorActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Colors.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/FileUtil.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/InitAlarmReceiver.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Item.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAdapter.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidget.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidgetConfigureActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemDAO.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MainActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MapsActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MyDBHelper.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PlayActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PrefActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/RecordActivity.java create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable-hdpi/ic_launcher.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable-mdpi/ic_launcher.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable-nodpi/example_appwidget_preview.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable-xhdpi/ic_launcher.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable-xxhdpi/ic_launcher.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/alarm_icon.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/item_drawable.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/location_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/pause_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/play_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_dark_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_red_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_sound_icon.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/retangle_drawable.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/select_color_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/selected_icon.png create mode 100755 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/stop_icon.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/drawable/take_picture_icon.png create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_about.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_color.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_item.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_main.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_maps.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_play.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_record.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/item_app_widget.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/layout/single_item.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/menu/menu_main.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/transition/explode_transition.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/transition/fade_transition.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/transition/slide_transition.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values-en/strings.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values-v14/dimens.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values-v21/styles.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values-w820dp/dimens.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values/colors.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values/dimens.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values/strings.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/values/styles.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/xml/item_app_widget_info.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/main/res/xml/mypreference.xml create mode 100644 examples/0601/MyAndroidTutorial/app/src/release/res/values/google_maps_api.xml create mode 100644 examples/0601/MyAndroidTutorial/build.gradle create mode 100644 examples/0601/MyAndroidTutorial/gradle.properties create mode 100644 examples/0601/MyAndroidTutorial/gradle/wrapper/gradle-wrapper.jar create mode 100644 examples/0601/MyAndroidTutorial/gradle/wrapper/gradle-wrapper.properties create mode 100755 examples/0601/MyAndroidTutorial/gradlew create mode 100644 examples/0601/MyAndroidTutorial/gradlew.bat create mode 100644 examples/0601/MyAndroidTutorial/settings.gradle create mode 100644 examples/0602/MyAndroidTutorial/.gitignore create mode 100644 examples/0602/MyAndroidTutorial/.idea/.name create mode 100644 examples/0602/MyAndroidTutorial/.idea/compiler.xml create mode 100644 examples/0602/MyAndroidTutorial/.idea/copyright/profiles_settings.xml create mode 100644 examples/0602/MyAndroidTutorial/.idea/gradle.xml create mode 100644 examples/0602/MyAndroidTutorial/.idea/misc.xml create mode 100644 examples/0602/MyAndroidTutorial/.idea/modules.xml create mode 100644 examples/0602/MyAndroidTutorial/.idea/vcs.xml create mode 100644 examples/0602/MyAndroidTutorial/MyAndroidTutorial.iml create mode 100644 examples/0602/MyAndroidTutorial/app/.gitignore create mode 100644 examples/0602/MyAndroidTutorial/app/app.iml create mode 100644 examples/0602/MyAndroidTutorial/app/build.gradle create mode 100644 examples/0602/MyAndroidTutorial/app/proguard-rules.pro create mode 100644 examples/0602/MyAndroidTutorial/app/src/androidTest/java/net/macdidi/myandroidtutorial/ApplicationTest.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/debug/res/values/google_maps_api.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/AndroidManifest.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AboutActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AlarmReceiver.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ColorActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Colors.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/FileUtil.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/InitAlarmReceiver.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Item.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAdapter.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAdapterRV.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidget.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidgetConfigureActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemDAO.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MainActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MapsActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MyDBHelper.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PlayActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PrefActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/RecordActivity.java create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable-hdpi/ic_launcher.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable-mdpi/ic_launcher.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable-nodpi/example_appwidget_preview.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable-xhdpi/ic_launcher.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable-xxhdpi/ic_launcher.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/alarm_icon.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/item_drawable.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/location_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/pause_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/play_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/record_dark_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/record_red_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/record_sound_icon.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/retangle_drawable.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/select_color_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/selected_icon.png create mode 100755 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/stop_icon.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/drawable/take_picture_icon.png create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_about.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_color.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_item.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_main.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_maps.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_play.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/activity_record.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/item_app_widget.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/layout/single_item.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/menu/menu_main.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/transition/explode_transition.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/transition/fade_transition.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/transition/slide_transition.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values-en/strings.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values-v14/dimens.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values-v21/styles.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values-w820dp/dimens.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values/colors.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values/dimens.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values/strings.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/values/styles.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/xml/item_app_widget_info.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/main/res/xml/mypreference.xml create mode 100644 examples/0602/MyAndroidTutorial/app/src/release/res/values/google_maps_api.xml create mode 100644 examples/0602/MyAndroidTutorial/build.gradle create mode 100644 examples/0602/MyAndroidTutorial/gradle.properties create mode 100644 examples/0602/MyAndroidTutorial/gradle/wrapper/gradle-wrapper.jar create mode 100644 examples/0602/MyAndroidTutorial/gradle/wrapper/gradle-wrapper.properties create mode 100755 examples/0602/MyAndroidTutorial/gradlew create mode 100644 examples/0602/MyAndroidTutorial/gradlew.bat create mode 100644 examples/0602/MyAndroidTutorial/settings.gradle diff --git a/examples/0601/MyAndroidTutorial/.gitignore b/examples/0601/MyAndroidTutorial/.gitignore new file mode 100644 index 0000000..afbdab3 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.gitignore @@ -0,0 +1,6 @@ +.gradle +/local.properties +/.idea/workspace.xml +/.idea/libraries +.DS_Store +/build diff --git a/examples/0601/MyAndroidTutorial/.idea/.name b/examples/0601/MyAndroidTutorial/.idea/.name new file mode 100644 index 0000000..5bb7a85 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/.name @@ -0,0 +1 @@ +MyAndroidTutorial \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/.idea/compiler.xml b/examples/0601/MyAndroidTutorial/.idea/compiler.xml new file mode 100644 index 0000000..9a8b7e5 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/compiler.xml @@ -0,0 +1,22 @@ + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/.idea/copyright/profiles_settings.xml b/examples/0601/MyAndroidTutorial/.idea/copyright/profiles_settings.xml new file mode 100644 index 0000000..e7bedf3 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/copyright/profiles_settings.xml @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/.idea/gradle.xml b/examples/0601/MyAndroidTutorial/.idea/gradle.xml new file mode 100644 index 0000000..c595ad9 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/gradle.xml @@ -0,0 +1,19 @@ + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/.idea/misc.xml b/examples/0601/MyAndroidTutorial/.idea/misc.xml new file mode 100644 index 0000000..b0c0cbc --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/misc.xml @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/.idea/modules.xml b/examples/0601/MyAndroidTutorial/.idea/modules.xml new file mode 100644 index 0000000..5edca7e --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/modules.xml @@ -0,0 +1,9 @@ + + + + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/.idea/vcs.xml b/examples/0601/MyAndroidTutorial/.idea/vcs.xml new file mode 100644 index 0000000..6564d52 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/MyAndroidTutorial.iml b/examples/0601/MyAndroidTutorial/MyAndroidTutorial.iml new file mode 100644 index 0000000..62d899f --- /dev/null +++ b/examples/0601/MyAndroidTutorial/MyAndroidTutorial.iml @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/.gitignore b/examples/0601/MyAndroidTutorial/app/.gitignore new file mode 100644 index 0000000..796b96d --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/.gitignore @@ -0,0 +1 @@ +/build diff --git a/examples/0601/MyAndroidTutorial/app/app.iml b/examples/0601/MyAndroidTutorial/app/app.iml new file mode 100644 index 0000000..486024a --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/app.iml @@ -0,0 +1,113 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/build.gradle b/examples/0601/MyAndroidTutorial/app/build.gradle new file mode 100644 index 0000000..9423d38 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/build.gradle @@ -0,0 +1,26 @@ +apply plugin: 'com.android.application' + +android { + compileSdkVersion 21 + buildToolsVersion "21.1.2" + + defaultConfig { + applicationId "net.macdidi.myandroidtutorial" + minSdkVersion 16 + targetSdkVersion 21 + versionCode 1 + versionName "1.0" + } + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } +} + +dependencies { + compile fileTree(dir: 'libs', include: ['*.jar']) + compile 'com.android.support:appcompat-v7:22.0.0' + compile 'com.google.android.gms:play-services:7.0.0' +} diff --git a/examples/0601/MyAndroidTutorial/app/proguard-rules.pro b/examples/0601/MyAndroidTutorial/app/proguard-rules.pro new file mode 100644 index 0000000..b5fa7ec --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/proguard-rules.pro @@ -0,0 +1,17 @@ +# Add project specific ProGuard rules here. +# By default, the flags in this file are appended to flags specified +# in /Users/macdidi5/Library/Android/sdk/tools/proguard/proguard-android.txt +# You can edit the include path and order by changing the proguardFiles +# directive in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# Add any project specific keep options here: + +# 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 *; +#} diff --git a/examples/0601/MyAndroidTutorial/app/src/androidTest/java/net/macdidi/myandroidtutorial/ApplicationTest.java b/examples/0601/MyAndroidTutorial/app/src/androidTest/java/net/macdidi/myandroidtutorial/ApplicationTest.java new file mode 100644 index 0000000..2cb214e --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/androidTest/java/net/macdidi/myandroidtutorial/ApplicationTest.java @@ -0,0 +1,13 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Application; +import android.test.ApplicationTestCase; + +/** + * Testing Fundamentals + */ +public class ApplicationTest extends ApplicationTestCase { + public ApplicationTest() { + super(Application.class); + } +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/debug/res/values/google_maps_api.xml b/examples/0601/MyAndroidTutorial/app/src/debug/res/values/google_maps_api.xml new file mode 100644 index 0000000..7341c8d --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/debug/res/values/google_maps_api.xml @@ -0,0 +1,18 @@ + + + + AIzaSyCZg9YWlfokPA96VxWGYr6u4C12jL16VhM + + diff --git a/examples/0601/MyAndroidTutorial/app/src/main/AndroidManifest.xml b/examples/0601/MyAndroidTutorial/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..50449c2 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/AndroidManifest.xml @@ -0,0 +1,128 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AboutActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AboutActivity.java new file mode 100644 index 0000000..42dddeb --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AboutActivity.java @@ -0,0 +1,24 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.os.Bundle; +import android.view.View; +import android.view.Window; + +public class AboutActivity extends Activity { + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + // 取消元件的應用程式標題 + requestWindowFeature(Window.FEATURE_NO_TITLE); + setContentView(R.layout.activity_about); + } + + // 結束按鈕 + public void clickOk(View view) { + // 呼叫這個方法結束Activity元件 + finish(); + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AlarmReceiver.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AlarmReceiver.java new file mode 100644 index 0000000..368165c --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/AlarmReceiver.java @@ -0,0 +1,86 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Notification; +import android.app.NotificationManager; +import android.content.BroadcastReceiver; +import android.content.Context; +import android.content.Intent; +import android.graphics.Bitmap; +import android.graphics.BitmapFactory; +import android.support.v4.app.NotificationCompat; + +import java.io.File; + +public class AlarmReceiver extends BroadcastReceiver { + + @Override + public void onReceive(Context context, Intent intent) { + // 讀取記事標題 + //String title = intent.getStringExtra("title"); + // 顯示訊息框 + //Toast.makeText(context, title, Toast.LENGTH_LONG).show(); + + // 讀取記事編號 + long id = intent.getLongExtra("id", 0); + + if (id != 0) { + sendNotify(context, id); + } + } + + private void sendNotify(Context context, long id) { + // 建立資料庫物件 + ItemDAO itemDAO = new ItemDAO(context.getApplicationContext()); + // 讀取指定編號的記事物件 + Item item = itemDAO.get(id); + + // 建立照片檔案物件 + File file = new File(FileUtil.getExternalStorageDir(FileUtil.APP_DIR), + "P" + item.getFileName() + ".jpg"); + + // 是否儲存照片檔案 + boolean isPicture = (item.getFileName() != null && + item.getFileName().length() > 0 && + file.exists()); + + // 取得NotificationManager物件 + NotificationManager nm = (NotificationManager) + context.getSystemService(Context.NOTIFICATION_SERVICE); + + // 如果有儲存照片檔案 + if (isPicture) { + // 建立Notification.Builder物件,因為要設定大型圖片樣式 + // 所以不能使用NotificationCompat.Builder + Notification.Builder builder = new Notification.Builder(context); + builder.setSmallIcon(android.R.drawable.star_on) + .setWhen(System.currentTimeMillis()) + .setContentTitle(context.getString(R.string.app_name)); + + // 建立大型圖片樣式物件 + Notification.BigPictureStyle bigPictureStyle = + new Notification.BigPictureStyle(); + // 設定圖片與簡介 + Bitmap bitmap = BitmapFactory.decodeFile(file.getAbsolutePath()); + bigPictureStyle.bigPicture(bitmap) + .setSummaryText(item.getTitle()); + // 設定樣式為大型圖片 + builder.setStyle(bigPictureStyle); + // 發出通知 + nm.notify((int)item.getId(), builder.build()); + } + // 如果沒有儲存照片檔案 + else { + // 建立NotificationCompat.Builder物件 + NotificationCompat.Builder builder = + new NotificationCompat.Builder(context); + // 設定圖示、時間、內容標題和內容訊息 + builder.setSmallIcon(android.R.drawable.star_big_on) + .setWhen(System.currentTimeMillis()) + .setContentTitle(context.getString(R.string.app_name)) + .setContentText(item.getTitle()); + // 發出通知 + nm.notify((int)item.getId(), builder.build()); + } + } + +} diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ColorActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ColorActivity.java new file mode 100644 index 0000000..182cd55 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ColorActivity.java @@ -0,0 +1,75 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.content.Intent; +import android.content.SharedPreferences; +import android.os.Bundle; +import android.preference.PreferenceManager; +import android.view.View; +import android.view.View.OnClickListener; +import android.widget.Button; +import android.widget.LinearLayout; + +public class ColorActivity extends Activity { + + private LinearLayout color_gallery; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_color); + + processViews(); + + ColorListener listener = new ColorListener(); + + for (Colors c : Colors.values()) { + Button button = new Button(this); + button.setId(c.parseColor()); + LinearLayout.LayoutParams layout = + new LinearLayout.LayoutParams(128, 128); + layout.setMargins(6, 6, 6, 6); + button.setLayoutParams(layout); + button.setBackgroundColor(c.parseColor()); + + button.setOnClickListener(listener); + + color_gallery.addView(button); + } + } + + private void processViews() { + color_gallery = (LinearLayout) findViewById(R.id.color_gallery); + } + + private class ColorListener implements OnClickListener { + + @Override + public void onClick(View view) { + String action = ColorActivity.this.getIntent().getAction(); + + // 經由設定元件啟動 + if (action != null && + action.equals("net.macdidi.myandroidtutorial.CHOOSE_COLOR")) { + // 建立SharedPreferences物件 + SharedPreferences.Editor editor = + PreferenceManager.getDefaultSharedPreferences( + ColorActivity.this).edit(); + // 儲存預設顏色 + editor.putInt("DEFAULT_COLOR", view.getId()); + // 寫入設定值 + editor.commit(); + finish(); + } + // 經由新增或修改記事的元件啟動 + else { + Intent result = getIntent(); + result.putExtra("colorId", view.getId()); + setResult(Activity.RESULT_OK, result); + finish(); + } + } + + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Colors.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Colors.java new file mode 100644 index 0000000..1462149 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Colors.java @@ -0,0 +1,24 @@ +package net.macdidi.myandroidtutorial; + +import android.graphics.Color; + +public enum Colors { + + LIGHTGREY("#D3D3D3"), BLUE("#33B5E5"), PURPLE("#AA66CC"), + GREEN("#99CC00"), ORANGE("#FFBB33"), RED("#FF4444"); + + private String code; + + private Colors(String code) { + this.code = code; + } + + public String getCode() { + return code; + } + + public int parseColor() { + return Color.parseColor(code); + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/FileUtil.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/FileUtil.java new file mode 100644 index 0000000..1fb41be --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/FileUtil.java @@ -0,0 +1,112 @@ +package net.macdidi.myandroidtutorial; + +import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.BitmapFactory; +import android.os.Environment; +import android.util.Log; +import android.widget.ImageView; + +import java.io.File; +import java.text.SimpleDateFormat; +import java.util.Date; + +public class FileUtil { + + // 應用程式儲存檔案的目錄 + public static final String APP_DIR = "androidtutorial"; + + // 外部儲存設備是否可寫入 + public static boolean isExternalStorageWritable() { + // 取得目前外部儲存設備的狀態 + String state = Environment.getExternalStorageState(); + + // 判斷是否可寫入 + if (Environment.MEDIA_MOUNTED.equals(state)) { + return true; + } + + return false; + } + + // 外部儲存設備是否可讀取 + public static boolean isExternalStorageReadable() { + // 取得目前外部儲存設備的狀態 + String state = Environment.getExternalStorageState(); + + // 判斷是否可讀取 + if (Environment.MEDIA_MOUNTED.equals(state) || + Environment.MEDIA_MOUNTED_READ_ONLY.equals(state)) { + return true; + } + + return false; + } + + // 建立並傳回在公用相簿下參數指定的路徑 + public static File getPublicAlbumStorageDir(String albumName) { + // 取得公用的照片路徑 + File pictures = Environment.getExternalStoragePublicDirectory( + Environment.DIRECTORY_PICTURES); + // 準備在照片路徑下建立一個指定的路徑 + File file = new File(pictures, albumName); + + // 如果建立路徑不成功 + if (!file.mkdirs()) { + Log.e("getAlbumStorageDir", "Directory not created"); + } + + return file; + } + + // 建立並傳回在應用程式專用相簿下參數指定的路徑 + public static File getAlbumStorageDir(Context context, String albumName) { + // 取得應用程式專用的照片路徑 + File pictures = context.getExternalFilesDir( + Environment.DIRECTORY_PICTURES); + // 準備在照片路徑下建立一個指定的路徑 + File file = new File(pictures, albumName); + + // 如果建立路徑不成功 + if (!file.mkdirs()) { + Log.e("getAlbumStorageDir", "Directory not created"); + } + + return file; + } + + // 建立並傳回外部儲存媒體參數指定的路徑 + public static File getExternalStorageDir(String dir) { + File result = new File( + Environment.getExternalStorageDirectory(), dir); + + if (!isExternalStorageWritable()) { + return null; + } + + if (!result.exists() && !result.mkdirs()) { + return null; + } + + return result; + } + + // 讀取指定的照片檔案名稱設定給ImageView元件 + public static void fileToImageView(String fileName, ImageView imageView) { + if (new File(fileName).exists()) { + Bitmap bitmap = BitmapFactory.decodeFile(fileName); + imageView.setImageBitmap(bitmap); + } + else { + Log.e("fileToImageView", fileName + " not found."); + } + } + + // 產生唯一的檔案名稱 + public static String getUniqueFileName() { + // 使用年月日_時分秒格式為檔案名稱 + SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd_HHmmss"); + return sdf.format(new Date()); + } + +} diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/InitAlarmReceiver.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/InitAlarmReceiver.java new file mode 100644 index 0000000..959785f --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/InitAlarmReceiver.java @@ -0,0 +1,49 @@ +package net.macdidi.myandroidtutorial; + +import android.app.AlarmManager; +import android.app.PendingIntent; +import android.content.BroadcastReceiver; +import android.content.Context; +import android.content.Intent; + +import java.util.Calendar; +import java.util.List; + +public class InitAlarmReceiver extends BroadcastReceiver { + + @Override + public void onReceive(Context context, Intent intent) { + // 建立資料庫物件 + ItemDAO itemDAO = new ItemDAO(context.getApplicationContext()); + // 讀取資料庫所有記事資料 + List items = itemDAO.getAll(); + + // 讀取目前時間 + long current = Calendar.getInstance().getTimeInMillis(); + + AlarmManager am = (AlarmManager) + context.getSystemService(Context.ALARM_SERVICE); + + for (Item item : items) { + long alarm = item.getAlarmDatetime(); + + // 如果沒有設定提醒或是提醒已經過期 + if (alarm == 0 || alarm <= current) { + continue; + } + + // 設定提醒 + Intent alarmIntent = new Intent(context, AlarmReceiver.class); + //alarmIntent.putExtra("title", item.getTitle()); + + // 加入記事編號 + intent.putExtra("id", item.getId()); + + PendingIntent pi = PendingIntent.getBroadcast( + context, (int)item.getId(), + alarmIntent, PendingIntent.FLAG_ONE_SHOT); + am.set(AlarmManager.RTC_WAKEUP, item.getAlarmDatetime(), pi); + } + } + +} diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Item.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Item.java new file mode 100644 index 0000000..e2bc93d --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/Item.java @@ -0,0 +1,156 @@ +package net.macdidi.myandroidtutorial; + +import java.util.Date; +import java.util.Locale; + +public class Item implements java.io.Serializable { + + // 編號、日期時間、顏色、標題、內容、照片檔案名稱、錄音檔案名稱、經緯度、修改、已選擇 + private long id; + private long datetime; + private Colors color; + private String title; + private String content; + private String fileName; + private String recFileName; + private double latitude; + private double longitude; + private long lastModify; + private boolean selected; + + // 提醒日期時間 + private long alarmDatetime; + + public Item() { + title = ""; + content = ""; + color = Colors.LIGHTGREY; + } + + public Item(long id, long datetime, Colors color, String title, + String content, String fileName, String recFileName, + double latitude, double longitude, long lastModify) { + this.id = id; + this.datetime = datetime; + this.color = color; + this.title = title; + this.content = content; + this.fileName = fileName; + this.recFileName = recFileName; + this.latitude = latitude; + this.longitude = longitude; + this.lastModify = lastModify; + } + + public long getId() { + return id; + } + + public void setId(long id) { + this.id = id; + } + + public long getDatetime() { + return datetime; + } + + // 裝置區域的日期時間 + public String getLocaleDatetime() { + return String.format(Locale.getDefault(), "%tF % 0) { + // 照片檔案物件 + File file = configFileName("P", ".jpg"); + + // 如果照片檔案存在 + if (file.exists()) { + // 顯示照片元件 + picture.setVisibility(View.VISIBLE); + // 設定照片 + FileUtil.fileToImageView(file.getAbsolutePath(), picture); + } + } + } + + @Override + protected void onActivityResult(int requestCode, int resultCode, Intent data) { + if (resultCode == Activity.RESULT_OK) { + switch (requestCode) { + // 照像 + case START_CAMERA: + // 設定照片檔案名稱 + item.setFileName(fileName); + break; + case START_RECORD: + // 設定錄音檔案名稱 + item.setRecFileName(fileName); + break; + case START_LOCATION: + // 讀取與設定座標 + double lat = data.getDoubleExtra("lat", 0.0); + double lng = data.getDoubleExtra("lng", 0.0); + item.setLatitude(lat); + item.setLongitude(lng); + break; + case START_ALARM: + break; + // 設定顏色 + case START_COLOR: + int colorId = data.getIntExtra( + "colorId", Colors.LIGHTGREY.parseColor()); + item.setColor(getColors(colorId)); + break; + } + } + } + + public static Colors getColors(int color) { + Colors result = Colors.LIGHTGREY; + + if (color == Colors.BLUE.parseColor()) { + result = Colors.BLUE; + } + else if (color == Colors.PURPLE.parseColor()) { + result = Colors.PURPLE; + } + else if (color == Colors.GREEN.parseColor()) { + result = Colors.GREEN; + } + else if (color == Colors.ORANGE.parseColor()) { + result = Colors.ORANGE; + } + else if (color == Colors.RED.parseColor()) { + result = Colors.RED; + } + + return result; + } + + private void processViews() { + title_text = (EditText) findViewById(R.id.title_text); + content_text = (EditText) findViewById(R.id.content_text); + // 取得顯示照片的ImageView元件 + picture = (ImageView) findViewById(R.id.picture); + } + + // 點擊確定與取消按鈕都會呼叫這個方法 + public void onSubmit(View view) { + + if (view.getId() == R.id.ok_teim) { + String titleText = title_text.getText().toString(); + String contentText = content_text.getText().toString(); + + item.setTitle(titleText); + item.setContent(contentText); + + if (getIntent().getAction().equals( + "net.macdidi.myandroidtutorial.EDIT_ITEM")) { + item.setLastModify(new Date().getTime()); + } + // 新增記事 + else { + item.setDatetime(new Date().getTime()); + } + + Intent result = getIntent(); + result.putExtra("net.macdidi.myandroidtutorial.Item", item); + setResult(Activity.RESULT_OK, result); + } + + // 結束 + finish(); + } + + public void clickFunction(View view) { + int id = view.getId(); + + switch (id) { + case R.id.take_picture: + // 啟動相機元件用的Intent物件 + Intent intentCamera = + new Intent(MediaStore.ACTION_IMAGE_CAPTURE); + + // 照片檔案名稱 + File pictureFile = configFileName("P", ".jpg"); + Uri uri = Uri.fromFile(pictureFile); + // 設定檔案名稱 + intentCamera.putExtra(MediaStore.EXTRA_OUTPUT, uri); + // 啟動相機元件 + startActivityForResult(intentCamera, START_CAMERA); + break; + case R.id.record_sound: + // 錄音檔案名稱 + final File recordFile = configRecFileName("R", ".mp3"); + + // 如果已經有錄音檔,詢問播放或重新錄製 + if (recordFile.exists()) { + // 詢問播放還是重新錄製的對話框 + AlertDialog.Builder d = new AlertDialog.Builder(this); + + d.setTitle(R.string.title_record) + .setCancelable(false); + d.setPositiveButton(R.string.record_play, + new DialogInterface.OnClickListener() { + public void onClick(DialogInterface dialog, int which) { + // 播放 + Intent playIntent = new Intent( + ItemActivity.this, PlayActivity.class); + playIntent.putExtra("fileName", + recordFile.getAbsolutePath()); + startActivity(playIntent); + } + }); + d.setNeutralButton(R.string.record_new, + new DialogInterface.OnClickListener() { + public void onClick(DialogInterface dialog, int which) { + goToRecord(recordFile); + } + }); + d.setNegativeButton(android.R.string.cancel, null); + + // 顯示對話框 + d.show(); + } + // 如果沒有錄音檔,啟動錄音元件 + else { + goToRecord(recordFile); + } + + break; + case R.id.set_location: + // 啟動地圖元件用的Intent物件 + Intent intentMap = new Intent(this, MapsActivity.class); + + // 設定儲存的座標 + intentMap.putExtra("lat", item.getLatitude()); + intentMap.putExtra("lng", item.getLongitude()); + intentMap.putExtra("title", item.getTitle()); + intentMap.putExtra("datetime", item.getLocaleDatetime()); + + // 啟動地圖元件 + startActivityForResult(intentMap, START_LOCATION); + break; + case R.id.set_alarm: + // 設定提醒日期時間 + processSetAlarm(); + break; + case R.id.select_color: + // 啟動設定顏色的Activity元件 + startActivityForResult( + new Intent(this, ColorActivity.class), START_COLOR); + break; + } + + } + + // 設定提醒日期時間 + private void processSetAlarm() { + Calendar calendar = Calendar.getInstance(); + + if (item.getAlarmDatetime() != 0) { + // 設定為已經儲存的提醒日期時間 + calendar.setTimeInMillis(item.getAlarmDatetime()); + } + + // 讀取年、月、日、時、分 + int year = calendar.get(Calendar.YEAR); + int month = calendar.get(Calendar.MONTH); + int day = calendar.get(Calendar.DAY_OF_MONTH); + int hour = calendar.get(Calendar.HOUR_OF_DAY); + int minute = calendar.get(Calendar.MINUTE); + + // 儲存設定的提醒日期時間 + final Calendar alarm = Calendar.getInstance(); + + // 設定提醒時間 + TimePickerDialog.OnTimeSetListener timeSetListener = + new TimePickerDialog.OnTimeSetListener() { + @Override + public void onTimeSet(TimePicker view, + int hourOfDay, int minute) { + alarm.set(Calendar.HOUR_OF_DAY, hourOfDay); + alarm.set(Calendar.MINUTE, minute); + + item.setAlarmDatetime(alarm.getTimeInMillis()); + } + }; + + // 選擇時間對話框 + final TimePickerDialog tpd = new TimePickerDialog( + this, timeSetListener, hour, minute, true); + + // 設定提醒日期 + DatePickerDialog.OnDateSetListener dateSetListener = + new DatePickerDialog.OnDateSetListener() { + @Override + public void onDateSet(DatePicker view, + int year, + int monthOfYear, + int dayOfMonth) { + alarm.set(Calendar.YEAR, year); + alarm.set(Calendar.MONTH, monthOfYear); + alarm.set(Calendar.DAY_OF_MONTH, dayOfMonth); + + // 繼續選擇提醒時間 + tpd.show(); + } + }; + + // 建立與顯示選擇日期對話框 + final DatePickerDialog dpd = new DatePickerDialog( + this, dateSetListener, year, month, day); + dpd.show(); + } + + private void goToRecord(File recordFile) { + // 錄音 + Intent recordIntent = new Intent(this, RecordActivity.class); + recordIntent.putExtra("fileName", recordFile.getAbsolutePath()); + startActivityForResult(recordIntent, START_RECORD); + } + + private File configFileName(String prefix, String extension) { + // 如果記事資料已經有檔案名稱 + if (item.getFileName() != null && item.getFileName().length() > 0) { + fileName = item.getFileName(); + } + // 產生檔案名稱 + else { + fileName = FileUtil.getUniqueFileName(); + } + + return new File(FileUtil.getExternalStorageDir(FileUtil.APP_DIR), + prefix + fileName + extension); + } + + private File configRecFileName(String prefix, String extension) { + // 如果記事資料已經有檔案名稱 + if (item.getRecFileName() != null && item.getRecFileName().length() > 0) { + recFileName = item.getRecFileName(); + } + // 產生檔案名稱 + else { + recFileName = FileUtil.getUniqueFileName(); + } + + return new File(FileUtil.getExternalStorageDir(FileUtil.APP_DIR), + prefix + recFileName + extension); + } + +} diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAdapter.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAdapter.java new file mode 100644 index 0000000..85b40ba --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAdapter.java @@ -0,0 +1,80 @@ +package net.macdidi.myandroidtutorial; + +import android.content.Context; +import android.graphics.drawable.GradientDrawable; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ArrayAdapter; +import android.widget.ImageView; +import android.widget.LinearLayout; +import android.widget.RelativeLayout; +import android.widget.TextView; + +import java.util.List; + +public class ItemAdapter extends ArrayAdapter { + + // 畫面資源編號 + private int resource; + // 包裝的記事資料 + private List items; + + public ItemAdapter(Context context, int resource, List items) { + super(context, resource, items); + this.resource = resource; + this.items = items; + } + + @Override + public View getView(int position, View convertView, ViewGroup parent) { + LinearLayout itemView; + // 讀取目前位置的記事物件 + final Item item = getItem(position); + + if (convertView == null) { + // 建立項目畫面元件 + itemView = new LinearLayout(getContext()); + String inflater = Context.LAYOUT_INFLATER_SERVICE; + LayoutInflater li = (LayoutInflater) + getContext().getSystemService(inflater); + li.inflate(resource, itemView, true); + } + else { + itemView = (LinearLayout) convertView; + } + + // 讀取記事顏色、已選擇、標題與日期時間元件 + RelativeLayout typeColor = (RelativeLayout) itemView.findViewById(R.id.type_color); + ImageView selectedItem = (ImageView) itemView.findViewById(R.id.selected_item); + TextView titleView = (TextView) itemView.findViewById(R.id.title_text); + TextView dateView = (TextView) itemView.findViewById(R.id.date_text); + + // 設定記事顏色 + GradientDrawable background = (GradientDrawable)typeColor.getBackground(); + background.setColor(item.getColor().parseColor()); + + // 設定標題與日期時間 + titleView.setText(item.getTitle()); + dateView.setText(item.getLocaleDatetime()); + + // 設定是否已選擇 + selectedItem.setVisibility(item.isSelected() ? View.VISIBLE : View.INVISIBLE); + + return itemView; + } + + // 設定指定編號的記事資料 + public void set(int index, Item item) { + if (index >= 0 && index < items.size()) { + items.set(index, item); + notifyDataSetChanged(); + } + } + + // 讀取指定編號的記事資料 + public Item get(int index) { + return items.get(index); + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidget.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidget.java new file mode 100644 index 0000000..9301f57 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidget.java @@ -0,0 +1,72 @@ +package net.macdidi.myandroidtutorial; + +import android.app.PendingIntent; +import android.appwidget.AppWidgetManager; +import android.appwidget.AppWidgetProvider; +import android.content.Context; +import android.content.Intent; +import android.widget.RemoteViews; + + +public class ItemAppWidget extends AppWidgetProvider { + + @Override + public void onUpdate(Context context, + AppWidgetManager appWidgetManager, + int[] appWidgetIds) { + final int N = appWidgetIds.length; + + for (int i = 0; i < N; i++) { + updateAppWidget(context, appWidgetManager, appWidgetIds[i]); + } + } + + @Override + public void onDeleted(Context context, int[] appWidgetIds) { + final int N = appWidgetIds.length; + for (int i = 0; i < N; i++) { + // 刪除小工具已經儲存的記事編號 + ItemAppWidgetConfigureActivity.deleteItemPref( + context, appWidgetIds[i]); + } + } + + @Override + public void onEnabled(Context context) { + + } + + @Override + public void onDisabled(Context context) { + + } + + static void updateAppWidget(Context context, + AppWidgetManager appWidgetManager, + int appWidgetId) { + // 讀取小工具儲存的記事編號 + long id = ItemAppWidgetConfigureActivity.loadItemPref( + context, appWidgetId); + // 建立小工具畫面元件 + RemoteViews views = new RemoteViews( + context.getPackageName(), R.layout.item_app_widget); + // 讀取指定編號的記事物件 + ItemDAO itemDAO = new ItemDAO(context.getApplicationContext()); + Item item = itemDAO.get(id); + + // 設定小工具畫面顯示記事標題 + views.setTextViewText(R.id.appwidget_text, + item != null ? item.getTitle() : "NA"); + + // 點選小工具畫面的記事標題後,啟動記事應用程式 + Intent intent = new Intent(context, MainActivity.class); + PendingIntent pending = PendingIntent.getActivity( + context, 0, intent, 0); + views.setOnClickPendingIntent(R.id.appwidget_text, pending); + + // 更新小工具 + appWidgetManager.updateAppWidget(appWidgetId, views); + } +} + + diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidgetConfigureActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidgetConfigureActivity.java new file mode 100644 index 0000000..f83be71 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemAppWidgetConfigureActivity.java @@ -0,0 +1,120 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.appwidget.AppWidgetManager; +import android.content.Context; +import android.content.Intent; +import android.content.SharedPreferences; +import android.os.Bundle; +import android.view.View; +import android.widget.AdapterView; +import android.widget.ListView; + +import java.util.List; + + +public class ItemAppWidgetConfigureActivity extends Activity { + + int mAppWidgetId = AppWidgetManager.INVALID_APPWIDGET_ID; + + private static final String PREFS_NAME = + "net.macdidi.myandroidtutorial.ItemAppWidget"; + private static final String PREF_PREFIX_KEY = "appwidget_"; + + // 選擇小工具使用的記事項目 + private ListView item_list; + private ItemAdapter itemAdapter; + private List items; + private ItemDAO itemDAO; + + public ItemAppWidgetConfigureActivity() { + super(); + } + + @Override + public void onCreate(Bundle icicle) { + super.onCreate(icicle); + + setResult(RESULT_CANCELED); + + // 改為使用應用程式主畫面 + setContentView(R.layout.activity_main); + + // 建立與設定選擇小工具使用的記事項目需要的物件 + item_list = (ListView)findViewById(R.id.item_list); + itemDAO = new ItemDAO(getApplicationContext()); + items = itemDAO.getAll(); + itemAdapter = new ItemAdapter(this, R.layout.single_item, items); + item_list.setAdapter(itemAdapter); + item_list.setOnItemClickListener(itemListener); + + Intent intent = getIntent(); + Bundle extras = intent.getExtras(); + + if (extras != null) { + mAppWidgetId = extras.getInt( + AppWidgetManager.EXTRA_APPWIDGET_ID, + AppWidgetManager.INVALID_APPWIDGET_ID); + } + + if (mAppWidgetId == AppWidgetManager.INVALID_APPWIDGET_ID) { + finish(); + return; + } + + } + + // 選擇記事項目 + AdapterView.OnItemClickListener itemListener = + new AdapterView.OnItemClickListener() { + @Override + public void onItemClick(AdapterView parent, View view, + int position, long id) { + final Context context = ItemAppWidgetConfigureActivity.this; + + // 讀取與儲存選擇的記事物件 + Item item = itemAdapter.getItem(position); + saveItemPref(context, mAppWidgetId, item.getId()); + + AppWidgetManager appWidgetManager = + AppWidgetManager.getInstance(context); + ItemAppWidget.updateAppWidget( + context, appWidgetManager, mAppWidgetId); + Intent resultValue = new Intent(); + resultValue.putExtra( + AppWidgetManager.EXTRA_APPWIDGET_ID, mAppWidgetId); + setResult(RESULT_OK, resultValue); + + finish(); + } + }; + + // 儲存選擇的記事編號 + static void saveItemPref(Context context, int appWidgetId, long id) { + SharedPreferences.Editor prefs = + context.getSharedPreferences(PREFS_NAME, 0).edit(); + prefs.putLong(PREF_PREFIX_KEY + appWidgetId, id); + prefs.commit(); + } + + // 讀取記事編號 + static long loadItemPref(Context context, int appWidgetId) { + SharedPreferences prefs = + context.getSharedPreferences(PREFS_NAME, 0); + long idValue = prefs.getLong(PREF_PREFIX_KEY + appWidgetId, 0); + + return idValue; + } + + // 刪除記事編號 + static void deleteItemPref(Context context, int appWidgetId) { + SharedPreferences.Editor prefs = + context.getSharedPreferences(PREFS_NAME, 0).edit(); + prefs.remove(PREF_PREFIX_KEY + appWidgetId); + prefs.commit(); + } + +} + + + diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemDAO.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemDAO.java new file mode 100644 index 0000000..b3c7b49 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/ItemDAO.java @@ -0,0 +1,214 @@ +package net.macdidi.myandroidtutorial; + +import android.content.ContentValues; +import android.content.Context; +import android.database.Cursor; +import android.database.sqlite.SQLiteDatabase; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +// 資料功能類別 +public class ItemDAO { + // 表格名稱 + public static final String TABLE_NAME = "item"; + + // 編號表格欄位名稱,固定不變 + public static final String KEY_ID = "_id"; + + // 其它表格欄位名稱 + public static final String DATETIME_COLUMN = "datetime"; + public static final String COLOR_COLUMN = "color"; + public static final String TITLE_COLUMN = "title"; + public static final String CONTENT_COLUMN = "content"; + public static final String FILENAME_COLUMN = "filename"; + public static final String RECFILENAME_COLUMN = "recfilename"; + public static final String LATITUDE_COLUMN = "latitude"; + public static final String LONGITUDE_COLUMN = "longitude"; + public static final String LASTMODIFY_COLUMN = "lastmodify"; + + // 提醒日期時間 + public static final String ALARMDATETIME_COLUMN = "alarmdatetime"; + + // 使用上面宣告的變數建立表格的SQL指令 + public static final String CREATE_TABLE = + "CREATE TABLE " + TABLE_NAME + " (" + + KEY_ID + " INTEGER PRIMARY KEY AUTOINCREMENT, " + + DATETIME_COLUMN + " INTEGER NOT NULL, " + + COLOR_COLUMN + " INTEGER NOT NULL, " + + TITLE_COLUMN + " TEXT NOT NULL, " + + CONTENT_COLUMN + " TEXT NOT NULL, " + + FILENAME_COLUMN + " TEXT, " + + RECFILENAME_COLUMN + " TEXT, " + + LATITUDE_COLUMN + " REAL, " + + LONGITUDE_COLUMN + " REAL, " + + LASTMODIFY_COLUMN + " INTEGER, " + + ALARMDATETIME_COLUMN + " INTEGER)"; + + // 資料庫物件 + private SQLiteDatabase db; + + // 建構子,一般的應用都不需要修改 + public ItemDAO(Context context) { + db = MyDBHelper.getDatabase(context); + } + + // 關閉資料庫,一般的應用都不需要修改 + public void close() { + db.close(); + } + + // 新增參數指定的物件 + public Item insert(Item item) { + // 建立準備新增資料的ContentValues物件 + ContentValues cv = new ContentValues(); + + // 加入ContentValues物件包裝的新增資料 + // 第一個參數是欄位名稱, 第二個參數是欄位的資料 + cv.put(DATETIME_COLUMN, item.getDatetime()); + cv.put(COLOR_COLUMN, item.getColor().parseColor()); + cv.put(TITLE_COLUMN, item.getTitle()); + cv.put(CONTENT_COLUMN, item.getContent()); + cv.put(FILENAME_COLUMN, item.getFileName()); + cv.put(RECFILENAME_COLUMN, item.getRecFileName()); + cv.put(LATITUDE_COLUMN, item.getLatitude()); + cv.put(LONGITUDE_COLUMN, item.getLongitude()); + cv.put(LASTMODIFY_COLUMN, item.getLastModify()); + + // 提醒日期時間 + cv.put(ALARMDATETIME_COLUMN, item.getAlarmDatetime()); + + // 新增一筆資料並取得編號 + // 第一個參數是表格名稱 + // 第二個參數是沒有指定欄位值的預設值 + // 第三個參數是包裝新增資料的ContentValues物件 + long id = db.insert(TABLE_NAME, null, cv); + + // 設定編號 + item.setId(id); + // 回傳結果 + return item; + } + + // 修改參數指定的物件 + public boolean update(Item item) { + // 建立準備修改資料的ContentValues物件 + ContentValues cv = new ContentValues(); + + // 加入ContentValues物件包裝的修改資料 + // 第一個參數是欄位名稱, 第二個參數是欄位的資料 + cv.put(DATETIME_COLUMN, item.getDatetime()); + cv.put(COLOR_COLUMN, item.getColor().parseColor()); + cv.put(TITLE_COLUMN, item.getTitle()); + cv.put(CONTENT_COLUMN, item.getContent()); + cv.put(FILENAME_COLUMN, item.getFileName()); + cv.put(RECFILENAME_COLUMN, item.getRecFileName()); + cv.put(LATITUDE_COLUMN, item.getLatitude()); + cv.put(LONGITUDE_COLUMN, item.getLongitude()); + cv.put(LASTMODIFY_COLUMN, item.getLastModify()); + + // 提醒日期時間 + cv.put(ALARMDATETIME_COLUMN, item.getAlarmDatetime()); + + // 設定修改資料的條件為編號 + // 格式為「欄位名稱=資料」 + String where = KEY_ID + "=" + item.getId(); + + // 執行修改資料並回傳修改的資料數量是否成功 + return db.update(TABLE_NAME, cv, where, null) > 0; + } + + // 刪除參數指定編號的資料 + public boolean delete(long id){ + // 設定條件為編號,格式為「欄位名稱=資料」 + String where = KEY_ID + "=" + id; + // 刪除指定編號資料並回傳刪除是否成功 + return db.delete(TABLE_NAME, where , null) > 0; + } + + // 讀取所有記事資料 + public List getAll() { + List result = new ArrayList<>(); + Cursor cursor = db.query( + TABLE_NAME, null, null, null, null, null, null, null); + + while (cursor.moveToNext()) { + result.add(getRecord(cursor)); + } + + cursor.close(); + return result; + } + + // 取得指定編號的資料物件 + public Item get(long id) { + // 準備回傳結果用的物件 + Item item = null; + // 使用編號為查詢條件 + String where = KEY_ID + "=" + id; + // 執行查詢 + Cursor result = db.query( + TABLE_NAME, null, where, null, null, null, null, null); + + // 如果有查詢結果 + if (result.moveToFirst()) { + // 讀取包裝一筆資料的物件 + item = getRecord(result); + } + + // 關閉Cursor物件 + result.close(); + // 回傳結果 + return item; + } + + // 把Cursor目前的資料包裝為物件 + public Item getRecord(Cursor cursor) { + // 準備回傳結果用的物件 + Item result = new Item(); + + result.setId(cursor.getLong(0)); + result.setDatetime(cursor.getLong(1)); + result.setColor(ItemActivity.getColors(cursor.getInt(2))); + result.setTitle(cursor.getString(3)); + result.setContent(cursor.getString(4)); + result.setFileName(cursor.getString(5)); + result.setRecFileName(cursor.getString(6)); + result.setLatitude(cursor.getDouble(7)); + result.setLongitude(cursor.getDouble(8)); + result.setLastModify(cursor.getLong(9)); + + // 提醒日期時間 + result.setAlarmDatetime(cursor.getLong(9)); + + // 回傳結果 + return result; + } + + // 取得資料數量 + public int getCount() { + int result = 0; + Cursor cursor = db.rawQuery("SELECT COUNT(*) FROM " + TABLE_NAME, null); + + if (cursor.moveToNext()) { + result = cursor.getInt(0); + } + + return result; + } + + // 建立範例資料 + public void sample() { + Item item = new Item(0, new Date().getTime(), Colors.RED, "關於Android Tutorial的事情.", "Hello content", "", "", 0, 0, 0); + Item item2 = new Item(0, new Date().getTime(), Colors.BLUE, "一隻非常可愛的小狗狗!", "她的名字叫「大熱狗」,又叫\n作「奶嘴」,是一隻非常可愛\n的小狗。", "", "", 25.04719, 121.516981, 0); + Item item3 = new Item(0, new Date().getTime(), Colors.GREEN, "一首非常好聽的音樂!", "Hello content", "", "", 0, 0, 0); + Item item4 = new Item(0, new Date().getTime(), Colors.ORANGE, "儲存在資料庫的資料", "Hello content", "", "", 0, 0, 0); + + insert(item); + insert(item2); + insert(item3); + insert(item4); + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MainActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MainActivity.java new file mode 100644 index 0000000..3434bdd --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MainActivity.java @@ -0,0 +1,351 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.app.ActivityOptions; +import android.app.AlarmManager; +import android.app.AlertDialog; +import android.app.PendingIntent; +import android.content.Context; +import android.content.DialogInterface; +import android.content.Intent; +import android.os.Build; +import android.os.Bundle; +import android.view.Menu; +import android.view.MenuInflater; +import android.view.MenuItem; +import android.view.View; +import android.widget.AdapterView; +import android.widget.ListView; +import android.widget.TextView; + +import java.util.List; + +public class MainActivity extends Activity { + + private ListView item_list; + private TextView show_app_name; + + // ListView使用的自定Adapter物件 + private ItemAdapter itemAdapter; + // 儲存所有記事本的List物件 + private List items; + + // 選單項目物件 + private MenuItem add_item, search_item, revert_item, share_item, delete_item; + + // 已選擇項目數量 + private int selectedCount = 0; + + // 宣告資料庫功能類別欄位變數 + private ItemDAO itemDAO; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_main); + + processViews(); + processControllers(); + + // 建立資料庫物件 + itemDAO = new ItemDAO(getApplicationContext()); + + // 如果資料庫是空的,就建立一些範例資料 + // 這是為了方便測試用的,完成應用程式以後可以拿掉 + if (itemDAO.getCount() == 0) { + itemDAO.sample(); + } + + // 取得所有記事資料 + items = itemDAO.getAll(); + + itemAdapter = new ItemAdapter(this, R.layout.single_item, items); + item_list.setAdapter(itemAdapter); + } + + @Override + protected void onActivityResult(int requestCode, int resultCode, Intent data) { + if (resultCode == Activity.RESULT_OK) { + Item item = (Item) data.getExtras().getSerializable( + "net.macdidi.myandroidtutorial.Item"); + + // 是否修改提醒設定 + boolean updateAlarm = false; + + if (requestCode == 0) { + // 新增記事資料到資料庫 + item = itemDAO.insert(item); + + items.add(item); + itemAdapter.notifyDataSetChanged(); + } + else if (requestCode == 1) { + int position = data.getIntExtra("position", -1); + + if (position != -1) { + // 讀取原來的提醒設定 + Item ori = itemDAO.get(item.getId()); + // 判斷是否需要設定提醒 + updateAlarm = (item.getAlarmDatetime() != ori.getAlarmDatetime()); + + // 修改資料庫中的記事資料 + itemDAO.update(item); + + items.set(position, item); + itemAdapter.notifyDataSetChanged(); + } + } + + // 設定提醒 + if (item.getAlarmDatetime() != 0 && updateAlarm) { + Intent intent = new Intent(this, AlarmReceiver.class); + //intent.putExtra("title", item.getTitle()); + + // 加入記事編號 + intent.putExtra("id", item.getId()); + + PendingIntent pi = PendingIntent.getBroadcast( + this, (int)item.getId(), + intent, PendingIntent.FLAG_ONE_SHOT); + + AlarmManager am = (AlarmManager) + getSystemService(Context.ALARM_SERVICE); + am.set(AlarmManager.RTC_WAKEUP, item.getAlarmDatetime(), pi); + } + } + } + + private void processViews() { + item_list = (ListView)findViewById(R.id.item_list); + show_app_name = (TextView) findViewById(R.id.show_app_name); + } + + private void processControllers() { + + // 建立選單項目點擊監聽物件 + AdapterView.OnItemClickListener itemListener = new AdapterView.OnItemClickListener() { + @Override + public void onItemClick(AdapterView parent, View view, + int position, long id) { + // 讀取選擇的記事物件 + Item item = itemAdapter.getItem(position); + + // 如果已經有勾選的項目 + if (selectedCount > 0) { + // 處理是否顯示已選擇項目 + processMenu(item); + // 重新設定記事項目 + itemAdapter.set(position, item); + } + else { + Intent intent = new Intent( + "net.macdidi.myandroidtutorial.EDIT_ITEM"); + + // 設定記事編號與記事物件 + intent.putExtra("position", position); + intent.putExtra("net.macdidi.myandroidtutorial.Item", item); + + // 依照版本啟動Acvitity元件 + startActivityForVersion(intent, 1); + } + } + }; + + // 註冊選單項目點擊監聽物件 + item_list.setOnItemClickListener(itemListener); + + // 建立記事項目長按監聽物件 + AdapterView.OnItemLongClickListener itemLongListener = new AdapterView.OnItemLongClickListener() { + @Override + public boolean onItemLongClick(AdapterView parent, View view, + int position, long id) { + // 讀取選擇的記事物件 + Item item = itemAdapter.getItem(position); + // 處理是否顯示已選擇項目 + processMenu(item); + // 重新設定記事項目 + itemAdapter.set(position, item); + return true; + } + }; + + // 註冊記事項目長按監聽物件 + item_list.setOnItemLongClickListener(itemLongListener); + + // 建立長按監聽物件 + View.OnLongClickListener listener = new View.OnLongClickListener() { + + @Override + public boolean onLongClick(View view) { + AlertDialog.Builder dialog = + new AlertDialog.Builder(MainActivity.this); + dialog.setTitle(R.string.app_name) + .setMessage(R.string.about) + .show(); + return false; + } + + }; + + // 註冊長按監聽物件 + show_app_name.setOnLongClickListener(listener); + } + + // 處理是否顯示已選擇項目 + private void processMenu(Item item) { + // 如果需要設定記事項目 + if (item != null) { + // 設定已勾選的狀態 + item.setSelected(!item.isSelected()); + + // 計算已勾選數量 + if (item.isSelected()) { + selectedCount++; + } + else { + selectedCount--; + } + } + + // 根據選擇的狀況,設定是否顯示選單項目 + add_item.setVisible(selectedCount == 0); + search_item.setVisible(selectedCount == 0); + revert_item.setVisible(selectedCount > 0); + share_item.setVisible(selectedCount > 0); + delete_item.setVisible(selectedCount > 0); + } + + // 載入選單資源 + @Override + public boolean onCreateOptionsMenu(Menu menu) { + MenuInflater menuInflater = getMenuInflater(); + menuInflater.inflate(R.menu.menu_main, menu); + + // 取得選單項目物件 + add_item = menu.findItem(R.id.add_item); + search_item = menu.findItem(R.id.search_item); + revert_item = menu.findItem(R.id.revert_item); + share_item = menu.findItem(R.id.share_item); + delete_item = menu.findItem(R.id.delete_item); + + // 設定選單項目 + processMenu(null); + + return true; + } + + // 使用者選擇所有的選單項目都會呼叫這個方法 + public void clickMenuItem(MenuItem item) { + // 使用參數取得使用者選擇的選單項目元件編號 + int itemId = item.getItemId(); + + switch (itemId) { + case R.id.search_item: + break; + // 使用者選擇新增選單項目 + case R.id.add_item: + // 使用Action名稱建立啟動另一個Activity元件需要的Intent物件 + Intent intent = new Intent("net.macdidi.myandroidtutorial.ADD_ITEM"); + // 依照版本啟動Acvitity元件 + startActivityForVersion(intent, 0); + break; + // 取消所有已勾選的項目 + case R.id.revert_item: + for (int i = 0; i < itemAdapter.getCount(); i++) { + Item ri = itemAdapter.getItem(i); + + if (ri.isSelected()) { + ri.setSelected(false); + itemAdapter.set(i, ri); + } + } + + selectedCount = 0; + processMenu(null); + + break; + // 刪除 + case R.id.delete_item: + if (selectedCount == 0) { + break; + } + + AlertDialog.Builder d = new AlertDialog.Builder(this); + String message = getString(R.string.delete_item); + d.setTitle(R.string.delete) + .setMessage(String.format(message, selectedCount)); + d.setPositiveButton(android.R.string.yes, + new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialog, int which) { + // 取得最後一個元素的編號 + int index = itemAdapter.getCount() - 1; + + while (index > -1) { + Item item = itemAdapter.get(index); + + if (item.isSelected()) { + itemAdapter.remove(item); + // 刪除資料庫中的記事資料 + itemDAO.delete(item.getId()); + } + + index--; + } + + itemAdapter.notifyDataSetChanged(); + selectedCount = 0; + processMenu(null); + } + }); + d.setNegativeButton(android.R.string.no, null); + d.show(); + + break; + case R.id.googleplus_item: + break; + case R.id.facebook_item: + break; + } + + } + + public void aboutApp(View view) { + Intent intent = new Intent(this, AboutActivity.class); + startActivity(intent); + } + + public void clickPreferences(MenuItem item) { + // 依照版本啟動Acvitity元件 + startActivityForVersion(new Intent(this, PrefActivity.class)); + } + + private void startActivityForVersion(Intent intent, int requestCode) { + // 如果裝置的版本是LOLLIPOP + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + // 加入畫面轉換設定 + startActivityForResult(intent, requestCode, + ActivityOptions.makeSceneTransitionAnimation( + MainActivity.this).toBundle()); + } + else { + startActivityForResult(intent, requestCode); + } + } + + private void startActivityForVersion(Intent intent) { + // 如果裝置的版本是LOLLIPOP + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { + // 加入畫面轉換設定 + startActivity(intent, + ActivityOptions.makeSceneTransitionAnimation( + MainActivity.this).toBundle()); + } + else { + startActivity(intent); + } + } + +} + + diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MapsActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MapsActivity.java new file mode 100644 index 0000000..abab911 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MapsActivity.java @@ -0,0 +1,338 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.app.AlertDialog; +import android.content.DialogInterface; +import android.content.Intent; +import android.location.Location; +import android.os.Bundle; +import android.support.v4.app.FragmentActivity; +import android.widget.Toast; + +import com.google.android.gms.common.ConnectionResult; +import com.google.android.gms.common.api.GoogleApiClient; +import com.google.android.gms.common.api.GoogleApiClient.ConnectionCallbacks; +import com.google.android.gms.common.api.GoogleApiClient.OnConnectionFailedListener; + +import com.google.android.gms.location.LocationListener; +import com.google.android.gms.location.LocationRequest; +import com.google.android.gms.location.LocationServices; + +import com.google.android.gms.maps.CameraUpdateFactory; +import com.google.android.gms.maps.GoogleMap; +import com.google.android.gms.maps.SupportMapFragment; +import com.google.android.gms.maps.model.BitmapDescriptor; +import com.google.android.gms.maps.model.BitmapDescriptorFactory; +import com.google.android.gms.maps.model.CameraPosition; +import com.google.android.gms.maps.model.LatLng; +import com.google.android.gms.maps.model.Marker; +import com.google.android.gms.maps.model.MarkerOptions; + +public class MapsActivity extends FragmentActivity + implements ConnectionCallbacks, + OnConnectionFailedListener, + LocationListener { + + private GoogleMap mMap; + + // Google API用戶端物件 + private GoogleApiClient googleApiClient; + + // Location請求物件 + private LocationRequest locationRequest; + + // 記錄目前最新的位置 + private Location currentLocation; + + // 顯示目前與儲存位置的標記物件 + private Marker currentMarker, itemMarker; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_maps); + setUpMapIfNeeded(); + + // 建立Google API用戶端物件 + configGoogleApiClient(); + + // 建立Location請求物件 + configLocationRequest(); + + // 讀取記事儲存的座標 + Intent intent = getIntent(); + double lat = intent.getDoubleExtra("lat", 0.0); + double lng = intent.getDoubleExtra("lng", 0.0); + + // 如果記事已經儲存座標 + if (lat != 0.0 && lng != 0.0) { + // 建立座標物件 + LatLng itemPlace = new LatLng(lat, lng); + // 加入地圖標記 + addMarker(itemPlace, intent.getStringExtra("title"), + intent.getStringExtra("datetime")); + // 移動地圖 + moveMap(itemPlace); + } + else { + // 連線到Google API用戶端 + if (!googleApiClient.isConnected()) { + googleApiClient.connect(); + } + } + } + + @Override + protected void onResume() { + super.onResume(); + setUpMapIfNeeded(); + + // 連線到Google API用戶端 + if (!googleApiClient.isConnected() && currentMarker != null) { + googleApiClient.connect(); + } + } + + @Override + protected void onPause() { + super.onPause(); + + // 移除位置請求服務 + if (googleApiClient.isConnected()) { + LocationServices.FusedLocationApi.removeLocationUpdates( + googleApiClient, this); + } + } + + @Override + protected void onStop() { + super.onStop(); + + // 移除Google API用戶端連線 + if (googleApiClient.isConnected()) { + googleApiClient.disconnect(); + } + } + + // 建立Google API用戶端物件 + private synchronized void configGoogleApiClient() { + googleApiClient = new GoogleApiClient.Builder(this) + .addConnectionCallbacks(this) + .addOnConnectionFailedListener(this) + .addApi(LocationServices.API) + .build(); + } + + // 建立Location請求物件 + private void configLocationRequest() { + locationRequest = new LocationRequest(); + // 設定讀取位置資訊的間隔時間為一秒(1000ms) + locationRequest.setInterval(1000); + // 設定讀取位置資訊最快的間隔時間為一秒(1000ms) + locationRequest.setFastestInterval(1000); + // 設定優先讀取高精確度的位置資訊(GPS) + locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY); + } + + private void setUpMapIfNeeded() { + if (mMap == null) { + mMap = ((SupportMapFragment) getSupportFragmentManager(). + findFragmentById(R.id.map)).getMap(); + + if (mMap != null) { + // 移除地圖設定 + //setUpMap(); + processController(); + } + } + } + + // 移除地圖設定方法 + private void setUpMap() { + // 建立位置的座標物件 + LatLng place = new LatLng(25.033408, 121.564099); + // 移動地圖 + moveMap(place); + + // 加入地圖標記 + addMarker(place, "Hello!", " Google Maps v2!"); + } + + private void processController() { + // 對話框按鈕事件 + final DialogInterface.OnClickListener listener = + new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialog, int which) { + switch (which) { + // 更新位置資訊 + case DialogInterface.BUTTON_POSITIVE: + // 連線到Google API用戶端 + if (!googleApiClient.isConnected()) { + googleApiClient.connect(); + } + break; + // 清除位置資訊 + case DialogInterface.BUTTON_NEUTRAL: + Intent result = new Intent(); + result.putExtra("lat", 0); + result.putExtra("lng", 0); + setResult(Activity.RESULT_OK, result); + finish(); + break; + // 取消 + case DialogInterface.BUTTON_NEGATIVE: + break; + } + } + }; + + // 標記訊息框點擊事件 + mMap.setOnInfoWindowClickListener(new GoogleMap.OnInfoWindowClickListener() { + @Override + public void onInfoWindowClick(Marker marker) { + // 如果是記事儲存的標記 + if (marker.equals(itemMarker)) { + AlertDialog.Builder ab = new AlertDialog.Builder(MapsActivity.this); + + ab.setTitle(R.string.title_update_location) + .setMessage(R.string.message_update_location) + .setCancelable(true); + + ab.setPositiveButton(R.string.update, listener); + ab.setNeutralButton(R.string.clear, listener); + ab.setNegativeButton(android.R.string.cancel, listener); + + ab.show(); + } + } + }); + + // 標記點擊事件 + mMap.setOnMarkerClickListener(new GoogleMap.OnMarkerClickListener() { + @Override + public boolean onMarkerClick(Marker marker) { + // 如果是目前位置標記 + if (marker.equals(currentMarker)) { + AlertDialog.Builder ab = new AlertDialog.Builder(MapsActivity.this); + + ab.setTitle(R.string.title_current_location) + .setMessage(R.string.message_current_location) + .setCancelable(true); + + ab.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialog, int which) { + Intent result = new Intent(); + result.putExtra("lat", currentLocation.getLatitude()); + result.putExtra("lng", currentLocation.getLongitude()); + setResult(Activity.RESULT_OK, result); + finish(); + } + }); + ab.setNegativeButton(android.R.string.cancel, null); + + ab.show(); + + return true; + } + + return false; + } + }); + } + + // 移動地圖到參數指定的位置 + private void moveMap(LatLng place) { + // 建立地圖攝影機的位置物件 + CameraPosition cameraPosition = + new CameraPosition.Builder() + .target(place) + .zoom(17) + .build(); + + // 使用動畫的效果移動地圖 + mMap.animateCamera(CameraUpdateFactory.newCameraPosition(cameraPosition), + new GoogleMap.CancelableCallback() { + @Override + public void onFinish() { + if (itemMarker != null) { + itemMarker.showInfoWindow(); + } + } + + @Override + public void onCancel() { + + } + }); + } + + // 在地圖加入指定位置與標題的標記 + private void addMarker(LatLng place, String title, String snippet) { + BitmapDescriptor icon = + BitmapDescriptorFactory.fromResource(R.drawable.ic_launcher); + + MarkerOptions markerOptions = new MarkerOptions(); + markerOptions.position(place) + .title(title) + .snippet(snippet) + .icon(icon); + + // 加入並設定記事儲存的位置標記 + itemMarker = mMap.addMarker(markerOptions); + } + + // ConnectionCallbacks + @Override + public void onConnected(Bundle bundle) { + // 已經連線到Google Services + // 啟動位置更新服務 + // 位置資訊更新的時候,應用程式會自動呼叫LocationListener.onLocationChanged + LocationServices.FusedLocationApi.requestLocationUpdates( + googleApiClient, locationRequest, MapsActivity.this); + } + + // ConnectionCallbacks + @Override + public void onConnectionSuspended(int i) { + // Google Services連線中斷 + // int參數是連線中斷的代號 + } + + // OnConnectionFailedListener + @Override + public void onConnectionFailed(ConnectionResult connectionResult) { + // Google Services連線失敗 + // ConnectionResult參數是連線失敗的資訊 + int errorCode = connectionResult.getErrorCode(); + + // 裝置沒有安裝Google Play服務 + if (errorCode == ConnectionResult.SERVICE_MISSING) { + Toast.makeText(this, R.string.google_play_service_missing, + Toast.LENGTH_LONG).show(); + } + } + + // LocationListener + @Override + public void onLocationChanged(Location location) { + // 位置改變 + // Location參數是目前的位置 + currentLocation = location; + LatLng latLng = new LatLng( + location.getLatitude(), location.getLongitude()); + + // 設定目前位置的標記 + if (currentMarker == null) { + currentMarker = mMap.addMarker(new MarkerOptions().position(latLng)); + } + else { + currentMarker.setPosition(latLng); + } + + // 移動地圖到目前的位置 + moveMap(latLng); + } + +} diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MyDBHelper.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MyDBHelper.java new file mode 100644 index 0000000..9350577 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/MyDBHelper.java @@ -0,0 +1,47 @@ +package net.macdidi.myandroidtutorial; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteDatabase.CursorFactory; +import android.database.sqlite.SQLiteOpenHelper; + +public class MyDBHelper extends SQLiteOpenHelper { + + // 資料庫名稱 + public static final String DATABASE_NAME = "mydata.db"; + // 資料庫版本,資料結構改變的時候要更改這個數字,通常是加一 + public static final int VERSION = 3; + // 資料庫物件,固定的欄位變數 + private static SQLiteDatabase database; + + // 建構子,在一般的應用都不需要修改 + public MyDBHelper(Context context, String name, CursorFactory factory, + int version) { + super(context, name, factory, version); + } + + // 需要資料庫的元件呼叫這個方法,這個方法在一般的應用都不需要修改 + public static SQLiteDatabase getDatabase(Context context) { + if (database == null || !database.isOpen()) { + database = new MyDBHelper(context, DATABASE_NAME, + null, VERSION).getWritableDatabase(); + } + + return database; + } + + @Override + public void onCreate(SQLiteDatabase db) { + // 建立應用程式需要的表格 + db.execSQL(ItemDAO.CREATE_TABLE); + } + + @Override + public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { + // 刪除原有的表格 + db.execSQL("DROP TABLE IF EXISTS " + ItemDAO.TABLE_NAME); + // 呼叫onCreate建立新版的表格 + onCreate(db); + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PlayActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PlayActivity.java new file mode 100644 index 0000000..5cfb523 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PlayActivity.java @@ -0,0 +1,64 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.content.Intent; +import android.media.MediaPlayer; +import android.net.Uri; +import android.os.Bundle; +import android.view.View; + +public class PlayActivity extends Activity { + + private MediaPlayer mediaPlayer; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_play); + + Intent intent = getIntent(); + String fileName = intent.getStringExtra("fileName"); + + // 建立指定資源的MediaPlayer物件 + Uri uri = Uri.parse(fileName); + mediaPlayer = MediaPlayer.create(this, uri); + } + + @Override + protected void onStop() { + if (mediaPlayer.isPlaying()) { + // 停止播放 + mediaPlayer.stop(); + } + + // 清除MediaPlayer物件 + mediaPlayer.release(); + super.onStop(); + } + + public void onSubmit(View view) { + // 結束Activity元件 + finish(); + } + + public void clickPlay(View view) { + // 開始播放 + mediaPlayer.start(); + } + + public void clickPause(View view) { + // 暫停播放 + mediaPlayer.pause(); + } + + public void clickStop(View view) { + // 停止播放 + if (mediaPlayer.isPlaying()) { + mediaPlayer.stop(); + } + + // 回到開始的位置 + mediaPlayer.seekTo(0); + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PrefActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PrefActivity.java new file mode 100644 index 0000000..b2dcc4a --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/PrefActivity.java @@ -0,0 +1,38 @@ +package net.macdidi.myandroidtutorial; + +import android.content.SharedPreferences; +import android.os.Bundle; +import android.preference.Preference; +import android.preference.PreferenceActivity; +import android.preference.PreferenceManager; + +public class PrefActivity extends PreferenceActivity { + + private SharedPreferences sharedPreferences; + private Preference defaultColor; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + // 指定使用的設定畫面配置資源 + addPreferencesFromResource(R.xml.mypreference); + defaultColor = (Preference)findPreference("DEFAULT_COLOR"); + // 建立SharedPreferences物件 + sharedPreferences = + PreferenceManager.getDefaultSharedPreferences(this); + } + + @Override + protected void onResume() { + super.onResume(); + // 讀取設定的預設顏色 + int color = sharedPreferences.getInt("DEFAULT_COLOR", -1); + + if (color != -1) { + // 設定顏色說明 + defaultColor.setSummary(getString(R.string.default_color_summary) + + ": " + ItemActivity.getColors(color)); + } + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/RecordActivity.java b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/RecordActivity.java new file mode 100644 index 0000000..62b5e93 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/java/net/macdidi/myandroidtutorial/RecordActivity.java @@ -0,0 +1,180 @@ +package net.macdidi.myandroidtutorial; + +import android.app.Activity; +import android.content.Intent; +import android.media.MediaRecorder; +import android.os.AsyncTask; +import android.os.Bundle; +import android.util.Log; +import android.view.View; +import android.widget.ImageButton; +import android.widget.ProgressBar; + +import java.io.IOException; + +public class RecordActivity extends Activity { + + private ImageButton record_button; + private boolean isRecording = false; + private ProgressBar record_volumn; + + private MyRecoder myRecoder; + + private String fileName; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_record); + + processViews(); + + // 讀取檔案名稱 + Intent intent = getIntent(); + fileName = intent.getStringExtra("fileName"); + } + + public void onSubmit(View view) { + if (isRecording) { + // 停止錄音 + myRecoder.stop(); + } + + // 確定 + if (view.getId() == R.id.record_ok) { + Intent result = getIntent(); + setResult(Activity.RESULT_OK, result); + } + + finish(); + } + + private void processViews() { + record_button = (ImageButton) findViewById(R.id.record_button); + record_volumn = (ProgressBar) findViewById(R.id.record_volumn); + // 隱藏狀態列ProgressBar + setProgressBarIndeterminateVisibility(false); + } + + public void clickRecord(View view) { + // 切換 + isRecording = !isRecording; + + // 開始錄音 + if (isRecording) { + // 設定按鈕圖示為錄音中 + record_button.setImageResource(R.drawable.record_red_icon); + // 建立錄音物件 + myRecoder = new MyRecoder(fileName); + // 開始錄音 + myRecoder.start(); + // 建立並執行顯示麥克風音量的AsyncTask物件 + new MicLevelTask().execute(); + } + // 停止錄音 + else { + // 設定按鈕圖示為停止錄音 + record_button.setImageResource(R.drawable.record_dark_icon); + // 麥克風音量歸零 + record_volumn.setProgress(0); + // 停止錄音 + myRecoder.stop(); + } + } + + // 在錄音過程中顯示麥克風音量 + private class MicLevelTask extends AsyncTask { + @Override + protected Void doInBackground(Void... args) { + while (isRecording) { + publishProgress(); + + try { + Thread.sleep(200); + } + catch (InterruptedException e) { + Log.d("RecordActivity", e.toString()); + } + } + + return null; + } + + @Override + protected void onProgressUpdate(Void... values) { + record_volumn.setProgress((int) myRecoder.getAmplitudeEMA()); + } + + } + + // 執行錄音並且可以取得麥克風音量的錄音物件 + private class MyRecoder { + + private static final double EMA_FILTER = 0.6; + private MediaRecorder recorder = null; + private double mEMA = 0.0; + private String output; + + // 建立錄音物件,參數為錄音儲存的位置與檔名 + MyRecoder(String output) { + this.output = output; + } + + // 開始錄音 + public void start() { + if (recorder == null) { + // 建立錄音用的MediaRecorder物件 + recorder = new MediaRecorder(); + // 設定錄音來源為麥克風,必須在setOutputFormat方法之前呼叫 + recorder.setAudioSource(MediaRecorder.AudioSource.MIC); + // 設定輸出格式為3GP壓縮格式,必須在setAudioSource方法之後, + // 在prepare方法之前呼叫 + recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP); + // 設定錄音的編碼方式,必須在setOutputFormat方法之後, + // 在prepare方法之前呼叫 + recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB); + // 設定輸出的檔案名稱,必須在setOutputFormat方法之後, + // 在prepare方法之前呼叫 + recorder.setOutputFile(output); + + try { + // 準備執行錄音工作,必須在所有設定之後呼叫 + recorder.prepare(); + } + catch (IOException e) { + Log.d("RecordActivity", e.toString()); + } + + // 開始錄音 + recorder.start(); + mEMA = 0.0; + } + } + + // 停止錄音 + public void stop() { + if (recorder != null) { + // 停止錄音 + recorder.stop(); + // 清除錄音資源 + recorder.release(); + recorder = null; + } + } + + public double getAmplitude() { + if (recorder != null) + return (recorder.getMaxAmplitude() / 2700.0); + else + return 0; + } + + // 取得麥克風音量 + public double getAmplitudeEMA() { + double amp = getAmplitude(); + mEMA = EMA_FILTER * amp + (1.0 - EMA_FILTER) * mEMA; + return mEMA; + } + } + +} \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable-hdpi/ic_launcher.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable-hdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..96a442e5b8e9394ccf50bab9988cb2316026245d GIT binary patch literal 9397 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 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable-nodpi/example_appwidget_preview.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable-nodpi/example_appwidget_preview.png new file mode 100644 index 0000000000000000000000000000000000000000..894b069a4907d258f60b1b2406b90f5a0fe1c35b GIT binary patch literal 3522 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 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable-xxhdpi/ic_launcher.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable-xxhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..4df18946442ed763bd52cf3adca31617848656fa GIT binary patch literal 19388 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 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/alarm_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/alarm_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..c6cac8879daf79f0732b99457184bc8fd628b792 GIT binary patch literal 1605 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVit#Ld~<$;Hsr$-u?X(ACh=#L&{!#L3yw z&D7A`#K{n**Cju>G&eP`1g19yq1OnfUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dPPaI_ z;WQ7bHwCv_3~=hz2RcR{6tzfE4HE*U9}p9saDg0n(of9;ruQOX!ro=I?IZ&O({@i6 z$B>F!Z>D=@h&YP0rJJa^`6ewoJV7SLfo1Wc_CWovt_2<9<|3WaZbzG&*o%b@Hh0J! zS-e9g)=}7z#XROfx77~mazPuH$r+0r?<{)z{QJ5K zWRH7_Y~uA0*&)g*664VMU4EvIT6oN*T@Q3?I3+g=y?QA3=WAKo=9_zDH=KNam`y@n zu})`Ad+SayiB}EFd0TxAf>*yU;#?oNpYxAidd|`Kh0aQ6I+ZWTx;OCdlU?RuJf|sf zgTyjNDK^b*;AKeU#7Qfsnjs!`cmk%MKat;4x0<4f`jfcd%N5qtt+L+Ce4@ z#`J~4+YY34zUb9T;5&2aRPHC&4PtSJ6f2hPh`L$J9yLW-J7IOpy!Nb!wR3g|Je9b1 z!s_x&*`KPuMF33)} z)aCZ7uHvn*HCx!LuO)ozzI}JRbUo_J0|UX`4?DZgxXW(cGO2V^jy+d+(q2yHTe`kh z;otNwPP-xbpLhO4r8^v(ZPzulOj;9Zem5i6GiY-3+j{XSA1znMeq0_I=&+FWk9Y#Z Xdifc%87r2p0Tt|?u6{1-oD!M literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/item_drawable.xml b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/item_drawable.xml new file mode 100644 index 0000000..37607e2 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/item_drawable.xml @@ -0,0 +1,13 @@ + + + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/location_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/location_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..4c3c514bb08dcd0426c33cd0f914dda4a5b1e405 GIT binary patch literal 1496 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVit)!5a*+1SX`$-u?X(ACh=#L&{!#L3yw z&D7A`#K{n**Cju>G&eP`1g19yq1PCvUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dHn+Gq z8JSo*;x-SeHwBwpAbL%3xp! z0|S$nr;B4q#jQ8fH)aP1inQO)w%QUhM{9{_=GKgiZdtFkfD5iC-e`4Pa+TR~@aVxN z-Y$>1IijV)b7ic5v#)RtSs~G(;3<%`&`);~-`4Z@8{Z#vy|nMy)2lph8~H!I+xz@p z@%f$4??~HoFs+|_>1k;MAMcBQ+ZAV@-L|OS)8xpdI8m>qm)@~$(O7=eZMjpzvHyaO z*3y}w7sV$lska?9GCHfH?Pv`1KJu5a{@c+ejWX2BFeVyTp<}G0qUZhah zS@|)wy)gY~YGJre$5WMa2ROyncRbSTuKpq_8SSuLmpLeD)?EEbMQZM@aWTy{xpK*3 z7iXsR-B2{Ro_>URqvvhU!dMyc+gs*}iyz@W*WO#J`_AC_^BrwTrxR$S)E5KvfUA^S~g2%k!W-m`$=B7T&&}0!ms(I(Oe%Rc7iO1ibO_^=D za7~bQnkDzNMfr<(UI@CcD5_)n?(5L8YgXmDnT+dN`#W+q!rq>9ZQ1gyX8FmsGxGKvFRhtvFlA?7?YhT-r?x!(wRy?<#|FyptobK--pOHG zQ^+^%+h<21v8nwDyUb58R4eEFeQ*9yY`6c>Sewr9!l}XX(SHvAZ)3mLu%C&CVYXn` Uqs*7f0zhT0r>mdKI;Vst0KeiiegFUf literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/pause_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/pause_icon.png new file mode 100755 index 0000000000000000000000000000000000000000..a5aee6f2cb2e807e8e8f98be18766581bc7fad2e GIT binary patch literal 1107 zcmaJ=OK1~87+$rCtpuwPYYUcf^Xb*Z&DEIotdBi`~KHAgB|Ug zJry+-1VMP>tzrW2z+DR$;J=g4f5BTdYVJavW(s9w8xm2~?1mt&%RMjwWi_|&2@Dg& zyfs>~3w23tf@12F?D|ky-NI;s2sdOcS?Prcbi*Fah>#O^?~_1NBjh$tVk9dD(^_lZ zhMoELq>}Gdc$I9}1j1PX6X+1hAgk{+93dMar+5W?cFi;ird&{Ogq%*QOX>hI(}sYf z7@xw_v!Fgmu|b9p^6P=0VHukB(@e<6G6EM87#7TYB#vgQDIp=Y%*4XC2$@F65@{i3?g(CTeFa68o*_gyUjinA#tYDCFs^H zt>MhH2{(++$`;L1j9XICQIh`es_V1f4obj3`TmpGN#-m_C!k~Y*$N(9YPB265@I%# zk!dGQb8oSW9cdGpPTI6UtdpZyAjyhmxRxn|BnffDL9(I1xELWZ2c>DMz;OOhfRFK9 z6U(q{Gs_2AzB$+wi?(oVDA>dnv7)K;>Ciw$tojEVEs1r-pj$Yy2yJaYR9kFQ2UA50 zTIpOuCHZEsYUx}yl*H0F8QLA~U!yLXuzuX*tZ?yVR{GGudbhE#uV3xv@S}Jf7o*AS z%keYOiigZH@zitg?MbqB=uG7&iwYmwcw*$(?XmKkugm-2eje1uE)RHDtUa~r*R9I4 zElbLKhnH>LRR$Wjy?8$q2}ZwN5x+hi*?Dgz8m*dqG42`B>l$kh9I5><(*MSLr>5XT zh5Cw~Ij;iC=d}qxy@juj-h~^jU4g=e-2KywJ-JtN=T;VoGVoxc>SQ`OxnuI$$qVya zQ?I`t8ofB2XdMj^hwHqTw$#_q4ZBn8w*Lrxes+{NxcKAJJ;eEH?#8*}o<;66k2klA J!%cezeglElYo-7I literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/play_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/play_icon.png new file mode 100755 index 0000000000000000000000000000000000000000..6a40cd5f7bff6c57d8349bcba17648e2fdabedc4 GIT binary patch literal 1248 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVil%*epP$->gq$G&eP`1g19yq1O?oUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dZnv1@ zG!Lpb1-Dx)aO%|uIz}H9wMbD769T3m5EGtofgE_!Pt60S_ab1z_LS;2W?*0p@^oU1PZ9MzCuS-Z@1@eQA(D8tnc61wVVaLqcBX29V# zIWsfCC{pB#;hbN;Y=1mG{NVSVChH#?3JPqh|LrZGSga@d$!j@N{9T4gOd3khts8O< z?X^rPXb9`z;YkuzpTR45fIZHj&o?BH>tajpOh?&liEZx)I}SJ}I5=yUdI zwcx(@vCZ>MBKP9@Zw9BTD$?TGZ|G+IGUZf=3##35+)i(c`JQwIC;NGYYfAs7gf=z( zd%E<}-7T*g=REtzXT4)%)}bwz{w}bZpZ#W!zhlhvxU=Q4`JVk@$q@kz*H;xE&f-7v zoz3Lf+=R;*m_w)X*Ww_78!_em8^&}%-Jpxqbc)I$z JtaD0e0syqNvA+NS literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_dark_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_dark_icon.png new file mode 100755 index 0000000000000000000000000000000000000000..bcf83caa1eddd80dfcb257777042a75a77602129 GIT binary patch literal 1319 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVi;&CSu#($UP+$-vFf(ACh=#L&{!#L3yw z&D7A`#K{n**Cju>G&eP`1g19yq1POzUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dZnv1= zG!Lpb1-DyFaq86vIz}H9wMbD769T3m5EGtofgE_!Pt60S_ab1zo-nT_hk=1{rl*Tz zNX4x;*UoFRB+4BBD4Sib8mTU0@uu;~f;%T#i!NzBdGz3bqt`mWb9pjLj!3faw8-IX z?Ph)&AHd}~QE6J{@!u=^XI(MceD+P;vz6h$&%Al_x#s!J_n!|J$*6~&3=&;k(8;u- zg5j#_zdV-YjK-6XvU46E$#)Lp+?*ie{X-}?NF#x7dE$Ro+gER7x9!O^sXjfCd0X4x za!rw)@z+#J7G+AX3WPBj&nQo=-k7v+^M;0(3d|W7UVV>WCcZ_EC%ceKr$N_dmE8Zw zJ9o@jT4{WO;mmB+kb0-ywp(}mc9wT@xIv2M*{3C}B?nd#!zCmzL}ua4Mt zqsK<-h~ll1yNx<+6E5zxlUaB4+>+1dKVRB+z~3-(Ut7A4)`W-|j8~Ew?q;~@8om1A);`HP z=FyqgIv<1&sDzaW{R%l=shSw1o!~WJIIcgfjw$1c&Wj-1w^2Etyj+a`uivo#bwII# h6~7B3OVB=62?k@QQ!eNC1|I+wU7oIfF6*2UngDLe-ZuaM literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_red_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_red_icon.png new file mode 100755 index 0000000000000000000000000000000000000000..2b44af0bb4e9dce2d433b53154b9281b1ac27fdc GIT binary patch literal 3361 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 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_sound_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/record_sound_icon.png new file mode 100755 index 0000000000000000000000000000000000000000..a1382ac68845a99ec7d9a9e1bed0ffabc910b3f4 GIT binary patch literal 1318 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVi-z}VQp#l_Xs$-u?X(ACh=#L&{!#L3yw z&D7A`#K{n**Cju>G&eP`1g19yq1O$kUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dZnrq& zG!Lpb1-DyVaO%|uIz}H9wMbD769T3m5EGtofgE_!Pt60S_ab1zexSzsmVtqBhNp{T zNX4x;*Y|oKc91#taoN6wj}lxRUBu^dDS3#;2gDq8yu;$Y{2$|A<~gz=;^yM&2OULZ zTml3{G9B3+j~qR@bl;*M*Uakp=Q(j`2(D*SVV956%lsWP>G$2Mg|X&hY2C-2@&X$! zFo&>PYEJd~xx3A_d~#qt-u$*Ew8VSawT8Mc3JX_WTgPs)*w-^| zpx5SvPe3J?r>mdKI;Vst01cPZvj6}9 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/retangle_drawable.xml b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/retangle_drawable.xml new file mode 100644 index 0000000..51d1e84 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/retangle_drawable.xml @@ -0,0 +1,13 @@ + + + + + + + + \ No newline at end of file diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/select_color_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/select_color_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..8567d5e49bca19a5e93448016b104c3e77d5286e GIT binary patch literal 1455 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVip%-q7#$;imm$-u?X(ACh=#L&{!#L3yw z&D7A`#K{n**Cju>G&eP`1g19yq1PCvUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dZnqfV zG!Lpb1-Dxaaq86vIz}H9wMbD769T3m5EGtofgE_!Pt60S_ab1z7SfZM$-uy*=;`7Z zQgQ3e^t0M5i89CECWN1QJ;_xgUtE0CMY(_-788!1JbCu4x5!WHDMt=EcI=t9ee#q! zVUG@Yo(g>H_-Wtn$G@+XXJ4xNHq&r6vw!WZ+-;Tn?*H6-w#WJU*CqF3qAuyLuV>cW z=vTEj?xv}!gUGq5ksBQp7Nj|Bx^$4?1oQI;5-|+yzN$8?YZoxII0&y`woKqMxizgR zKg=wHSzdv6O_Q?$*D6QJ-npt;OtL(Ch2J#Tc7->%PvBv5tX1L>X%c%@#r&qZuxtN^ zniE}|q0Dm+@J>4*ID_HogPGMkD*hN`I~{cp&0ynxFy+zGJ;f9I)Hkp-iI|IW`ab+0 zcds&J&jkg&zRd~ZcN#)AzwAhN@L6&|{9}0Y50f2ibq|;YxwMx%cVAxgUB+bc0cFAO zpZ_qn-0ugXuEyC@bsk%i)@<}`jJ=rJd=2~!mQ(!UEdq>b{|P) zQ?hwwa*;vHF~2Mj@5R&&N+&``}r+rBz1QG4g0ZR)6>E?DJ+Y&3Ou=?+o`Cx z?12&Qtxt9eW_wc@uWg!Ne9+*;cITLDta{%zYV4Ez;IPo(KIh%Cj?+pLPM!VzW79;> z!j1QM1%FN4ai(GN7U?X9|3W)Cx-V|papr(wvCOoXa&Eu6U$GB<{5Sjanaz$>f}wlz V!sw^-FC>79P)}Dsmvv4FO#nf~6GH$1 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/selected_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/selected_icon.png new file mode 100755 index 0000000000000000000000000000000000000000..b8915716e06792d926f397a7b17346f656c86aa1 GIT binary patch literal 1546 zcmaJ>eM}Q~7(NOZ>jx?|Mx7)_1qW{Q?n+DPg;FVn(PM_rcwjwgk7z zaExq;FcCM$1T(WNW-4=WI(1VrnIo~NbIM>M*htY0#}vP^FxeFW{X=%i-TV8!d7tO~ zdUEwS*{>~}w{ji;01FLSdXr$+1>d=Igzo{O;<{i~@EK-4moDL57>fgGHrj%N1`;d9 zO*m%rRCM7f01z5Ogb|R~a#dE8&^Q=`4tJ)^jrh+MMR3IR69s}KEImMxFjdYS170y97Bn2l! zc!x&%v?#MN2TZ3~98}8S1S_0`fJq7&qJY&3^{b#9Mqmh$L$E3Vfl;Ljg%NP-kqX&Z zTM25?XHMl3ZW^hb=NS}&TrQW)l_;axQV3D2)etO)%@DX$DNsRmu?1h*=3L=$S+q zji`a*c#N{*2E9fqaL5S4h9>Hh6v?`@WTg&HMvx3dtw7Wn3SD|yrV>#pbjUPTPg|WN zPVv)N+cT_sR%}oVBqLvOnw4(~Yn#0m)vQ=ZC?~O*G{jdJEH8vU^=Avr+i|X?`*2LY9 z7jHvHE29PrBH!kINN(*uSt$Oev6h7iu1N7t^Bcq8%eH4kerw;l_*m?|A2AK`y6?34 zP`@{J#lYB+FlJ}1sp{bG{!_iCO_Q1c_7CzZkyMzhv`>@qo7TfY@Gmt3uSG*tgxNWQc=v|JYQSD*fn7li5 z#PB$#i7UEVf00h?$hp1uLQj2zQtZFGsWX7jD_#?wJrFj3Z4#6Vd{fyGdK_MQaQVc! gtQRZb>MHSlrsCb%kZjuDTvPEKT0lu;tH^SKIYs znNAoCPYT1Uf{LJ`M-L7}QM@Xs$DL%Cfrxkz>r2;m9;|_v_uh~1`+k4l>g2@v&W;lu z6h(C=$CMP=L9q6>lfPdoeJ9&-9M9k>XBHPU4^a`r$s&-nv^kVQnz6L_4h>UO>nSsx z!5MX2(jAM@0t{2MT*9WP;gO=N>GKGKESfWIncjZ!k_M(B)6;^=scsDA%(0S(rb-iO zy)>_j20d~P3>PILU?Hr5qP1ZAQcx)M1DW|jt32%eYeouD%6B#1d40s_N@bZ!uW zL7ss;C-UMM5auAq!Z6DXg&-#hLlOr;?W2h|&zO}`N}}eA++;e3u`97`p-^B71B~O% zu}~C6mJ74taEKs6{*sNgV#xOU>Iw?-bDVAp)Ur+<%QT5}Ck4xG$l89bOr&6J zQFB?ya6w9SpsM~KYFQ1mk5gzj-hT@F=_MDjDdanOPbZC=?F(GFQp`gdcD%IXEYzzw znRBq?=NuQrrUV88Rntv7uv8hUDkW_nYqpM(icAv?#xxCy6XU{Jo`*sdauCL$$U`yC zM`MwM0EhS}taBAd&s)gGb*{0?4L9Wm!(h3@vw}SH3NjL&V}WYYlG!wtfkaci8rNtZ z3)hs(l3>`NwSTp`ZX)v&92@sco)o9SVB6-^(Q*5dcD?kC@7nJF*aF_`<)24Z4wpYip0)mtgzkT6;hvA( zT(0ESFPEn)Ii@GS@t`%)%Wah}>^rp8exR?r|5z7)ZDsW9Q*iO&Eq-kM?c#=c@6BG} kj`8W}+K)AY-Rz~hW~lBby{BW3UdzE#O~xmb$I+SPKUWG$(*OVf literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/take_picture_icon.png b/examples/0601/MyAndroidTutorial/app/src/main/res/drawable/take_picture_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..08fb5140031c75b9ced9b45b53f29ef840b1662f GIT binary patch literal 1460 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdz$r9IylHmNblJdl&R0hYC{G?O` z&)mfH)S%SFl*+=BsWuD@%qp275hW46K32*3xq68pHF_1f1wh>l3^w)^1&PVosU-?Y zsp*+{wo31J?^jaDOtDo8H}y5}EpSfF$n>ZxN)4{^3rViZPPR-@vbR&PsjvbXkegbP zs8ErclUHn2VXFi-*9yo63F|8hm3bwJ6}oxF$}kgLQj3#|G7CyF^YauyCMG83 zmzLNn0bL65LT&-v*t}wBFaZNhzap_f-%!s00+w{G(#^lGsVit(b(D9#L3ar$-u?X(ACh=#L&{!#L3yw z&D7A`#K{n**Cju>G&eP`1g19yq1OqgUQlAlEdbi=l3J8mmYU*Ll%J~r_Ow+dZns$A zG!Lpb1-Dx)aq86vIz}H9wMbD769T3m5EGtofgE_!Pt60S_ab1z{*s_DkAZAv0}ffC2xUzIe~Wf9bkjpz=0n^hCZ@#vC5+SOJ13lh2&UT}YrT(m%6P-si` z#HD=9>tZ@~%|0X|zDVrx3){c)KUmvu@BMu3K+d~H`NOu)Eua6KUp()*rO%5w+`Dd8 zHR)~laZp?EnyKhxpVPtw`{tw=mqq?EUB*6QN4Mb7SnT}P_ig>-^GV$| zHk7%a>8d<(U5I7JkB`eaH^jyyzOfg1_Qz$9D7%i;sT=!l2&-;9bB>Ud_Q{wl3| z5C1IG=rTN}8(`m&%d+pmiN$_b8Lh9CmSt7$@-q8!DE8x=H#bD%U!^#mW^z)hnf!iH zxf0*1FQOYY3t1D@S!SHwxk4Z(nbTEc!TDo(-R~bQn)x+a_h{neGCkF4w}b>H-WGnu zK68TqhRyqugNi3v%5plnE6#dW-CJ4dr=s$pYLAw=^Zaae-sUO4V-r(3OQiZotZ)79H@{gF$>D_!vlMHjdqrR8e>=SiveF=CwAkjMQ) a`2fTAxy{{g7+1xB%27{OKbLh*2~7ZWBp}29 literal 0 HcmV?d00001 diff --git a/examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_about.xml b/examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_about.xml new file mode 100644 index 0000000..211a9b6 --- /dev/null +++ b/examples/0601/MyAndroidTutorial/app/src/main/res/layout/activity_about.xml @@ -0,0 +1,36 @@ + + + + + + +