Git GitExtensionsでwindowsに環境を作成する

少人数のチームでネットワークで繋がったWindowsマシンに
GitとGit Extensionsを使用してバージョン管理ができる環境を作ります。

環境


サーバー(windows)
開発者A用のローカルマシン(windows)
開発者B用のローカルマシン(windows)
それぞれのマシンはネットワークで繋がっているものとします。

インストール


Git for Windows
http://msysgit.github.com/
2012/12/26現在の最新版:Git-1.8.0-preview20121022.exe
上記のサイトよりGit for Windowsをダウンロードします。

Git Extensions
Gitを操作するためのGUIツールです。
http://code.google.com/p/gitextensions/
2012/12/26現在の最新版:GitExtensions243Setup.msi
上記のサイトよりGit Extensionsをダウンロードします。

ダウンロードしたそれぞれのファイルを3つのマシンにインストールします。
インストールはデフォルトのままです。

サーバーにリモートリポジトリを作成する


リポジトリを配置するフォルダを作成し、共有設定をしておきます。
私はDドライブ直下にGitRepoフォルダを作成し、共有設定をしました。
フォルダ名に日本語が含まれると操作に失敗します。

次にGitExtensionsを起動します。
設定画面が起動します。
「チェックリスト」タブの「起動時に設定を確認」がチェックONだとGitExtensionsを起動するたびに表示されます。
特に設定する項目はないので「OK」ボタンでダイアログを閉じます。

GitExtensionsの「リポジトリの作成」を選択します。
「リポジトリの作成」ダイアログで
ディレクトリに「D:¥GitRepo¥プロジェクト名など」を入力し
「Working dirを持たないCentralリポジトリ」を選択し
作成ボタンでリポジトリを作成します。
「Initialized empty shared Git repository in D:/GitRepo/MyProjectName/」とメッセージが表示れるのでOKボタンで進みます。
「D:/GitRepo/MyProjectName」フォルダにいくつかファイルができます。
これで共有リポジトリの作成は終了です。

1台目のローカルマシン


1台目のローカルマシーンでGitExtensionsを起動します。
こちらも設定画面が表示されますが、特に設定はないので終了します。

GitExtensionsの「リポジトリのclone」を選択します。
「クローン」ダイアログで
クローンするリポジトリ:先ほど作成した共有リポジトリのパス
クローン先:ローカルマシンのリポジトリを作成するフォルダ
サブディレクトリの作成:プロジェクト名など
リポジトリの種類:個人リポジトリ を選択し
「クローン」ボタンでリポジトリをクローンします。
フォルダ名に日本語が含まれると操作に失敗します。
進捗ダイアログが表示されるのでOkボタンで進みます。
リポジトリを開くかメッセージが表示されるので「はい」を選択します。
「.gitignoreの編集」ボタンをクリックし、開いたダイアログでバージョン管理に含めないファイルの拡張子を登録します。
VisualStudioであれば「デフォルトの無視ファイル」で追加されるパターンで大丈夫そうです。
保存ボタンでダイアログを終了します。
「コミット」ボタンで変更した「.gitignore」ファイルをコミットします。
「作業ディレクトリの変更点」に表示されている「.gitignore」を「Stage」ボタンでステージエリアに移動させます。
コミットメッセージを入力し、「コミット」をクリックします。
進捗ダイアログが表示されるので「OK」ボタンで進みます。
コミット内容を表示する画面が表示されます。
次にプログラムをコミットします。
ファイルエクスプローラからE:/GitRepo/MyProjectNameにプログラムをドラッグします。
そうするとGitExtensionsのコミットの色が青から赤に変わります。
ファイルを変更し、コミットされていないファイルがあれば知らせてくれます。
「Gitコマンド」 > 「コミット」で追加したプログラムをコミットします。
「作業ディレクトリの変更点」に表示されているプログラムファイルを「StageAll」ボタンでステージエリアに移動します。
コミットメッセージを入力し「コミット」ボタンをクリックします。
進捗ダイアログが表示されるので「OK」ボタンで進みます。
現在のコミット状態
コミットしたプログラムをリモートリポジトリに送ります。
「Gitコマンド」 > 「Push(リモートへ反映)」で開いたダイアログより
リモート「origin」でブランチ「master」から「master」で「Push」します。
これでローカルブランチの内容をリモートブランチへ送信します。
「はい」を選択。
以上で1台目のローカルマシンからの設定は終了です。

2台目のローカルマシーン

2台目のローカルマシーンでGitExtensionsを起動します。
こちらも設定画面が表示されますが、特に設定はないので終了します。

GitExtensionsの「リポジトリのclone」を選択します。
「クローン」ダイアログで
クローンするリポジトリ:先ほど作成した共有リポジトリのパス
クローン先:ローカルマシンのリポジトリを作成するフォルダ
サブディレクトリの作成:プロジェクト名など
1台目と違ってブランチは「master」を選択します。
リポジトリの種類:個人リポジトリ を選択し
「クローン」ボタンでリポジトリをクローンします。
フォルダ名に日本語が含まれると操作に失敗します。
進捗ダイアログが表示されるので「OK」ボタンで先に進みます。
リポジトリを開くかメッセージが表示されるので「はい」を選択します。
1台目のローカルマシーンでコミットした内容が表示れます。

これで2台目のマシーンの設定は終了です。

Git GitExtensionsをインストールするとVisualStudioのセットアップが作成できない

GitとGitExtensionsをインストールしたら、VisualStudioのSetUpプロジェクトがビルドできなくなってしまいました。
SetUpでユーザのデスクトップにショートカットを作成しているのですが、
ショートカットにアイコンをセットした状態でビルドすると
「'[DesktopFolder]' にあるショートカット 'ProjectName (アクティブ) の プライマリ出力 へのショートカット' をビルドできません。」
とエラーになってしまいます。


解決方法は
VisualStudioの「ツール」メニュー「アドインマネージャ」で「Git Extemsions」のチェックをOFFにします。
VisualStudioのメニューバーに「Git」が表示されていますが、チェックOFFにすると「Git」が表示されなくなります。
「Git」メニューが表示されている場合はVisualStudioを再起動してください。
「Git」メニューがない状態でビルドするとエラーはでなくなりました。

Android CheckBoxの画像を変更する(BackGround) その2

前回「Android CheckBoxの画像を変更する その1」でコチラ「Android GridView CheckBox付きの画像を表示」で作成したチェックボックスの画像を変更してみました。
CheckBoxを画像の右上に配置したいのですが、これ以上動きません。

実はCheckBox画像の右横にもう一つ画像があるのです。
CheckBoxのbackgroundに設定されている@android:drawable/btn_check_label_background は透明の画像で
チェック画像とテキストの間隔を調整しているようです。
ということでbackgroundに@nullを設定してみました。
一つ目のCheckBoxはデフォルトの状態です。
二つ目のCheckBoxはtextに値を設定し、backgroundに@nullを設定しています。
三つ目のCheckBoxはtextに値を設定せず、backgroundに@nullを設定していますが、表示されません。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity" >

    <CheckBox
        android:id="@+id/checkBox1"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="CheckBox" />
    
    <CheckBox
        android:id="@+id/CheckBox02"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="CheckBox" 
        android:background="@null" />   

    <CheckBox
        android:id="@+id/CheckBox03"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
         android:text=""
        android:background="@null" />

</LinearLayout>
btn_check_label_backgroundに替わる画像を作ってあげればよさそうですが、
検索すると「Y.A.M の 雑記帳」さんの Android RadioButton の画像とテキストの間隔を広げる でshapeを使った方法が紹介されていました。

バックグランドに設定するShapeを定義する

res/drawableに「custom_checkbox_background.xml」を作成します。
<?xml version="1.0" encoding="utf-8"?>
<shape  xmlns:android="http://schemas.android.com/apk/res/android" android:shape="rectangle">
     <solid android:color="#A9A9A9" />
     <padding        
         android:left="22dip"        
         android:top="0dip"        
         android:right="0dip"        
         android:bottom="0dip" />
</shape>

styleを定義する

res/values/styles.xmlを編集し、CheckBoxに設定するスタイルを追記します。
<resources>

 <style name="CustomCheckBox" parent="android:Widget.CompoundButton.CheckBox">
  <item name="android:button">@drawable/custom_checkbox</item>
  <item name="android:background">@drawable/custom_checkbox_background</item>
 </style>
    
</resources>

CheckBoxにStyleを設定する

res/layout/activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity" >

    <CheckBox
        android:id="@+id/checkBox1"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="CheckBox" />
    
    <CheckBox
        android:id="@+id/CheckBox02"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="" />   

</LinearLayout>

実行してみた



GridViewのCheckBoxにも適用してみました。
右上に配置できています。
shapeの角を丸くして背景色をグラデーションにすればいい感じじゃないでしょうか。

Android CheckBoxの画像を変更する その1

ここ「Android Wiki ボタンの画像変更(CheckBox,RadioButton)」を見れば一発だけど、一応メモ。

画像を用意する

まずチェックボックスの4つの状態を表す画像を用意します。
画像はres/drawable-nodpiに配置します。
  • チェックONかつ選択されていない状態の画像 [checkbox_checked]


  • チェックONかつ選択されている状態の画像 [checkbox_checked_selected]


  • チェックOFFかつ選択されていない状態の画像 [checkbox_unchecked]


  • チェックOFFかつ選択されている状態の画像 [checkbox_unchecked_selected]

画像を定義する

res/drawableフォルダに「custom_checkbox.xml」を作成します。
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <!-- チェックON かつ 選択されている状態 -->
    <item
        android:state_checked="true"
        android:state_pressed="true"
        android:drawable="@drawable/checkbox_checked_selected" />
    <!-- チェックON かつ 選択されていない状態-->
    <item
        android:state_checked="true"
        android:drawable="@drawable/checkbox_checked" />
    <!-- チェックOFF かつ 選択されている状態-->
    <item
        android:state_checked="false"
        android:state_pressed="true"
        android:drawable="@drawable/checkbox_unchecked_selected" />
    <!-- チェックOFF かつ 選択されていない状態-->
    <item
        android:state_checked="false"
        android:drawable="@drawable/checkbox_unchecked" />
</selector>

styleを定義する

res/values/styles.xmlを編集し、CheckBoxに設定するスタイルを追記します。
<resources>
    <style name="CustomCheckBox" parent="android:Widget.CompoundButton.CheckBox">
        <item name="android:button">@drawable/custom_checkbox</item>
    </style>
</resources>

CheckBoxにStyleを設定する

res/layout/activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity" >

    <CheckBox
        android:id="@+id/CheckBox03"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="CheckBox" />

    <CheckBox
        android:id="@+id/CheckBox02"
        style="@style/CustomCheckBox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:checked="true"
        android:text="CheckBox" />

</LinearLayout>

実行してみた

ハイ。ちゃんとできました。

そしてハマった。

で、以前コチラ「Android GridView CheckBox付きの画像を表示」で作成したチェックボックスに適用してみたのですよ。
私はCheckBoxを画像の右上に配置したいのに、これ以上ガンとして動きません。
レイアウトをいじり倒しても動かないし・・・結構ハマりました・・・

実はCheckBox画像の右横にもう一つ画像があるのですよ。
次回はこの画像をどうにかしたいと思います。

Android ActionBarとFragmentを利用したTab画面でデータを(GridView)表示する

以前作成したTab画面に、以前作成したSDカード内の画像をチェックボックス付きで表示するGridViewを乗せてみました。
Android ActionBarとFragmentを使用してTab画面を表示する(Android 2.x)
Android GridView CheckBox付きの画像を表示

Fragment再作成時にデータを保存・復元する


まずグリッドの個々のアイテムであるチェックボックスと画像を管理するクラスCheckedImageに
Parcelableインターフェースを実装しBundleに追加できるようにします。
boolean型をParcelableで保存するには  dest.writeByte((byte) (mBoolean ? 1 : 0));  とします。
boolean型をParcelableで復元するには  mBoolean = (in.readByte() == 1);  とします。
import android.content.Context;
import android.graphics.Bitmap;
import android.os.Parcel;
import android.os.Parcelable;
import android.provider.MediaStore;

public class CheckedImage implements Parcelable{
    private boolean mChecked = false;
    private long mBitmapId;
    private Bitmap mBitmap = null;

    public boolean getChecked() {
        return mChecked;
    }
    public void setChecked(boolean checked) {
        this.mChecked = checked;
    }
    public long getBitmapId() {
        return mBitmapId;
    }
    public void setBitmapId(long bitmapId) {
        this.mBitmapId = bitmapId;
    }
    public Bitmap getBitmap(Context context) {
        if (mBitmap == null){
            mBitmap = MediaStore.Images.Thumbnails.getThumbnail(context.getContentResolver(), mBitmapId, MediaStore.Images.Thumbnails.MICRO_KIND, null);
        }
        return mBitmap;
    }
    /**
     * コンストラクタ
     * @param checked
     * @param bitmapId
     */
    public CheckedImage(boolean checked, long bitmapId){
        mChecked = checked;
        mBitmapId = bitmapId;
    }

    /*****************************
     * Parcelable実装のために必要
     *****************************/
    public CheckedImage(Parcel in) {
        // 必ず read と write は同じ順番で入れること。  
        mChecked = (in.readByte() == 1);
        mBitmapId = in.readLong();
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        // 必ず read と write は同じ順番で入れること。  
        dest.writeByte((byte) (mChecked ? 1 : 0));
        dest.writeLong(mBitmapId);    
    }

    @Override
    public int describeContents() {
        // ファイルディスクリプターを使用しないのであれば常に0
        return 0;
    }

    // これは定型文なのでそのまま使える。
    public static final Creator CREATOR = new Creator() {
        public CheckedImage createFromParcel(Parcel in) {
            return new CheckedImage(in);
        }
        public CheckedImage[] newArray(int size) {
            return new CheckedImage[size];
        }
    };


}

次にデータを保存・復元する際に便利なようにArrayAdapterを改良し
adapterに設定されているデータを取得するメソッドgetItemList()を追加します。
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.CheckBox;
import android.widget.CompoundButton;
import android.widget.ImageView;
import android.widget.CompoundButton.OnCheckedChangeListener;


public class CheckedImageArrayAdapter extends ArrayAdapter<CheckedImage> {
    private static class ItemView {
        ImageView imageview = null;
        CheckBox checkbox = null;
    }
    
    private final static int LAYOUT_ID = R.layout.checked_image; 
    
    private Context mContext;
    private LayoutInflater mInflater;   
    private List<CheckedImage> mItemList;

    /**
     * コンストラクタ
     * @param context
     * @param objects
     */
    public CheckedImageArrayAdapter(Context context, List<CheckedImage> itemlist) {
        super(context, LAYOUT_ID, itemlist);
        mContext = context;
        mItemList = itemlist;
        mInflater = (LayoutInflater)mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE); 
    }
    
    /**
     * チェックボックスのチェックONリスナー
     */
    private OnCheckedChangeListener CheckBox1_OnCheckedChangeListener = new OnCheckedChangeListener(){
        public void onCheckedChanged(CompoundButton buttonView,    boolean isChecked) {
            int position =  (Integer)buttonView.getTag();
            CheckedImage item = CheckedImageArrayAdapter.this.getItem(position);
            item.setChecked(isChecked);    
        }};

    /**
     * adapterに設定されているデータを取得する
     * @return
     */
    public ArrayList<CheckedImage> getItemList() {
        return (ArrayList<CheckedImage>) mItemList;
    }
 
   /**
     * adapterに設定されているデータのうちチェックONのデータを取得する
     * @return
     */   
    public List<CheckedImage> getCheckedItem(){
        List<CheckedImage> lstItem = new ArrayList<CheckedImage>();
        for ( int i = 0; i < getCount(); i++) {
            if (getItem(i).getChecked()){
                lstItem.add(getItem(i));
            }
        }
        return lstItem;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ItemView itemview;
        if (convertView == null) {
            convertView = mInflater.inflate(LAYOUT_ID, null);
            itemview = new ItemView();
            itemview.imageview = (ImageView) convertView.findViewById(R.id.imageView1);
            itemview.checkbox = (CheckBox) convertView.findViewById(R.id.checkBox1);
            itemview.checkbox.setOnCheckedChangeListener(CheckBox1_OnCheckedChangeListener);
            convertView.setTag(itemview);
        } else {
            itemview = (ItemView)convertView.getTag();
        }

        CheckedImage item = getItem(position);
        itemview.checkbox.setTag(position);
        itemview.imageview.setImageBitmap(item.getBitmap(mContext));
        itemview.checkbox.setChecked(item.getChecked());

        return convertView;
    }   
}
次にFragmentです。
縦横切り替えしたときにFragmentの再作成をしたくないので、コンストラクタで  setRetainInstance(true); を設定します。
Fragmentを再作成したときデータを復元できるように、onSaveInstanceStateでadapterに設定してるデータを保存します。
タブページを変更したとき、データを再取得したくないのとチェック状態を保持しておきたいので、onCreateでグリッドに設定するadapterを作成します。 このときonCreateの引数savedInstanceStateに復元データがあれば、そのデータを使用するようにします。
import java.util.ArrayList;
import java.util.List;

import android.content.ContentResolver;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteStatement;
import android.net.Uri;
import android.os.Bundle;
import android.provider.MediaStore;
import android.support.v4.app.Fragment;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.GridView;



public class MainTab1Fragment extends Fragment {
   
    private static final String CHECKED_IMAGE_LIST = "CheckedImageList";

    private GridView mGridView = null;
    private Button mButtonSelect = null;
    private CheckedImageArrayAdapter mAdapter = null;

    public MainTab1Fragment(){
        setRetainInstance(true);
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {    
        super.onCreate(savedInstanceState);

        if (mAdapter == null) {
            if (savedInstanceState != null) {
                @SuppressWarnings("unchecked")
                List<CheckedImage> lstItem = (List<CheckedImage>) savedInstanceState.get(CHECKED_IMAGE_LIST);
                mAdapter = new CheckedImageArrayAdapter(getActivity(), lstItem);
            } else {
                //SDカードより画像データのIDを取得
                ContentResolver cr = getActivity().getContentResolver();
                Uri uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; //SDカード       
                Cursor cursor = cr.query(uri, null, null, null, null);        
                List<CheckedImage> lstItem = new ArrayList<CheckedImage>();
                cursor.moveToFirst();   
                for (int i = 0; i < cursor.getCount(); i++){
                    long id = cursor.getLong(cursor.getColumnIndexOrThrow("_id"));
                    lstItem.add(new CheckedImage(false,id));
                    cursor.moveToNext();
                }    
                //グリッド用のアダプターを作成
                mAdapter = new CheckedImageArrayAdapter(getActivity(), lstItem);
            };
        };
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        // 第3引数のbooleanは"container"にreturnするViewを追加するかどうか
        //trueにすると最終的なlayoutに再度、同じView groupが表示されてしまうのでfalseでOKらしい
        View view = inflater.inflate(R.layout.fragment_main_tab1, container, false);
        mGridView = (GridView)view.findViewById(R.id.gridView);
        mButtonSelect = (Button)view.findViewById(R.id.select);
        mButtonSelect.setOnClickListener(ButtonSelect_OnClickListener);    
        //グリッドにアダプターをセット
        mGridView.setAdapter(mAdapter);
        return view;
    }

    @Override
    public void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        outState.putParcelableArrayList(CHECKED_IMAGE_LIST, mAdapter.getItemList());        
    }


}

Android ActionBarとFragmentを利用したTab画面でFragment#OnCreateが2回走る

結構ハマった・・・。

以前作ったActionBarを使ったTab画面で
Android ActionBarとFragmentを使用してTab画面を表示する(Android 2.x)
Android ActionBarとFragmentを使用してTab画面を表示する(Android 4.0以上)
エミュレータの縦横を切り替えると、描画が2重になっているような現象になってました。
FragmentのOnCreateが2回呼ばれていて、Fragmentが2つ重なっているような感じ。

原因はTabLisnerでした。
エミュレータの縦横を切り替えると、Activityが再作成される。
このときFragmentは自動で復元されており1回目のonCreateが走る。
で、Activityが再作成されたので、TabLisnerも再作成されてonTabSelectedが走る。
この時インスタンス変数mFragmentはnullなのでFragmentManagerに再度addされ
2つ目のFragmentがインスタンス化され2回目のonCreateが走る。
たぶん・・・こんな感じ・・・?

なのでコンストラクタでFragmentを探すように変更し
onTabSelectedでもdetachされていないときだけattachするよう変更しました

コードは Android ActionBarとFragmentを使用してTab画面を表示する(Android 2.x)
のTabListenerです。
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentTransaction;

import com.actionbarsherlock.app.ActionBar;
import com.actionbarsherlock.app.ActionBar.Tab;
import com.actionbarsherlock.app.SherlockFragmentActivity;

public class TabListener implements ActionBar.TabListener {

    private Fragment mFragment;
    private final SherlockFragmentActivity mActivity;
    private final String mTag;
    private final Class mClass;
    
    /**
     * @brief コンストラクタ
     * @param activity
     * @param tag
     * @param clz
     */
    public TabListener(SherlockFragmentActivity activity, String tag, Class clz) {
        mActivity = activity;
        mTag = tag;
        mClass = clz;
        //FragmentManagerからFragmentを探す。
        mFragment = mActivity.getSupportFragmentManager().findFragmentByTag(mTag);
    }

    /**
     * @brief  タブが選択されたときの処理
     */
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        //ftはnullではないが使用するとNullPointExceptionで落ちる
        if (mFragment == null) {
            mFragment = Fragment.instantiate(mActivity, mClass.getName());
            FragmentManager fm = mActivity.getSupportFragmentManager();
            fm.beginTransaction().add(R.id.container, mFragment, mTag).commit();
        } else {
         //detachされていないときだけattachするよう変更
            if (mFragment.isDetached()) {
                FragmentManager fm = mActivity.getSupportFragmentManager();
                fm.beginTransaction().attach(mFragment).commit();
             }
        }
    }
    /**
     * @brief  タブの選択が解除されたときの処理
     */
    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        //ftはnullではないが使用するとNullPointExceptionで落ちる
        if (mFragment != null) {
            FragmentManager fm = mActivity.getSupportFragmentManager();
            fm.beginTransaction().detach(mFragment).commit();
       }    
    }
    /**
     * @brief タブが2度目以降に選択されたときの処理
     */
    public void onTabReselected(Tab tab, FragmentTransaction ft) {
    }
}