2015年8月13日 星期四

[Android] TextView 取代沒有資料的 ListView

有時候會發生沒有資料可以塞入 ListView ,此時需要一個 TextView 頂替:

<ListView
    android:id="@+id/listView"
    android:layout_width="match_parent"
    android:layout_height="0dp"
    android:layout_weight="1"/>

<TextView
    android:id="@+id/textView"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center_vertical|center_horizontal"
    android:text="no data"/>

接下來就是把 TextView 裝進 ListView:

ListView listView = (ListView) findViewById(R.id.listView);
listView.setEmptyView(findViewById(R.id.textView));
listView.setAdapter(adapter);



另外關於 layout_weight 的用法可以參考: Android 对Layout_weight属性完全解析以及使用ListView来实现表格

Android Studio 1.3
    compileSdkVersion 22
    buildToolsVersion "23.0.0 rc3"

2015年8月6日 星期四

[Android] EditText 做下拉選單

public class SpinnerView extends EditText
{
    private static class OnClickPopupSpinnerListener implements OnClickListener
    {
        private final SpinnerView spinnerView;
        private final List items;

        public OnClickPopupSpinnerListener(SpinnerView spinnerView, List items) {
            this.spinnerView = spinnerView;
            this.items = new ArrayList<>(items);
        }

        @Override
        public void onClick(View v) {
            final ListPopupWindow lpw = new ListPopupWindow(spinnerView.getContext());
            lpw.setOnItemClickListener(new AdapterView.OnItemClickListener() {
                @Override
                public void onItemClick(AdapterView parent, View view, int position, long id) {
                    String item = items.get(position);
                    spinnerView.setText(item);
                    lpw.dismiss();
                }
            });

            lpw.setAdapter(new ArrayAdapter(spinnerView.getContext(), android.R.layout.simple_list_item_1, items));
            lpw.setAnchorView(v);
            lpw.setModal(true);
            lpw.show();
        }
    }

    public SpinnerView(Context context) {
        super(context);
    }

    public SpinnerView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public SpinnerView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public SpinnerView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }

    public void setItems(List items) {
        setOnClickListener(new OnClickPopupSpinnerListener(this, items));
    }
}

在 layout.xml 就當作一般 EditText 使用。 在 activity 有設定 list 才會變成下拉選單。

@Override
protected void onCreate(Bundle savedInstanceState) {
    etSize = (SpinnerView) findViewById(R.id.editSize);
    etSize.setItems(Lists.newArrayList(new String[]{"small", "medium", "large"}));
    ...
}

參考資料:http://www.informit.com/articles/article.aspx?p=2078060&seqNum=4

Android Studio 1.3
    compileSdkVersion 22
    buildToolsVersion "23.0.0 rc3"

[Android] 點擊按鍵時關閉鍵盤

如果關注 EditText,點擊按鍵時鍵盤沒有自動關閉, 可以在 onClick 方法或是 Listener:

public void onClickButton(View view) {
    InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
    imm.hideSoftInputFromWindow(findViewById(R.id.button).getWindowToken(), 0);
}

findViewById(R.id.button).setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View v) {
        InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
        imm.hideSoftInputFromWindow(findViewById(R.id.button).getWindowToken(), 0);
    }
};


根據 API : public boolean hideSoftInputFromWindow (IBinder windowToken, int flags), flags 僅建議使用 0 或 HIDE_IMPLICIT_ONLY。

flags : Provides additional operating flags. Currently may be 0 or have the HIDE_IMPLICIT_ONLY bit set.

參考資料:

Android Studio 1.3
    compileSdkVersion 22
    buildToolsVersion "23.0.0 rc3"

2015年8月5日 星期三

[Android] ActionBar 顯示返回鍵 homeAsUp

先在 AndroidManifest.xml 設定 activity 關係:

<application ...>
    <activity android:name=".MainActivity" ...>
    </activity>
    <activity android:name=".SecondActivity"
              android:parentActivityName=".MainActivity" ...>
    </activity>
</application>
在 Activity 設定:
@Override
protected void onCreate(Bundle savedInstanceState) {
    ActionBar actionBar = getSupportActionBar();
    actionBar.setDisplayHomeAsUpEnabled(true);
}



網路上的方法不需要添加:

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
    case R.id.home:
        finish();
        return true;
    }
    return super.onOptionsItemSelected(item);
}
經過測試,id 既不是 R.id.home,也不是 R.id.homeAsUp, 應該是透過 AndroidManifest.xml 綁定 parent id。



前面返回時,是重建 Intent 轉回去呼叫 MainActivity#onCreate(),所以資料會都丟掉。 避免這種情況可以複寫這個方法:

@Override
public Intent getSupportParentActivityIntent() {
    finish();
    return null;
}

參考資料:

Android Studio 1.3
    compileSdkVersion 22
    buildToolsVersion "23.0.0 rc3"

    com.android.support:appcompat-v7:22.2.+

2015年8月4日 星期二

[Android] AsyncTask 使用 Toast

在 #doInBackground() 抓檔時,會想要即時用 Toast 顯示訊息。 但是會造成錯誤:

java.lang.RuntimeException: An error occured while executing doInBackground()
...

可行的方法是在 #postExecute() 才處理。

private Exception exception;

@Override
protected Result doInBackground(Params... requests) {
   try {
   } catch (Exception e) {
      // Toast.makeText(context, "error occurs", Toast.LENGTH_LONG).show();
      exception = e;
   }
}

@Override
protected void onPostExecute(List googleImages) {
   if (exception != null) {
      Toast.makeText(context, "error occurs", Toast.LENGTH_LONG).show();
   }
}

參考資料:StackOverflow: java.lang.RuntimeException: An error occured while executing doInBackground()

Android Studio 1.3
    compileSdkVersion 22
    buildToolsVersion "23.0.0 rc3"

2015年8月3日 星期一

[Android] ActionBar 顯示 app icon

API 21 之後預設移除 app icon。

android.support.v7.widget.Toolbar

The use of application icon plus title as a standard layout is discouraged on API 21 devices and newer.

android.support.v7.app.ActionBar

When using the Material themes (default in API 21 or newer) the navigation button (formerly "Home") takes over the space previously occupied by the application icon.

設定 app icon :

protected void onCreate(Bundle savedInstanceState) {
   ActionBar actionBar = getSupportActionBar();
   actionBar.setLogo(R.drawable.ic_laucher);
   actionBar.setDisplayUseLogoEnabled(true);
   actionBar.setDisplayShowHomeEnabled(true);
}

⚠ icon 需要在 drawable 目錄,不能使用 mipmap。
⚠ 3 個方法都要寫。

參考資料:StackOverflow: The application icon does not show on action bar

Android Studio 1.3
    compileSdkVersion 22
    buildToolsVersion "23.0.0 rc3"

    com.android.support:appcompat-v7:22.2.+

[Java] Invalid HTTP method: PATCH

最近系統需要使用 Netty4,所以把衝突的 Netty3 拆掉,然後就出現了例外。 pom.xml <dependency> <groupId>com.ning</groupId> <artifactId>as...