从片段调用Activity的方法

时间:2014-03-20 14:47:49

标签: java android bluetooth fragment

引言

我有一个蓝牙应用程序,它包含一个Activity,它有3个按钮:

  • 设置可见性并创建服务器连接
  • 查找设备并创建客户端连接
  • 发送内容

我有以下课程:

  • 主要活动:我拥有所有方法和按钮监听器
  • 3个不同的类与线程:ClientConnection,ServerConnection和ConnectedThread

现在我必须稍微更改设计,而不是使用所有这些操作,使用ActionBar上的Tabs我需要创建2个片段:服务器和客户端。

所以,基本上,我必须在Server片段中定义Server和send按钮,在Client片段中定义客户端按钮,并设置他们的监听器。

问题

从每个片段开始,当使用一个按钮时,我应该在主Activity中调用相应的方法ubicated。我想这样做,而不是复制每个片段中的所有方法和接口,只是为了让它更简单。

我尝试将方法设置为静态,但是方法中有一些定义无法通过静态引用调用。例如:

public class ServerFragment extends Fragment { 
    //...  
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.server_layout, container, false);

        btnServer = (Button) view.findViewById(R.id.buttonServer);   
        btnServer.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                onClickServer();
            }
        });
        //...

_

public class BluetoothActivity extends Activity {
    //...
    public void onClickServer() {
        Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
        discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 120);
        startActivityForResult(discoverableIntent, REQUEST_DISCOVERABLE);
    }

如果我将onClickServer()设置为静态,我必须将REQUEST_DISCOVERABLE设置为静态,而startActivityForResult()不允许我在那里使用静态变量。

我怎么能这样做?

UPDATE - 抛出NullpointerException

我创建了一个名为ClickInterface的新类作为建议,我已经定义了2个接口,一个用于服务器,另一个用于客户端。

不,从片段中,我试图在onClick方法中调用该方法,但是在我调用回调方法的行中抛出一个NPE:

public class ServerFragment extends Fragment implements OnClickListener {

    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.server_layout, container, false);

        btnServer = (Button) view.findViewById(R.id.buttonServer);
        btnServer.setOnClickListener(this);

        return view;
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.buttonServer:
                serverFragCallback.onServer(); //HERE I RECEIVE THE NPE
                break;
        }
    }

这是logcat的输出:

03-20 16:33:57.581: E/AndroidRuntime(24884): FATAL EXCEPTION: main
03-20 16:33:57.581: E/AndroidRuntime(24884): Process: com.uax.bluetoothconnection, PID: 24884
03-20 16:33:57.581: E/AndroidRuntime(24884): java.lang.NullPointerException
03-20 16:33:57.581: E/AndroidRuntime(24884):    at com.uax.bluetoothconnection.ServerFragment.onClick(ServerFragment.java:43)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at android.view.View.performClick(View.java:4633)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at android.view.View$PerformClick.run(View.java:19330)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at android.os.Handler.handleCallback(Handler.java:733)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at android.os.Handler.dispatchMessage(Handler.java:95)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at android.os.Looper.loop(Looper.java:157)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at android.app.ActivityThread.main(ActivityThread.java:5356)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at java.lang.reflect.Method.invokeNative(Native Method)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at java.lang.reflect.Method.invoke(Method.java:515)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1265)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1081)
03-20 16:33:57.581: E/AndroidRuntime(24884):    at dalvik.system.NativeStart.main(Native Method

4 个答案:

答案 0 :(得分:2)

让一切都变得静止是错误的做法。这样做:

  1. 使用从片段调用所需的方法定义新界面
  2. 让您的活动实现界面
  3. 在您的片段中,将getActivity()强制转换为您的界面并调用方法。 (请记得提前检查,即if (getActivity() instanceof MyInterface) ...

答案 1 :(得分:2)

您想要的是从Fragment到Activity的通信的回调接口。 docs已经为此提供了一个很好的例子。

定义您的Activity实现的接口:

public interface Callback {
    void doSomething();
}
public class YourActivity implements Callback {
    ...
    @Override
    public void doSomething() {
         // your implementation here
    }
}

在Fragment中,如果单击其中一个按钮,请使用此界面。

public class ServerFragment extends Fragment { 

    Callback iCallback;
    ...
    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        try {
            iCallback = (Callback) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException();
        }
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.server_layout, container, false);

        btnServer = (Button) view.findViewById(R.id.buttonServer);   
        btnServer.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                iCallback.doSomething();
            }
        });
    }
}

使用此方法,您可以将逻辑保留在Activity中并处理Fragment中的UI事件。通过Fragment中的事件,您可以调用Activity中的方法。

答案 2 :(得分:2)

正如克里斯所说,你让你的活动实现了一个界面,让我们说ClickInterface:

public class BluetoothActivity extends Activity implements ClickInterface{
//...
@override
public void onClickServer() {
    Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
    discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 120);
    startActivityForResult(discoverableIntent, REQUEST_DISCOVERABLE);
}

接口将是这样的:

public interface ClickInterface {
    public void onClickServer();
}

这是你的片段:

public class ServerFragment extends Fragment { 
    private ClickInterface mCallback;

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        mCallback = (ClickInterface) activity;
    }  
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.server_layout, container, false);

    btnServer = (Button) view.findViewById(R.id.buttonServer);   
    btnServer.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v) {
            mCallback.onClickServer();
        }
    });
}

所以,你在mCallback对象上调用“onClickServer()”方法,这个方法将在activity中调用,因为它实现了接口

答案 3 :(得分:1)

使用interface从任何片段页面调用Activity中的方法,例如:

public interface OnServerClickListener {
    public void onUserClickServer();
}

在活动中:

public class BluetoothActivity extends Activity implements OnServerClickListener {
//...
public void onUserClickServer() {
  onClickServer();
}
}

现在在片段中你可以使用它:

public class ServerFragment extends Fragment { 
//...  
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.server_layout, container, false);

    btnServer = (Button) view.findViewById(R.id.buttonServer);   
    btnServer.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v) {
             try{
            ((OnServerClickListener)getActivity()).onUserClickServer();
             }
             catch(Exception e){Log.e("ServerFragment","Main Activity doesnt implement OnServerClickListener");
        }
    });

享受!