Приложение вылетает после нажатия кнопки для загрузки макета вкладки - PullRequest
3 голосов
/ 07 июля 2011

[Edit] Когда я пытаюсь использовать предложение Sumant о переключении v.getContext () с «this» в конструкторе Intent, Eclipse говорит, что не может создать приложение. Когда вызывается метод onClick (), не передается ли ему текущее представление?

[Правка 2] Возможно, я понял проблему. Возможно, я использовал макет, предназначенный для Android 3.0, когда использовал эмулятор Android 2.3.3. Либо так, либо я просто использовал строковый массив, а не целочисленный массив.

В моем приложении для Android, когда пользователь нажимает кнопку, чтобы перевести их на вкладку, эмулятор, который я запускаю (использующий Android 2.3.3), принудительно закрывает мое приложение. Вот начальная активность.

Файл DungeonsDragonsAppActivity.java

package com.androidGuy.DnDApp;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
import android.util.Log;

public class DungeonsDragonsAppActivity extends Activity {

    /* Use this for debugging. */
    private static final String DEBUG_TAG = "DnDAppDebugLogging";

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

    /* The main buttons... Yeah. */
        final Button create_character_button = (Button) findViewById(R.id.createCharacterButton);
            final Button open_character_button = (Button) findViewById(R.id.openCharacterButton);
        final Button retrieve_character_button = (Button) findViewById(R.id.RetrieveCharacterButton);

        /* The button for actually creating the character.. */
        create_character_button.setOnClickListener(new View.OnClickListener() {
            public void onClick(View v) {
                try {
                    Intent intent = new Intent(v.getContext(),
                            CreateCharacterTabsActivity.class);
                    startActivityForResult(intent, 0);
                } catch (Exception e) {
                    Log.e(DEBUG_TAG, "Click failed", e);
                }
            }
        });

        /* The button for opening a pre-existing character. */
        open_character_button.setOnClickListener(new View.OnClickListener() {
            public void onClick(View v) {
                Toast.makeText(DungeonsDragonsAppActivity.this,
                        "You will be able to open a character... Eventually.",
                        Toast.LENGTH_SHORT).show();
            }
        });

        /* The button for importing from D & D insider. */
        retrieve_character_button
                .setOnClickListener(new View.OnClickListener() {
                    public void onClick(View v) {
                        Toast.makeText(
                                DungeonsDragonsAppActivity.this,
                                "You will be able to import characters... Hopefully.",
                                Toast.LENGTH_SHORT).show();
                    }
                }

                );
    }
}

Когда пользователь нажимает на кнопку, чтобы создать нового персонажа, он должен загрузить это действие:

CreateCharacterTabsActivity.java

package com.androidGuy.DnDApp;

import android.app.TabActivity;
import android.os.Bundle;
import android.widget.TabHost;
import android.content.Intent;

public class CreateCharacterTabsActivity extends TabActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        TabHost tabHost = getTabHost();
        TabHost.TabSpec spec;
        Intent intent;

        // Initialize a TabSpec for each tab and add it to the TabHost.
        spec = tabHost.newTabSpec("scores");
        spec.setContent(new Intent().setClass(this,
                CharacterAbilityScoresActivity.class));
        spec.setIndicator("Scores");
        tabHost.addTab(spec);

        intent = new Intent().setClass(this, CharacterClassActivity.class);
        spec = tabHost.newTabSpec("classes");
        spec.setContent(intent);
        spec.setIndicator("Classes");
        tabHost.addTab(spec);

        tabHost.setCurrentTab(0);
    }

}

Он использует этот XML-файл для рендеринга вкладок (я взял его из одного из учебных пособий по Android).

tabs.xml

<?xml version="1.0" encoding="utf-8"?>
<TabHost xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@android:id/tabhost"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">

    <LinearLayout
        android:orientation="vertical"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:padding="5dp">

        <FrameLayout
            android:id="@android:id/tabcontent"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:padding="5dp"
            android:layout_weight="1">

        </FrameLayout>

        <TabWidget
            android:id="@android:id/tabs"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_weight="0">
        </TabWidget>

    </LinearLayout>

</TabHost>

Как только это действие загружается, оно также должно загрузить эти действия на вкладки.

CharacterAbilityScoresActivity.java

package com.androidGuy.DnDApp;

import android.app.Activity;
import android.os.Bundle;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemSelectedListener;
import android.widget.ArrayAdapter;
import android.widget.Spinner;
import android.widget.Toast;
import android.view.View;

public class CharacterAbilityScoresActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.ability_score_spinners); // Set the scene.

        ArrayAdapter adapter = ArrayAdapter.createFromResource(this, R.array.ability_scores, android.R.layout.simple_spinner_item);
        adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);

        /* The ability score spinners */
        Spinner strengthSpinner = (Spinner) findViewById(R.id.StrengthSpinner);
        strengthSpinner.setAdapter(adapter);
        strengthSpinner
                .setOnItemSelectedListener(new abilityScoreListSelectedListener());

        Spinner constitutionSpinner = (Spinner) findViewById(R.id.ConstitutionSpinner);
        constitutionSpinner.setAdapter(adapter);
        constitutionSpinner
                .setOnItemSelectedListener(new abilityScoreListSelectedListener());
    }

    /* Now let's see what the user selects. */
    class abilityScoreListSelectedListener implements OnItemSelectedListener {
        public void onItemSelected(AdapterView<?> parent, View view, int pos,
                long id) {
            Toast.makeText(parent.getContext(),
                    "You selected an ability score.  Way to go.",
                    Toast.LENGTH_LONG).show();
        }

        public void onNothingSelected(AdapterView parent) {
            // Do nothing... For now.
        }
    }
}

ability_score_spinners.xml

<?xml version="1.0" encoding="utf-8"?>
    <LinearLayout android:id="@+id/linearLayout1" android:layout_width="fill_parent" xmlns:android="http://schemas.android.com/apk/res/android" android:layout_height="match_parent">
        <TableLayout android:layout_height="match_parent" android:id="@+id/tableLayout1" android:layout_width="wrap_content">
            <TableRow android:id="@+id/tableRow1" android:layout_width="wrap_content" android:layout_height="match_parent">
                <Spinner android:layout_height="wrap_content" android:minWidth="10dip" android:layout_width="wrap_content" android:layout_gravity="left" android:layout_weight="1" android:id="@+id/StrengthSpinner" android:entries="@array/ability_scores" android:prompt="@string/ability_score_prompt"></Spinner>
            </TableRow>
            <TableRow android:id="@+id/tableRow2" android:layout_width="wrap_content" android:layout_height="wrap_content">
                <Spinner android:layout_height="wrap_content" android:layout_weight="1" android:layout_width="wrap_content" android:id="@+id/ConstitutionSpinner" android:entries="@array/ability_scores" android:prompt="@string/ability_score_prompt"></Spinner>
            </TableRow>
            <TableRow android:id="@+id/tableRow4" android:layout_width="wrap_content" android:layout_height="wrap_content"></TableRow>
            <TableRow android:layout_height="wrap_content" android:id="@+id/tableRow3" android:layout_width="wrap_content">
            </TableRow>
        </TableLayout>
        <TableLayout android:layout_height="match_parent" android:id="@+id/tableLayout2" android:layout_width="match_parent">
            <TableRow android:id="@+id/tableRow5" android:layout_width="wrap_content" android:layout_height="wrap_content">
                <TextView android:text="@string/strengthText" android:layout_height="wrap_content" android:textAppearance="?android:attr/textAppearanceLarge" android:id="@+id/StrengthText"></TextView>
            </TableRow>
            <TableRow android:id="@+id/tableRow6" android:layout_width="wrap_content" android:layout_height="wrap_content">
                <TextView android:layout_height="wrap_content" android:layout_width="wrap_content" android:textAppearance="?android:attr/textAppearanceLarge" android:id="@+id/textView1" android:text="@string/constitionText"></TextView>
            </TableRow>
            <TableRow android:id="@+id/tableRow7" android:layout_width="wrap_content" android:layout_height="wrap_content"></TableRow>
            <TableRow android:id="@+id/tableRow8" android:layout_width="wrap_content" android:layout_height="wrap_content">
            </TableRow>
        </TableLayout>
    </LinearLayout>

CharacterClassesActivity.java

package com.androidGuy.DnDApp;

import android.app.Activity;
import android.os.Bundle;
import android.widget.RadioButton;
import android.widget.RadioGroup;
import android.widget.TextView;

public class CharacterClassActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.cha_classes);

        final RadioGroup classesButtons = (RadioGroup) findViewById(R.id.radioGroup1);
        classesButtons
                .setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
                    public void onCheckedChanged(RadioGroup group, int checkedId) {
                        TextView leTextView = (TextView) findViewById(R.id.chooseClassTextView);
                        if (checkedId != -1) {
                            RadioButton rb = (RadioButton) findViewById(checkedId);

                            if (rb != null) {
                                leTextView.setText("You chose: " + rb.getText());
                            }
                        } else {
                            leTextView.setText("Choose your class");
                        }
                    }
                });
    }
}

cha_classes.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout android:id="@+id/linearLayout1" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" xmlns:android="http://schemas.android.com/apk/res/android">
   <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:textAppearance="?android:attr/textAppearanceLarge" android:text="Choose your class" android:id="@+id/chooseClassTextView"></TextView>

    <RadioGroup android:id="@+id/radioGroup1" android:layout_width="wrap_content" android:layout_height="wrap_content">
        <RadioButton android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/ClericClassButton" android:text="Cleric"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/FighterClassButton" android:text="Fighter"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/PaladinClassButton" android:text="Paladin"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:text="Ranger" android:layout_height="wrap_content" android:id="@+id/RangerClassButton"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:text="Rogue" android:layout_height="wrap_content" android:id="@+id/RogueClassButton"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:text="Warlock" android:layout_height="wrap_content" android:id="@+id/WarlockClassButton"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:text="Warlord" android:layout_height="wrap_content" android:id="@+id/WarlordClassButton"></RadioButton>
        <RadioButton android:layout_width="wrap_content" android:text="Wizard" android:layout_height="wrap_content" android:id="@+id/WizardClassButton"></RadioButton>
    </RadioGroup>
</LinearLayout>

Оценки способностей должны быть на первой вкладке, а классы персонажей - на второй. Я записал целочисленный массив в каталог res / values ​​/ и попытался использовать его в счетчике, который используется с показателями способностей.

Сначала я подумал, что это макет вкладки и как она обрабатывается. Все, что нужно сделать, это настроить вкладку и заполнить ее соответствующей активностью. Я также подумал, что мне не хватает метода setContentView () в CreateCharacterTabsActivity. Но, похоже, дело не в этом. Я также подумал, что это из-за того, что в моем манифесте пропущены некоторые записи об активности, но это тоже неправильно.

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      android:versionCode="1"
      android:versionName="1.0" package="com.androidGuy.DnDApp">
    <uses-sdk android:minSdkVersion="10" />

    <application android:icon="@drawable/icon" android:label="@string/app_name">
        <activity android:name=".DungeonsDragonsAppActivity"
                  android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".CreateCharacterTabsActivity" android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <action android:name="android.intent.category.LAUNCHER"></action>
            </intent-filter>
        </activity>
        <activity android:name=".CharacterAbilityScoresActivity"></activity>
        <activity android:name=".CharacterClassActivity"></activity>

    </application>
</manifest>

Кажется, тогда ошибка может быть связана с тем, как я обрабатываю целочисленные данные для счетчиков. Я создал ArrayAdapter для приема целочисленных данных и установки данных в счетчики. Тем не менее, когда я запускаю эмулятор, он все равно вылетает.

Вот трассировка стека LogCat для случая сбоя.

ERROR/AndroidRuntime(363): FATAL EXCEPTION: main
ERROR/AndroidRuntime(363): java.lang.NullPointerException
ERROR/AndroidRuntime(363):     at android.widget.ArrayAdapter.createViewFromResource(ArrayAdapter.java:355)
ERROR/AndroidRuntime(363):     at android.widget.ArrayAdapter.getView(ArrayAdapter.java:323)
ERROR/AndroidRuntime(363):     at android.widget.AbsSpinner.onMeasure(AbsSpinner.java:192)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.widget.TableRow.getColumnsWidths(TableRow.java:308)
ERROR/AndroidRuntime(363):     at android.widget.TableLayout.findLargestCells(TableLayout.java:500)
ERROR/AndroidRuntime(363):     at android.widget.TableLayout.measureVertical(TableLayout.java:465)
ERROR/AndroidRuntime(363):     at android.widget.TableLayout.onMeasure(TableLayout.java:431)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1017)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.measureHorizontal(LinearLayout.java:701)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.onMeasure(LinearLayout.java:311)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.measureVertical(LinearLayout.java:531)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.onMeasure(LinearLayout.java:309)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.measureVertical(LinearLayout.java:531)
ERROR/AndroidRuntime(363):     at android.widget.LinearLayout.onMeasure(LinearLayout.java:309)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3138)
ERROR/AndroidRuntime(363):     at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
ERROR/AndroidRuntime(363):     at android.view.View.measure(View.java:8313)
ERROR/AndroidRuntime(363):     at android.view.ViewRoot.performTraversals(ViewRoot.java:839)
ERROR/AndroidRuntime(363):     at android.view.ViewRoot.handleMessage(ViewRoot.java:1859)
ERROR/AndroidRuntime(363):     at android.os.Handler.dispatchMessage(Handler.java:99)
ERROR/AndroidRuntime(363):     at android.os.Looper.loop(Looper.java:123)
ERROR/AndroidRuntime(363):     at android.app.ActivityThread.main(ActivityThread.java:3683)
ERROR/AndroidRuntime(363):     at java.lang.reflect.Method.invokeNative(Native Method)
ERROR/AndroidRuntime(363):     at java.lang.reflect.Method.invoke(Method.java:507)
ERROR/AndroidRuntime(363):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
ERROR/AndroidRuntime(363):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
ERROR/AndroidRuntime(363):     at dalvik.system.NativeStart.main(Native Method)

Спасибо за любые идеи, которые вы можете дать.

Ответы [ 2 ]

1 голос
/ 12 октября 2011

Похоже, в вашем манифесте определены две активности запуска:

...
<activity android:name=".DungeonsDragonsAppActivity"
          android:label="@string/app_name">
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity>
<activity android:name=".CreateCharacterTabsActivity" android:label="@string/app_name">
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <action android:name="android.intent.category.LAUNCHER"></action>
    </intent-filter>
</activity>
...

Это может вызвать сбой TabManager.

0 голосов
/ 12 октября 2011

Будьте осторожны, используя this внутри внутренних классов! Я делаю эту ошибку регулярно, и обычно в связи с передачей Context чему-либо! Убедитесь, что вы используете правильный синтаксис для перехода к «внешнему» this, например, MyOuterClass.this.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...