1

電話の内部カレンダーにイベントを追加するメソッドを使用しています。文脈はいつも私を混乱させてきました、そして今回も違いはありません。メソッドコントラクトには、アプリケーションコンテキストが必要です。メソッド呼び出しは次のとおりです。

addToCalendar(Context context, String title, long dtstart, long dtend);

これは、ボタンが押されたときにメソッドを呼び出すための私のコードです。

public class DateAdder extends Activity {
String shiftType;
Date d = new Date();


/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);
    setContentView(R.layout.main); 




    Button AShift = (Button) findViewById(R.id.AShift);
    AShift.setOnClickListener(new View.OnClickListener(){
        public void onClick(View v) {
            TextView BtmTxt = (TextView) findViewById(R.id.BtmTxt);
            shiftType="A Shift";
            addToCalendar(DateAdder.this.getApplicationContext(), shiftType, d.getDate(), d.getDate()+1000);                
            BtmTxt.setText("Done");
        }});
} //END of onCreate



/**
 * Adds the event to a calendar. It lets the user choose the calendar
 * @param ctx Context ( Please use the application context )
 * @param title title of the event
 * @param dtstart Start time: The value is the number of milliseconds since Jan. 1, 1970, midnight GMT.
 * @param dtend End time: The value is the number of milliseconds since Jan. 1, 1970, midnight GMT.
 */
private static void addToCalendar(Context ctx, final String title, final long dtstart, final long dtend) {
    final ContentResolver cr = ctx.getContentResolver();
    Cursor cursor ;
    if (Integer.parseInt(Build.VERSION.SDK) == 8 )
        cursor = cr.query(Uri.parse("content://com.android.calendar/calendars"), new String[]{ "_id", "displayname" }, null, null, null);
    else
        cursor = cr.query(Uri.parse("content://calendar/calendars"), new String[]{ "_id", "displayname" }, null, null, null);
    if ( cursor.moveToFirst() ) {
        final String[] calNames = new String[cursor.getCount()];
        final int[] calIds = new int[cursor.getCount()];
        for (int i = 0; i < calNames.length; i++) {
            calIds[i] = cursor.getInt(0);
            calNames[i] = cursor.getString(1);
            cursor.moveToNext();
        }

        AlertDialog.Builder builder = new AlertDialog.Builder(ctx);
        builder.setSingleChoiceItems(calNames, -1, new DialogInterface.OnClickListener() {

            public void onClick(DialogInterface dialog, int which) {
                ContentValues cv = new ContentValues();
                cv.put("calendar_id", calIds[which]);
                cv.put("title", title);
                cv.put("dtstart", dtstart );
                cv.put("dtend", dtend);
                cv.put("allday", 1);
                cv.put("hasAlarm", 0);

                Uri newEvent ;
                if (Integer.parseInt(Build.VERSION.SDK) == 8 )
                    newEvent = cr.insert(Uri.parse("content://com.android.calendar/events"), cv);
                else
                    newEvent = cr.insert(Uri.parse("content://com.android.calendar/events"), cv);

                if (newEvent != null) {
                    long id = Long.parseLong( newEvent.getLastPathSegment() );
                    ContentValues values = new ContentValues();
                    values.put( "event_id", id );
                    values.put( "method", 1 );
                    values.put( "minutes", 15 ); // 15 minuti
                    if (Integer.parseInt(Build.VERSION.SDK) == 8 )
                        cr.insert( Uri.parse( "content://com.android.calendar/reminders" ), values );
                    else
                        cr.insert( Uri.parse( "content://calendar/reminders" ), values );

                }
                dialog.cancel();
            }

        });

        builder.create().show();
    }
    cursor.close();
}

}

コードからこれを呼び出すと、力が近くなります。logcatは次のように読みます。

android.view.WindowManager$BadTokenException: Unable to add window -- token null is not for an application
4

4 に答える 4

3

問題は、非ビジュアルであるアプリケーションコンテキストにアタッチされたダイアログを作成していることです。私が見ているように、あなたがしているのはコンテンツリゾルバーにアクセスすることだけなので、とにかくアプリケーションコンテキストは必要ありません。

この行を変更する必要があります:

addToCalendar(DateAdder.this.getApplicationContext(), 
    shiftType, d.getDate(), d.getDate()+1000);

これに:

addToCalendar(DateAdder.this, 
    shiftType, d.getDate(), d.getDate()+1000);

そして、エラーはなくなります。(私はそれをテストしました)。

補遺:

また、addToCalendar()の最初の行を変更して、アプリケーションコンテキストを使用し、それによってコメント付きの要件を満たしますが、それが違いを生むとは確信していません。すなわち:

final ContentResolver cr = ctx.getApplicationContext().getContentResolver();
于 2010-12-10T14:43:29.943 に答える
0

これを変える

addToCalendar(getApplicationContext(), shiftType, d.getDate(), d.getDate()+1000);

addToCalendar(---yourClassName---.this.getApplicationContext(), shiftType, d.getDate(), d.getDate()+1000); 

例 :

addToCalendar(MainActivity.this.getApplicationContext(), shiftType, d.getDate(), d.getDate()+1000); 
于 2010-12-10T11:42:28.693 に答える
0

アクティビティ内にいる場合は、ActivityがContextを拡張するため、thisの代わりに直接渡すことができます。getApplicationContextApplicationContextはアプリケーション全体のコンテキストですが、this渡すと現在のコンテキストを渡すことができます。これで問題が解決する場合があります。

于 2010-12-10T12:38:00.863 に答える
0

時々あなたもすることができます

(アクティビティ)view.getParent()

Dateadder.thisまたはdateadder.this.getapplicationContext()の代わりに

于 2010-12-10T22:09:28.703 に答える