Android Studio - 对视图使用来自外部 类 的方法

Android Studio - Using methods from outside classes with views

我在从另一个 class 调用的方法中获取视图以正常运行时遇到了一些困难。

我的应用程序在主 activity(名为 Overlay)中包含一个体验栏,它使用进度栏跟踪用户进度。我想做的是在另一个 class 中有一个单独的方法来处理与经验栏相关的任何代码。

我已经为 class 创建了一个实例并且该方法被正确调用,但是某些 android/view 相关的部分导致了错误。 System.out.println 将在调试器中显示文本,但是使用 Toast 和其他视图则不会。这是两个 class 文件和相关的错误。


public class OverlayActivity extends AppCompatActivity
    implements NavigationView.OnNavigationItemSelectedListener {

NavigationView navigationView = null;
Toolbar toolbar = null;
int level = 0;
int addXP = 110;
ProgressBar experienceBar;
Button buttonXP;
TextView experienceLevel;
ExperienceManager exp;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_overlay);

    //Set the Overlay Fragment

    OverlayFragment fragment = new OverlayFragment();
    android.support.v4.app.FragmentTransaction fragmentTransaction =
            getSupportFragmentManager().beginTransaction();
    fragmentTransaction.replace(R.id.fragment_container, fragment);
    fragmentTransaction.commit();

    toolbar = (Toolbar) findViewById(R.id.toolbar);
    setSupportActionBar(toolbar);

    FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
    fab.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            Snackbar.make(view, "Hay! Don't touch me!", Snackbar.LENGTH_LONG)
                    .setAction("Action", null).show();
        }
    });

    DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
    ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
            this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
    drawer.setDrawerListener(toggle);
    toggle.syncState();

    navigationView = (NavigationView) findViewById(R.id.nav_view);
    navigationView.setNavigationItemSelectedListener(this);

    experienceBar = (ProgressBar) findViewById(R.id.experienceBar);
    experienceBar.setProgress(12);

    experienceLevel = (TextView) findViewById(R.id.ds_appBar_level);

    buttonXP = (Button) findViewById(R.id.buttonXP);
    buttonXP.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View View) {
            exp = new ExperienceManager();
            exp.addExperience(level, addXP);
        }
    });
}

public class ExperienceManager extends AppCompatActivity {

ProgressBar experienceBar;

public int addExperience(int level, int addXP) {
    experienceBar = (ProgressBar) findViewById(R.id.experienceBar);

    int addLevel = 0;
    while (addXP >= experienceBar.getMax()) {
        addXP -= experienceBar.getMax();
        addLevel += 1;
    }
    if (addXP >= experienceBar.getMax()-experienceBar.getProgress()) {

        experienceBar.setProgress(experienceBar.getProgress() + addXP - experienceBar.getMax());
        addLevel += 1;
    } else {
        experienceBar.setProgress(experienceBar.getProgress() + addXP);
    }
    if (addLevel >= 1) {
        level += addLevel;
        Toast toast = Toast.makeText(getBaseContext(), "Level up!", Toast.LENGTH_SHORT);
        toast.show();
        //experienceLevel.setText("Level " + level);
    }
    return level;
}

E/AndroidRuntime: FATAL EXCEPTION: main
              Process: com.teamcirelios.deepstone, PID: 4138
              java.lang.NullPointerException: Attempt to invoke virtual method 'android.view.Window$Callback android.view.Window.getCallback()' on a null object reference
                  at android.support.v7.app.AppCompatDelegateImplBase.<init>(AppCompatDelegateImplBase.java:116)
                  at android.support.v7.app.AppCompatDelegateImplV9.<init>(AppCompatDelegateImplV9.java:147)
                  at android.support.v7.app.AppCompatDelegateImplV11.<init>(AppCompatDelegateImplV11.java:27)
                  at android.support.v7.app.AppCompatDelegateImplV14.<init>(AppCompatDelegateImplV14.java:50)
                  at android.support.v7.app.AppCompatDelegateImplV23.<init>(AppCompatDelegateImplV23.java:29)
                  at android.support.v7.app.AppCompatDelegate.create(AppCompatDelegate.java:199)
                  at android.support.v7.app.AppCompatDelegate.create(AppCompatDelegate.java:181)
                  at android.support.v7.app.AppCompatActivity.getDelegate(AppCompatActivity.java:521)
                  at android.support.v7.app.AppCompatActivity.findViewById(AppCompatActivity.java:190)
                  at com.teamcirelios.deepstone.ExperienceManager.addExperience(ExperienceManager.java:15)
                  at com.teamcirelios.deepstone.OverlayActivity.onClick(OverlayActivity.java:77)
                  at android.view.View.performClick(View.java:5697)
                  at android.widget.TextView.performClick(TextView.java:10826)
                  at android.view.View$PerformClick.run(View.java:22526)
                  at android.os.Handler.handleCallback(Handler.java:739)
                  at android.os.Handler.dispatchMessage(Handler.java:95)
                  at android.os.Looper.loop(Looper.java:158)
                  at android.app.ActivityThread.main(ActivityThread.java:7225)
                  at java.lang.reflect.Method.invoke(Native Method)
                  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1230)
                  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1120)

同样,尝试在片段中注册按钮点击时也会出现类似的错误。体验栏本身显示了 Overlay 活动,但是为了让应用程序按预期工作,它需要接收输入以从片段内的按钮(或更准确地说,复选框)调用 class。

我能看到的唯一可行的解​​决方案是采用 experienceBar.getMax() 和类似的方法并将它们作为参数包含在内。但是,我需要 return 级别和当前 xp,据我所知这是不可能的。

总的来说,我有三个主要问题。

谢谢!

首先,您的实用程序 class,即体验管理器不应扩展 AppCompatActivity...您的实用程序 class 不是 activity! PS 您无法在 Experience Manager 中找到您的 progressBar 的 ViewById class,因为该 ProgressBar 不是您的 ExperienceManager Activity 的一部分,它是您的 OverlayActivity 的一部分。

如果您希望创建一个静态实用程序方法来添加和管理它,您应该这样编写代码。

public class ExperienceManager{

public static int addExperience(ProgressBar experienceBar, int level, int addXP){

    int addLevel = 0;
    while (addXP >= experienceBar.getMax()) {
        addXP -= experienceBar.getMax();
        addLevel += 1;
    }
    if (addXP >= experienceBar.getMax()-experienceBar.getProgress()) {

        experienceBar.setProgress(experienceBar.getProgress() + addXP - experienceBar.getMax());
        addLevel += 1;
    } else {
        experienceBar.setProgress(experienceBar.getProgress() + addXP);
    }
    if (addLevel >= 1) {
        level += addLevel;
        Toast toast = Toast.makeText(getBaseContext(), "Level up!", Toast.LENGTH_SHORT);
        toast.show();
        //experienceLevel.setText("Level " + level);
    }
    return level;
}
}

你可以这样调用这个方法:

buttonXP.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View View) {
            // Calling the static utility method over here...
            ExperienceManager.addExperience(experienceBar, level, addXP);

        }
});

这就是您需要做的全部:)

我建议在尝试 Android 之前熟悉 Java 设计模式和最佳编程实践。