Общайтесь и отправляйте простые данные между несколькими устройствами, подключенными через Wifi Direct - PullRequest
0 голосов
/ 16 июня 2019

Я так понимаю, что связь между устройствами по wifi direct должна идти через сервер.Если у меня есть устройства A, B, C, где A является владельцем группы, то я должен связаться с: Клиент B -> Сервер A -> Клиент C, Клиент C -> Сервер A -> Клиент B

Однако,после долгих исследований я не могу найти демонстрации того, как это реализовать.У меня есть рабочий код для связи между 1 сервером и 1 клиентом, но мне нужно иметь возможность общаться с несколькими клиентами, которые подключены через Wifi P2P.

Код, который работает от связи между 1 сервером и 1 клиентом:

Основная активность:

public class MainActivity extends AppCompatActivity {

    public static final String TAG = "nuala";

    Button btnOnOff, btnDiscover, btnSend;
    ListView listView;
    TextView read_msg_box, connectionStatus;
    EditText writeMsg;

    WifiManager wifiManager;
    WifiP2pManager mManager;
    WifiP2pManager.Channel mChannel;

    BroadcastReceiver mReceiver;
    IntentFilter mIntentFilter;

    //these 3 are created for getting the information about the devices
    List<WifiP2pDevice> peers = new ArrayList<WifiP2pDevice>();
    String[] deviceNameArray; //used to show the device name in the ListView
    WifiP2pDevice[] deviceArray; //used to connect to a device

    static final int MESSAGE_READ = 1; //for the handler

    ServerClass serverClass;
    ClientClass clientClass;
    SendReceive sendReceive;

    Button myTurnBtn;
 //   Boolean myTurn;

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

        initalWork();
        exqListener();
    }

    //for reading the message sent
    //Handler is needed to access variables from other threads
    //Handler is the best way to communicate between threads
    Handler handler = new Handler(new Handler.Callback() {
        @Override
        public boolean handleMessage(Message msg) {

            switch(msg.what){
                case MESSAGE_READ:
                    byte[] readBuff = (byte[]) msg.obj;
                    String tempMsg = new String(readBuff,0,msg.arg1);
                    //if tempMsg.equals("go") then show the myTurn button
                    if (tempMsg.equals("go")) {
                        myTurnBtn.setVisibility(View.VISIBLE);
                        Toast.makeText(MainActivity.this, "success", Toast.LENGTH_SHORT).show();
                } else {
                        Toast.makeText(MainActivity.this, "not valid", Toast.LENGTH_SHORT).show();
                    }

                    //  then set MyTurn button visible
                    //print to screen "success"
                    //else
                    //print not valid


                    read_msg_box.setText(tempMsg);
                    break;
            }

            return true;
        }
    });

    private void exqListener() {

        btnOnOff.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //turn wifi off if on and on if off
                //check status
                if (wifiManager.isWifiEnabled()) {
                    //turn off wifi
                    wifiManager.setWifiEnabled(false);
                    btnOnOff.setText("ON");
                } else {
                    //turn on wifi
                    wifiManager.setWifiEnabled(true);
                    btnOnOff.setText("OFF");
                }
            }
        });

        btnDiscover.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //discovering other devices
                mManager.discoverPeers(mChannel, new WifiP2pManager.ActionListener() {
                    @Override
                    public void onSuccess() {
                        connectionStatus.setText("Discovery Started");
                    }

                    @Override
                    public void onFailure(int reason) {
                        connectionStatus.setText("Discovery failed");
                    }
                });  //after discovering devices, need to get the information of the nearby devices (peers)
            }
        });

        //connect to the device that is clicked
        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                final WifiP2pDevice device = deviceArray[position]; //linking up the position in the ListView with deviceArray position
                WifiP2pConfig config = new WifiP2pConfig();
                config.deviceAddress = device.deviceAddress;
               // config.groupOwnerIntent = 15;
                config.wps.setup = WpsInfo.PBC;

                //connect to the chosen device - only tells of success or failure: need to use ConnectionInfoListener
                mManager.connect(mChannel, config, new WifiP2pManager.ActionListener() {
                    @Override
                    public void onSuccess() {
                        Toast.makeText(getApplicationContext(), "Connected to " + device.deviceName, Toast.LENGTH_SHORT).show();
                        Log.d(TAG, "Connected to: " + device.deviceName);
                    }

                    @Override
                    public void onFailure(int reason) {
                        Toast.makeText(getApplicationContext(), "Not Connected", Toast.LENGTH_SHORT).show();
                    }
                });


            }
        });

      /*  btnSend.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                String msg = writeMsg.getText().toString();
                sendReceive.write(msg.getBytes());
                //
            }
        }); */

       myTurnBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //when button is clicked. Send message to receiving device ("go")
                String go = "go";
                sendReceive.write(go.getBytes()); //pass the go string into the write message

                //set button to be invisible
                myTurnBtn.setVisibility(View.INVISIBLE);
            }
        });
    }

    private void initalWork() {
        btnOnOff = findViewById(R.id.onOff);
        btnDiscover = findViewById(R.id.discover);
        btnSend = findViewById(R.id.sendButton);
        listView = findViewById(R.id.peerListView);
        read_msg_box = findViewById(R.id.readMsg);
        connectionStatus = findViewById(R.id.connectionStatus);
        writeMsg = findViewById(R.id.writeMsg);
        myTurnBtn = findViewById(R.id.myTurnBtn);

        wifiManager = (WifiManager) getApplicationContext().getSystemService(Context.WIFI_SERVICE);
        mManager = (WifiP2pManager) getSystemService(Context.WIFI_P2P_SERVICE); //this class provides the API for managing wifi peer to peer connectivity
        //anythign regarding the wifi p2p, this manager is used
        mChannel = mManager.initialize(this, getMainLooper(), null); //a channel that connects the application to the wifi p2p framework
        //most p2p operations require a channel as an argument

        mReceiver = new WifiDirectBroadcastReceiver(mManager, mChannel, this);

        mIntentFilter = new IntentFilter();
        //add all four actions
        mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTION);
        mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION);
        mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_CONNECTION_CHANGED_ACTION);
        mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_THIS_DEVICE_CHANGED_ACTION);

      //  myTurnBtn =findViewById(R.id.myTurnBtn);


    }

    //gathering and displaying devices available
    WifiP2pManager.PeerListListener peerListListener = new WifiP2pManager.PeerListListener() {
        @Override
        public void onPeersAvailable(WifiP2pDeviceList peerList) {
            if (!peerList.getDeviceList().equals(peers)) { //if the previous list != the current list
                peers.clear(); //clear the list
                peers.addAll(peerList.getDeviceList()); //add all avaailable peers

                //initialise deviceNameArray and deviceArray with a size of as many devices that have been found and stored in peers ArrayList
                deviceNameArray = new String[peerList.getDeviceList().size()];
                deviceArray = new WifiP2pDevice[peerList.getDeviceList().size()];

                int index = 0;
                //we have a peerList and two arrays
                //using the peerList, we can add all the devices in the two arrays
                for (WifiP2pDevice device : peerList.getDeviceList()) {  //for each device in the peerList.getDeviceList()
                    //each iteration, the value of a device is stored in the device variable
                    deviceNameArray[index] = device.deviceName; //add all devices from deviceNameArray - storing the deviceName
                    deviceArray[index] = device;  //add all devices from deviceArray
                    index++;


                    //when loop is finished:
                    //we have name of all devices in the deviceNameArray
                    //we have all device objects in the deviceArray
                }

                //shows the list of devices in the ListView
                ArrayAdapter<String> adapter = new ArrayAdapter<String>(getApplicationContext(), android.R.layout.simple_list_item_1
                        , deviceNameArray);
                listView.setAdapter(adapter);
            }
            if (peers.size() == 0) {
                Toast.makeText(getApplicationContext(), "No devices found", Toast.LENGTH_SHORT).show();
                return;
            }
        }
    };





    //get information about host and client servers and manage sending and recieving data
    WifiP2pManager.ConnectionInfoListener connectionInfoListener = new WifiP2pManager.ConnectionInfoListener() {
        @Override
        public void onConnectionInfoAvailable(WifiP2pInfo info) {
            final InetAddress groupOwnerAddress = info.groupOwnerAddress; //used for sending and receiving data

            //set groupOwn as the Host
            if (info.groupFormed && info.isGroupOwner) {
                connectionStatus.setText("Host");
                myTurnBtn.setVisibility(View.VISIBLE);
                serverClass = new ServerClass();
                Log.d(TAG, "server thread in groupFromed & groupOwner: " +Thread.currentThread().getId());
                serverClass.start();
            } else if (info.groupFormed) {
                connectionStatus.setText("Client");
                clientClass = new ClientClass(groupOwnerAddress);
                Log.d(TAG, "a client thread: " +Thread.currentThread().getId());
                clientClass.start();
            }
        }
    };


    //register the broadcast recevier
    @Override
    protected void onResume() {
        super.onResume();
        registerReceiver(mReceiver, mIntentFilter);
    }

    //unregister the broadcast receiver
    @Override
    protected void onPause() {
        super.onPause();
        unregisterReceiver(mReceiver);
    }


    public class ServerClass extends Thread {

        Socket socket;
        ServerSocket serverSocket;


        @Override
        public void run() {
            try {
                serverSocket = new ServerSocket(8888); //port must always be the same - can choose any number
                Log.e(TAG, getClass().getSimpleName() + "running on port: " + serverSocket.getLocalPort());
                socket = serverSocket.accept(); //socket is ready to accept message
                InetAddress add = socket.getInetAddress();
                int port = socket.getPort();
                Log.e(TAG, "Inet Address : " +add + "port: " +port);
                sendReceive = new SendReceive(socket);
                sendReceive.start();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }



    private class SendReceive extends Thread {
        private Socket socket;
        private InputStream inputStream;
        private OutputStream outputStream;

        //constructor
        public SendReceive(Socket skt) {
            socket = skt;
            try {
                Log.d(TAG, "SendReceive thread: " +Thread.currentThread().getId());
                inputStream = socket.getInputStream();
                outputStream = socket.getOutputStream();
            } catch (IOException e) {
                e.printStackTrace();
            }


        }
        //override run method
        @Override
        public void run() {
            byte[] buffer = new byte[1024]; //can choose any value
            int bytes;

            while (socket != null) { //always read to receive message
                try {
                    bytes = inputStream.read(buffer); //buffer contains the message, bytes contains the number of bytes in the message
                   // Log.d(TAG, "recieved message: " +buffer);
                    if (bytes > 0) {
                        handler.obtainMessage(MESSAGE_READ,bytes,-1,buffer).sendToTarget();
                        //handler used to deal with threading issues
                        //obtain message and send the data
                    }
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

        //method to send data
        public void write(byte[] bytes) {
            try {
                outputStream.write(bytes);
                Log.d(TAG, "write() Thread: "+Thread.currentThread().getId());
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }


    public class ClientClass extends Thread {

        Socket socket;
        String hostAddress;

        //hostAddress is provided at runtime when ClientClass is created
        public ClientClass(InetAddress hostAddress) {
            this.hostAddress = hostAddress.getHostAddress();
            socket = new Socket();
        }

        @Override
        public void run() {
            try {
                socket.connect(new InetSocketAddress(hostAddress, 8888), 500); //same port number as server side
                Log.e(TAG, getClass().getSimpleName() + "running on port: " + socket.getLocalPort());
                Log.d(TAG, "a client thread: " +Thread.currentThread().getId());
                sendReceive = new SendReceive(socket);
                sendReceive.start();
                Log.d(TAG, "a client thread: " +Thread.currentThread().getId());
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

Приемник Wifi Broadcast

public class WifiDirectBroadcastReceiver extends BroadcastReceiver {

    private WifiP2pManager mManager;
    private WifiP2pManager.Channel mChannel;
    private MainActivity mActivity;  //think about how this is used and how this can be possible with my app, on the Gameplay activity???

    /**
     * Constructor with args
     * @param manager
     * @param channel
     * @param activity
     */
    public WifiDirectBroadcastReceiver(WifiP2pManager manager, WifiP2pManager.Channel channel, MainActivity activity) {
        mManager = manager;
        mChannel = channel;
        mActivity = activity;

    }
    @Override
    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction(); //use this to check the current action

        if (WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTION.equals(action)) { //indicates whether Wi-Fi p2p is enabled
            int state = intent.getIntExtra(WifiP2pManager.EXTRA_WIFI_STATE, -1); //we will either have the value of the state or the default -1

            if (state == WifiP2pManager.WIFI_P2P_STATE_ENABLED) { //if wifi is enabled
                Toast.makeText(context, "Wifi is ON", Toast.LENGTH_SHORT).show();
            } else {
                Toast.makeText(context, "Wifi is OFF", Toast.LENGTH_SHORT).show();
            }

        } else if (WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION.equals(action)) { //indicates that the available peer list has changed
            //do something - get a list of peers by calling WifiP2pManager.requestPeers()
            if (mManager!=null) {
                mManager.requestPeers(mChannel, mActivity.peerListListener);  //pass in peerListListener from mainActivity
            }

        } else if (WifiP2pManager.WIFI_P2P_CONNECTION_CHANGED_ACTION.equals(action)) { //indicates the state of the wifi p2p connectivity has changed
            //do something - respond to a new connection or disconnections
            if (mManager == null) {
                return;
            }

            NetworkInfo networkInfo = intent.getParcelableExtra(WifiP2pManager.EXTRA_NETWORK_INFO);
            if (networkInfo.isConnected()) { //we are connect with the other device, requestion connection info to find group owner IP
                mManager.requestConnectionInfo(mChannel,mActivity.connectionInfoListener);

            } else {
                mActivity.connectionStatus.setText("Device Disconnected");
            }
        } else if (WifiP2pManager.WIFI_P2P_THIS_DEVICE_CHANGED_ACTION.equals(action)) { //indicates this device's configuration details have changed
            //do something - respond to this device's wifi state changing
        }

    }
}
...