質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.87%

  • Android

    6085questions

    Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

  • Android Studio

    3357questions

    Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

gridViewの各セルにはりつけたlayaout内のテキストビューを書き換えたい。

受付中

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 72

dabodaboda29

score 2

 前提・実現したいこと

androidでシフト管理アプリを開発しています。
シフトを登録したボタンをクリック→シフトを記入したい日付セルをクリック、でカレンダー上にシフトがのるようになっています。
最後に日付をキーにしてプレファレンスでシフトを記録、参照してOnResumeでシフトが記入されたカレンダーを復元したいのです。
このとき、calenderGridViewにinflateしているcalender_cellのテキスト(tx_shift)を変更したいのですが、viewが指定できず、nullPointExceptionが帰ってきてしまいます。

どのようにすれば、目的のviewを取得できるでしょうか。
また、その他にカレンダーを復元する方法はあるでしょうか。

 発生している問題・エラーメッセージ

FATAL EXCEPTION: main
    Process: com.example.shiftcalender, PID: 25808
    java.lang.RuntimeException: Unable to resume activity {com.example.shiftcalender/com.example.shiftcalender.MainActivity}: java.lang.NullPointerException: Attempt to invoke virtual method 'android.view.View android.view.View.findViewById(int)' on a null object reference
        at android.app.ActivityThread.performResumeActivity(ActivityThread.java:3158)
        at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:3189)
        at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2527)
        at android.app.ActivityThread.access$900(ActivityThread.java:154)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1389)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:224)
        at android.app.ActivityThread.main(ActivityThread.java:5514)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
     Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'android.view.View android.view.View.findViewById(int)' on a null object reference
        at com.example.shiftcalender.MainActivity.onResume(MainActivity.java:159)
        at android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1259)
        at android.app.Activity.performResume(Activity.java:6346)
        at android.app.ActivityThread.performResumeActivity(ActivityThread.java:3147)
        at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:3189) 
        at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2527) 
        at android.app.ActivityThread.access$900(ActivityThread.java:154) 
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1389) 
        at android.os.Handler.dispatchMessage(Handler.java:102) 
        at android.os.Looper.loop(Looper.java:224) 
        at android.app.ActivityThread.main(ActivityThread.java:5514) 
        at java.lang.reflect.Method.invoke(Native Method) 
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726) 
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616) 

 該当のソースコード

 protected void onResume() {
        super.onResume();

        calendarGridView =findViewById(R.id.calendarGridView);
        mCalendarAdapter = new CalendarAdapter(this);
        calendarGridView.setAdapter(mCalendarAdapter);

        View view = mCalendarAdapter.getPositionView(1);

        date = mCalendarAdapter.getDateText(1);
        tempShift = CalendarRead(date);

        if(tempShift != null){
            tx_shift = view.findViewById(R.id.tx_shift); //エラーの出る159行目
            tx_shift.setText(tempShift);
            Toast.makeText(this, tempShift,Toast.LENGTH_LONG).show();
        }else{
            Toast.makeText(this, "null",Toast.LENGTH_LONG).show();
        }
}

public void CalendarSave(String date, String shiftPT){
        SharedPreferences pref = getSharedPreferences("CalendarSave", MODE_PRIVATE);
        SharedPreferences.Editor editor = pref.edit();
        editor.putString(date, shiftPT);
        editor.commit();
}

public String CalendarRead(String date){
        SharedPreferences pref = getSharedPreferences("CalendarSave", MODE_PRIVATE);
        Toast.makeText(this, pref.getString(date,null),Toast.LENGTH_LONG).show();
        return pref.getString(date,null);
}
public class CalendarAdapter extends BaseAdapter {
    private List<Date> dateArray = new ArrayList();
    private Context mContext;
    private DateManager mDateManager;
    private LayoutInflater mLayoutInflater;
    private Map<Integer, View> positionView;

    //カスタムセルを拡張したらここでWidgetを定義
    private static class ViewHolder{
        public TextView dateText,tx_shift;
    }
public CalendarAdapter(Context context){
        mContext = context;
        mLayoutInflater = LayoutInflater.from(mContext);
        mDateManager = new DateManager();
        dateArray = mDateManager.getDays();
        positionView = new HashMap<Integer, View>(36);
}

@Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        if (convertView == null){
            convertView = mLayoutInflater.inflate(R.layout.calendar_cell, null);
            holder = new ViewHolder();
            holder.dateText = convertView.findViewById(R.id.dateText);
            holder.tx_shift = convertView.findViewById(R.id.tx_shift);
            convertView.setTag(holder);
        }else {
            holder = (ViewHolder) convertView.getTag();
        }

        //セルのサイズを指定
        float dp = mContext.getResources().getDisplayMetrics().density;
        AbsListView.LayoutParams params = new AbsListView.LayoutParams(parent.getWidth()/7 - (int)dp, (parent.getHeight()/5 - (int)dp));
        convertView.setLayoutParams(params);

        //日付を月/日で表示
        SimpleDateFormat dateFormat = new SimpleDateFormat("M/d", Locale.US);
        holder.dateText.setText(dateFormat.format(dateArray.get(position)));

        positionView.put(position, convertView);

        return convertView;
}

 public View getPositionView(int position){
        View targetView = positionView.get(position);
        return targetView;
}

public String getDateText(int position){
        SimpleDateFormat dateFormat = new SimpleDateFormat("M/d", Locale.US);
        return dateFormat.format(dateArray.get(position)).toString();
}

 試したこと

gridViewからgetChildAtで取れないかと試しましたが、これもnullが帰ってくる。
viewが取れないのであればと、calenderAdapterのgetView内でプレファレンスを調べて、データがあればtx_shiftに代入、と思いましたがgetSharedPrefarencesが使えず…。
手探りで開発しているので的外れな部分が多々あるかと思いますが、何卒よろしくお願いいたします。

 補足情報(FW/ツールのバージョンなど)

Android Studio 3.1.2

大元のソースはこちらを参考にさせていただいてます。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

0

GridViewなどのAdapterをセットする形式のViewは画面内に表示できるだけの子Viewを持つようになっているため、子Viewを直接取得して表示を書き換えるのではなく、Adapterにデータの更新を伝えることで子Viewの更新を促すようにすべきです。

ご質問のコードのCalendarAdapter内にある positionView のように、子Viewにアクセスするためのキャッシュを持っておくのはアンチパターンなので削除しましょう。その代わりに、tx_shiftに関連するデータをAdapterに持たせ、 getView() 内で子Viewの設定をすべて行うようにします。Activity側では、Adapter内のデータを変更して、変更を通知するために notifyDataSetChanged() を呼べばOKです。

// 修正例
public class CalendarAdapter extends BaseAdapter {

    private String tempShift = null;

    // 外部からtempShiftを設定するためのメソッド
    public void setTempShift(String tempShift) {
        this.tempShift = tempShift;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ...

        // tx_shiftの設定(ロジックは適当なので調整してください)
        if (position == 1 && tempShift != null) {
            // tempShiftがnullでなく、1番目のViewなら表示
            holder.tx_shift.setText(tempShift);
        } else {
            // それ以外なら空文字を表示
            holder.tx_shift.setText("");
        }

        return convertView;
    }    

}

// Activity側
protected void onResume() {
    super.onResume();

    calendarGridView = findViewById(R.id.calendarGridView);
    mCalendarAdapter = new CalendarAdapter(this);
    calendarGridView.setAdapter(mCalendarAdapter);

    date = mCalendarAdapter.getDateText(1);
    tempShift = CalendarRead(date);

    // tempShiftをセットし、データの更新を伝える
    mCalendarAdapter.setTempShift(tempShift);
    mCalendarAdapter.notifyDataSetChanged();
}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.87%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る

  • Android

    6085questions

    Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

  • Android Studio

    3357questions

    Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。