2017年12月17日

Android App Privacy Policy

隱私權政策

我們非常重視您的隱私,並致力於保護您的個人和敏感資料。本隱私權政策詳細說明我們如何收集、使用、分享和保護您的資料。使用本應用程式,即表示您同意本政策的條款。

1. 資料收集
我們可能會收集和處理以下類型的資料:
設備信息:設備類型、唯一設備標識碼、操作系統等。
位置信息:僅當應用程式需要提供基於位置的服務時,會請求訪問位置數據。
使用資料:包括應用程式的使用情況和互動記錄。

2. 適合兒童使用
本應用程式旨在成為家庭友好且適合兒童使用。為保護兒童的隱私和安全,我們遵守《兒童線上隱私保護法案》(COPPA)和其他相關的隱私法規:

我們不會在不經父母或監護人同意的情況下,向 13 歲以下兒童收集個人資料。
所有針對兒童的功能均以隱私和安全為前提設計。

3. 廣告資料收集和使用
本應用程式包含適合兒童的廣告,但我們確保廣告內容適合年齡:
設備識別碼:用於展示相關廣告。
使用數據:以了解用戶與廣告和應用程式的互動情況。
我們不會向未經授權的第三方出售或共享兒童的個人資料。廣告合作夥伴會根據法律和政策遵循適當的安全措施來保護兒童資料。

4. 資料使用
我們使用您的資料來:
提供並改善我們的服務和用戶體驗。
向您發送有關更新和應用程式的通知。
展示個性化和年齡適當的廣告內容。
確保應用程式的技術運行和安全性。

5. 資料共享
我們不會將您的個人和敏感資料出售或共享給未經授權的第三方。但我們可能會在以下情況中共享資料:
法律要求:在法律或法律程序要求時。
服務供應商:協助我們運營應用程式的可信服務供應商。
廣告合作夥伴:為了提供和優化廣告體驗,合作夥伴會遵守法律和隱私規範。

6. 第三方網站
本服務可能包含連至第三方服務或第三方其他網站之連結,這個第三方包括與您進行交易的零售商。請注意,本政策僅適用於我方透過服務收集的個人資訊,我方不對第三方可能透過其網站或服務收集、存儲及使用的個人資訊負有責任。您應始終仔細閱讀每個您瀏覽網站的隱私權政策。

7. 資料保護
我們採取適當的技術和組織措施來保護您的個人資料免遭未經授權的訪問、更改、洩露或銷毀。

8. 資料保留與刪除
我們僅在提供服務必要的期限內保留您的個人資料。在您的請求下,我們將刪除您的個人資料,除非法律要求保留。

9. 用戶權利
您有權查閱、更正或刪除您的個人資料,並有權限制或反對資料處理。如果您有此類請求,請與我們聯繫。

10. 聯絡方式
APP開發者為JackyWell所有,如有任何有關隱私權的問題或請求,請透過郵件聯繫我們
jackylee3324@gmail.com

11. 政策更新
我們可能會不時更新本隱私權政策。當我們更新時,會將新的政策發布在此頁面上,並更新政策的生效日期。

生效日期:2024/11/12

2017年12月14日

Android ToolBar

Android ToolBar

ToolBar是在安卓5.0中引入的(android.widget.Toolbar)。在不引入兼容庫的情况下,只能在5.0以上的版本運行;而在引用android.support.v7.widget.Toolbar類之後,可以支持安卓2.1以上版本。


引入了android.support.v7兼容庫。
//build.gradle
    compile 'com.android.support:appcompat-v7:25.3.0'


然後更改Application的主题,找到values文件夾下的styles.xml,更改主题為NoActionBar
<!--AndroidManifest.xml-->
<application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme
        <!--修改AppTheme的定义-->
        >

<!--styles.xml-->
<style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">



Layout布置

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/activity_main"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        tools:context="me.zxm.learnactionbartoolbar.MainActivity"
        >
    <android.support.v7.widget.Toolbar
            android:id="@+id/ToolBar"
            android:minHeight="?attr/actionBarSize"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            app:titleTextColor="@android:color/white"
            android:background="?attr/colorPrimary"
            ></android.support.v7.widget.Toolbar>
</RelativeLayout>




在res文件夾下新建menu文件夾,然後在该文件夾下新增一个布局文件

<?xml version="1.0" encoding="utf-8"?>
<menu
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:android="http://schemas.android.com/apk/res/android">
<item
        android:id="@+id/mySearch"
        android:icon="@android:drawable/ic_menu_search"
        app:showAsAction="ifRoom"
        android:title="Search">
</item>
<item
        android:id="@+id/myMore"
        android:icon="@android:drawable/ic_menu_more"
        app:showAsAction="ifRoom|withText"
        android:title="More">
</item>
</menu>




Activity Toolbar Code

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.Toolbar;
import android.view.Menu;
public class MainActivity extends AppCompatActivity implements Toolbar.OnMenuItemClickListener{
    private Toolbar toolbar;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        toolbar = (Toolbar) findViewById(R.id.ToolBar);
        toolbar.setNavigationIcon(R.mipmap.ic_launcher);
        toolbar.setLogo(R.mipmap.ic_launcher);
        toolbar.setTitle("ToolBar");

        toolbar.setSubtitle("toolbar");
        setSupportActionBar(toolbar);
        toolbar.setOnMenuItemClickListener(this);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.menu,toolbar.getMenu());
        return true;
    }
}




顯示Back圖示
toolbar = (Toolbar) findViewById(R.id.hack_toolbar);
        setSupportActionBar(toolbar);

        getSupportActionBar().setDisplayHomeAsUpEnabled(true);
        getSupportActionBar().setDisplayShowHomeEnabled(true);

toolbar.setNavigationOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                //do something you want
            }

        });

2017年9月24日

ViewTreeObserver

ViewTreeObserver

ViewTreeObserver vto =textViewLeft.getViewTreeObserver();vto.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {    @Override    public void onGlobalLayout() {        if (Build.VERSION.SDK_INT < 16) {            textViewLeft.getViewTreeObserver().removeGlobalOnLayoutListener(this);        }        else {            textViewLeft.getViewTreeObserver().removeOnGlobalLayoutListener(this);        }    }});

2017年8月1日

android E/eglCodecCommon 錯誤處理方式

android E/eglCodecCommon 錯誤處理方式

android studio 模擬器 api 22


錯誤訊息,重複出現
E/eglCodecCommon: **** ERROR unknown type 0x0 (glSizeof,73)
E/eglCodecCommon: glUtilsParamSize: unknow param 0x00000b44

處理方式
You can solve this by adding android:hardwareAccelerated="false" in AndroidManifest.xml

2017年5月8日

Android Dialog 標題換行

Android Dialog 標題換行

TextView view = new TextView(this);view.setText(appraisal.getTitle() + "\n" + appraisal.getNoticeInfo());view.setPadding(40, 20, 0, 20);view.setTextSize(12);
Dialog alertDialog = new AlertDialog.Builder(this).setCustomTitle(view);

2017年3月11日

Android 判斷手機或是平板

Android 判斷手機或是平板

public static boolean isPad(Context context) {
    return (context.getResources().getConfiguration().screenLayout
        & Configuration.SCREENLAYOUT_SIZE_MASK) >= Configuration.SCREENLAYOUT_SIZE_LARGE;
}

回傳true就是平板
回傳false就是手機

2017年2月20日

單例模式 Singleton

單例模式 Singleton

保證一個類別只有一個物件,而且要提供存取該物件的統一方法

public class FragmentOne extends Fragment {
    private static FragmentOne fragmentOne = null;

    private Singleton(){
    // 建立物件
    }

    public static FragmentOne getInstance() {
 
       if (fragmentOne == null) {
          // 多執行緒時,當物件需要被建立時才使用synchronized保證fragmentOne一定是單一的 ,增加程式校能
           synchronized(FragmentOne.class){
          if (fragmentOne == null) {
            // 第一次被呼叫的時候再建立物件
            fragmentOne = new FragmentOne();
          }
        }
        return fragmentOne;
    }

}



補充

package com.test;

public class TestSingletionPattern {

public static void main(String[] args) {
// TODO Auto-generated method stub
}

}

class Singleton {
private static final Singleton singleton = new Singleton();

private Singleton() {

}

public static Singleton getInstance() {
return singleton;
}

public static void doSomething() {

}
}

class SingletonB {
private static SingletonB singleton = new SingletonB();

private SingletonB() {

}

// 多執行緒時使用synchronized保證Singleton一定是單一的
public static synchronized SingletonB getInstance() {
if (singleton == null) {
singleton = new SingletonB();
}
return singleton;
}

public static void doSomething() {

}
}

class SingletonC {
private static SingletonC singleton = new SingletonC();

private SingletonC() {

}

// 多執行緒時,當物件需要被建立時才使用synchronized保證Singleton一定是單一的 ,增加程式校能
public static SingletonC getInstance() {
if (singleton == null) {
synchronized (SingletonC.class) {
if (singleton == null) {
singleton = new SingletonC();
}
}
}
return singleton;
}

public static void doSomething() {

}
}

2017年1月7日

Android get StatusBar Height and ActionBar Height


//取得狀態列高度
int statusBarHeight = 0;
int resource = this.getResources().getIdentifier("status_bar_height",
 "dimen", "android");
if (resource > 0) {
    statusBarHeight = this.getResources().getDimensionPixelSize(resource);
}


//取得標題列高度
int actionBarHeight=0;
TypedValue tv = new TypedValue();
if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true))
{
    actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,
getResources().getDisplayMetrics());
}

//應用區域高度
Rect outRect1 = new Rect();
getWindow().getDecorView().getWindowVisibleDisplayFrame(outRect1);

//VIEW繪製區域高度
Rect outRect2 = new Rect();
getWindow().findViewById(Window.ID_ANDROID_CONTENT).getDrawingRect(outRect2);