скачать приложение дрон дрон Apple version_обучающее видео по дрону
скачать приложение дрон дрон Apple version_обучающее видео по дрону

Всем привет, мы снова встретились, я ваш друг Цюаньчжаньцзюнь.

Оглавление

статья Оглавление

краткое содержание

В этом разделе в основном описывается процесс подключения HelloDrone и процесс разрешения протокола Mavlink.

1.Создать экземплярдрони объекты управления наземными станциями

Основное содержание:

controlTower = new ControlTower(context); drone = new Drone(context);

Язык кода:javascript
копировать
    @Override
    /** * Переопределить метод onCreate и запустить его перед методом onStart. */
    protected void onCreate(Bundle savedInstanceState) { 
   
        //Вызов onCreate родительского класса
        super.onCreate(savedInstanceState);
        //Установим информацию для отображения в главном окне
        setContentView(R.layout.activity_main);
        Log.i("lxw","+++++");

        //Получаем имя файла приложения
        final Context context = getApplicationContext();
        //Создаем элемент управления
        controlTower = new ControlTower(context);
        //Создаем экземпляр дрона
        drone = new Drone(context);
        //Выбор режима
        this.modeSelector = (Spinner) findViewById(R.id.modeSelect);
        //Добавляем событие прослушивания выбора режима
        this.modeSelector.setOnItemSelectedListener(new Spinner.OnItemSelectedListener()
        { 
   
            @Override
            public void onItemSelected(AdapterView<?> parent, View view, int position, long id)
            { 
   
                onFlightModeSelected(view);
            }

            @Override
            public void onNothingSelected(AdapterView<?> parent)
            { 
   
                // Do nothing
            }
        });

        final Button takePic = (Button) findViewById(R.id.take_photo_button);
        takePic.setOnClickListener(new View.OnClickListener() { 
   
            @Override
            public void onClick(View v) { 
   
                takePhoto();
            }
        });

        final Button toggleVideo = (Button) findViewById(R.id.toggle_video_recording);
        toggleVideo.setOnClickListener(new View.OnClickListener() { 
   
            @Override
            public void onClick(View v) { 
   
                toggleVideoRecording();
            }
        });

        videoView = (TextureView) findViewById(R.id.video_content);
        videoView.setSurfaceTextureListener(new TextureView.SurfaceTextureListener() { 
   
            @Override
            public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) { 
   
                alertUser("Video display is available.");
                startVideoStream.setEnabled(true);
                startVideoStreamUsingObserver.setEnabled(true);
            }

            @Override
            public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) { 
   

            }

            @Override
            public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) { 
   
                startVideoStream.setEnabled(false);
                startVideoStreamUsingObserver.setEnabled(false);
                return true;
            }

            @Override
            public void onSurfaceTextureUpdated(SurfaceTexture surface) { 
   

            }
        });

        startVideoStream = (Button) findViewById(R.id.start_video_stream);
        startVideoStream.setEnabled(false);
        startVideoStream.setOnClickListener(new View.OnClickListener() { 
   
            @Override
            public void onClick(View v) { 
   
                alertUser("Starting video stream.");
                startVideoStream(new Surface(videoView.getSurfaceTexture()));
            }
        });

        stopVideoStream = (Button) findViewById(R.id.stop_video_stream);
        stopVideoStream.setEnabled(false);
        stopVideoStream.setOnClickListener(new View.OnClickListener() { 
   
            @Override
            public void onClick(View v) { 
   
                alertUser("Stopping video stream.");
                stopVideoStream();
            }
        });

        startVideoStreamUsingObserver = (Button) findViewById(R.id.start_video_stream_using_observer);
        startVideoStreamUsingObserver.setEnabled(false);
        startVideoStreamUsingObserver.setOnClickListener(new View.OnClickListener() { 
   
            @Override
            public void onClick(View v) { 
   
                alertUser("Starting video stream using observer for video stream packets.");
                startVideoStreamForObserver();
            }
        });

        stopVideoStreamUsingObserver = (Button) findViewById(R.id.stop_video_stream_using_observer);
        stopVideoStreamUsingObserver.setEnabled(false);
        stopVideoStreamUsingObserver.setOnClickListener(new View.OnClickListener() { 
   
            @Override
            public void onClick(View v) { 
   
                alertUser("Stopping video stream using observer for video stream packets.");
                stopVideoStreamForObserver();
            }
        });

        // Initialize media codec manager to decode video stream packets.
        HandlerThread mediaCodecHandlerThread = new HandlerThread("MediaCodecHandlerThread");
        mediaCodecHandlerThread.start();
        Handler mediaCodecHandler = new Handler(mediaCodecHandlerThread.getLooper());
        mediaCodecManager = new MediaCodecManager(mediaCodecHandler);

        mainHandler = new Handler(getApplicationContext().getMainLooper());
    }

2. Привязка услуг

Язык кода:javascript
копировать
    @Override
    public void onStart()
    { 
   
        Log.i("lxw","Hello Drone на старте");
        super.onStart();
        //Выполняем задачи подключения
        Log.i(MAGTAG,"onStart");
        Log.i(MAGTAG,"this:"+this); //this:hello.MainActivity
        Log.i(MAGTAG,"this2:"+getApplicationContext()); //this2:hello.StarterApplication
        //Выполняем подключение контроллера
        this.controlTower.connect(this,getApplicationContext());
        //Обновляем тип режима
        updateVehicleModesForType(this.droneType);
        Log.i(MAGTAG,"connect");

    }

Основной код:

Язык кода:javascript
копировать
this.controlTower.connect(this,getApplicationContext());

Реализация привязки службы o3drServicesConnection

Язык кода:javascript
копировать
    public void connect(TowerListener listener,Context context) { 
   
        boolean value;
        Log.i("LXW","AAA");
        //Здесь нет мониторинга, условное суждение
        if (towerListener != null && (isServiceConnecting.get() || isTowerConnected()))
            return;
         //пусто
        if (listener == null)
        { 
   
            throw new IllegalArgumentException("ServiceListener argument cannot be null.");
        }
        Log.i("LXW","CCC");
        towerListener = listener;
        Log.i("LXW","DDD");
        if (!isTowerConnected() && !isServiceConnecting.get())
        { 
   
            Intent serviceIntent = ApiAvailability.getInstance().getAvailableServicesInstance(context);
            //value1:Intent { act=com.o3dr.services.android.lib.model.IDroidPlannerServices
            // cmp=com.o3dr.sample.hellodrone/org.droidplanner.services.android.impl.api.DroidPlannerService }
            Log.i("LXW","value1:"+ApiAvailability.getInstance().getAvailableServicesInstance(context));
            value= this.context.bindService(serviceIntent, o3drServicesConnection, Context.BIND_AUTO_CREATE);
            //value2:true
            Log.i("LXW","value2:"+value);
            isServiceConnecting.set(value);
        }
        Log.i("LXW","EEE");
    }

После привязки службы первым делом нужно запустить onCreate(), а затем метод onBind().

Нам нужно объявить службу в файле манифеста.

Давайте посмотрим на код DroidPlannerService и обнаружим, что он наследует Service, который также соответствует требованиям сервиса.

Язык кода:javascript
копировать
/** Реализация фонового сервиса Android DroneKit. * DroneKit-Android background service implementation. */
public class DroidPlannerService extends Service { 
   

        /** * Status bar notification id */
    private static final int FOREGROUND_ID = 101;

    /** * Set of actions to notify the local app's components of the service events. */
    public static final String ACTION_DRONE_CREATED = Utils.PACKAGE_NAME + ".ACTION_DRONE_CREATED";
    public static final String ACTION_DRONE_DESTROYED = Utils.PACKAGE_NAME + ".ACTION_DRONE_DESTROYED";
    public static final String ACTION_RELEASE_API_INSTANCE = Utils.PACKAGE_NAME + ".action.RELEASE_API_INSTANCE";
    public static final String EXTRA_API_INSTANCE_APP_ID = "extra_api_instance_app_id";

    /** * Used to broadcast service events. */
    private LocalBroadcastManager lbm;

    /** * Stores drone api instances per connected client. The client are denoted by their app id. */
    final ConcurrentHashMap<String, DroneApi> droneApiStore = new ConcurrentHashMap<>();

    /** * Caches drone managers per connection type. */
    final ConcurrentHashMap<ConnectionParameter, DroneManager> droneManagers = new ConcurrentHashMap<>();

    private DPServices dpServices;

    private CameraInfoLoader cameraInfoLoader;
    private List<CameraDetail> cachedCameraDetails;

    /** * Generate a drone api instance for the client denoted by the given app id. * * @param listener Used to retrieve api information. * @param appId Application id of the connecting client. * @return a IDroneApi instance */
    DroneApi registerDroneApi(IApiListener listener, String appId) { 
   
        if (listener == null)
            return null;

        DroneApi droneApi = new DroneApi(this, listener, appId);
        droneApiStore.put(appId, droneApi);
        lbm.sendBroadcast(new Intent(ACTION_DRONE_CREATED));
        updateForegroundNotification();
        return droneApi;
    }

    /** * Release the drone api instance attached to the given app id. * * @param appId Application id of the disconnecting client. */
    void releaseDroneApi(String appId) { 
   
        if (appId == null)
            return;

        DroneApi droneApi = droneApiStore.remove(appId);
        if (droneApi != null) { 
   
            Timber.d("Releasing drone api instance for " + appId);
            droneApi.destroy();
            lbm.sendBroadcast(new Intent(ACTION_DRONE_DESTROYED));
            updateForegroundNotification();
        }
    }

    /** * Установите соединение с автомобилем, используя заданные параметры соединения. * Establish a connection with a vehicle using the given connection parameter. * * @param connParams Параметры для подключения к автомобилю --- Параметры used to connect to the vehicle. * @param appId Идентификатор клиента подключающегося приложения --- Приложение id of the connecting client. * @param listener Обратный вызов для получения событий дрона. Перезвонить to receive drone events. * @return Экземпляр DroneManager, который действует как маршрутизатор между подключенными транспортными средствами и прослушивающими клиентами. * A DroneManager instance which acts as router between the connected vehicle and the listeneing client(s). */
    DroneManager connectDroneManager(ConnectionParameter connParams, String appId, DroneApi listener) { 
   

        if (connParams == null || TextUtils.isEmpty(appId) || listener == null)
        { 
   
            Log.i("lxw","connectDroneManager NULL");
            return null;
        }


        DroneManager droneMgr = droneManagers.get(connParams);
        Log.i("lxw","MYdroneMgr:"+droneMgr);
        if (droneMgr == null) { 
   
            final DroneManager temp = DroneManager.generateDroneManager(getApplicationContext(), connParams, new Handler(Looper.getMainLooper()));
            Log.i("lxw","temp:"+temp);
            droneMgr = droneManagers.putIfAbsent(connParams, temp);
            Log.i("lxw","MYdroneMgr2:"+droneMgr);
            if(droneMgr == null){ 
   
                Log.i("lxw","Generating:");
                Timber.d("Generating new drone manager.");
                droneMgr = temp;
            }
            else{ 
   
                temp.destroy();
            }
        }

        Timber.d("Drone manager connection for " + appId);
        Log.i("lxw","Generating connect");
        //устанавливаем соединение
        droneMgr.connect(appId, listener, connParams);
        return droneMgr;
    }

    /** * Disconnect the given client from the vehicle managed by the given drone manager. * * @param droneMgr Handler for the connected vehicle. * @param clientInfo Info of the disconnecting client. */
    void disconnectDroneManager(DroneManager droneMgr, DroneApi.ClientInfo clientInfo) { 
   
        if (droneMgr == null || clientInfo == null || TextUtils.isEmpty(clientInfo.appId))
            return;

        String appId = clientInfo.appId;
        Timber.d("Drone manager disconnection for " + appId);
        droneMgr.disconnect(clientInfo);
        if (droneMgr.getConnectedAppsCount() == 0) { 
   
            Timber.d("Destroying drone manager.");
            droneMgr.destroy();
            droneManagers.remove(droneMgr.getConnectionParameter());
        }
    }

    /** * Retrieves the set of camera info provided by the app. * * @return a list of {@link CameraDetail} objects. */
    synchronized List<CameraDetail> getCameraDetails() { 
   
        if (cachedCameraDetails == null) { 
   
            List<String> cameraInfoNames = cameraInfoLoader.getCameraInfoList();

            List<CameraInfo> cameraInfos = new ArrayList<>(cameraInfoNames.size());
            for (String infoName : cameraInfoNames) { 
   
                try { 
   
                    cameraInfos.add(cameraInfoLoader.openFile(infoName));
                } catch (Exception e) { 
   
                    Timber.e(e, e.getMessage());
                }
            }

            List<CameraDetail> cameraDetails = new ArrayList<>(cameraInfos.size());
            for (CameraInfo camInfo : cameraInfos) { 
   
                cameraDetails.add(new CameraDetail(camInfo.name, camInfo.sensorWidth,
                        camInfo.sensorHeight, camInfo.sensorResolution, camInfo.focalLength,
                        camInfo.overlap, camInfo.sidelap, camInfo.isInLandscapeOrientation));
            }

            cachedCameraDetails = cameraDetails;
        }

        return cachedCameraDetails;
    }

    @Override
    //Здесь возвращается ноль?
    public IBinder onBind(Intent intent) { 
   
        Log.d("lxw","onBind service " + intent);
        final String action = intent.getAction();
        if (IDroidPlannerServices.class.getName().equals(action))
        { 
   
            // Return binder to ipc client-server interaction.
            return dpServices;
        } else
            { 
   
            return null;
        }
    }

    @SuppressLint("NewApi")
    @Override
    public void onCreate() { 
   
        super.onCreate();
        Log.d("lxw","Binding DroidPlannerService onCreate " );
        Timber.d("Creating DroneKit-Android.");

        final Context context = getApplicationContext();

        dpServices = new DPServices(this);
        lbm = LocalBroadcastManager.getInstance(context);
        this.cameraInfoLoader = new CameraInfoLoader(context);

        updateForegroundNotification();
    }

    @SuppressLint("NewApi")
    private void updateForegroundNotification() { 
   
        final Context context = getApplicationContext();

        //Put the service in the foreground
        final NotificationCompat.Builder notifBuilder = new NotificationCompat.Builder(context)
                .setContentTitle("DroneKit-Android")
                .setPriority(NotificationCompat.PRIORITY_MIN)
                .setSmallIcon(R.drawable.ic_stat_notify);

        final int connectedCount = droneApiStore.size();
        if (connectedCount > 1) { 
   
            notifBuilder.setContentText(connectedCount + " connected apps");
        }

        final Notification notification = notifBuilder.build();
        startForeground(FOREGROUND_ID, notification);
    }

    @Override
    public void onDestroy() { 
   
        super.onDestroy();
        Timber.d("Destroying DroneKit-Android.");

        for (DroneApi droneApi : droneApiStore.values()) { 
   
            droneApi.destroy();
        }
        droneApiStore.clear();

        for (DroneManager droneMgr : droneManagers.values()) { 
   
            droneMgr.destroy();
        }
        droneManagers.clear();

        dpServices.destroy();

        stopForeground(true);

        //Disable this service. It'll be reenabled the next time its local client needs it.
        enableDroidPlannerService(getApplicationContext(), false);
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) { 
   
        Log.i("lxw","MYintent:"+intent);
        if (intent != null) { 
   
            final String action = intent.getAction();
            switch (action) { 
   

                case ACTION_RELEASE_API_INSTANCE:
                    final String appId = intent.getStringExtra(EXTRA_API_INSTANCE_APP_ID);
                    releaseDroneApi(appId);
                    break;
            }
        }

        stopSelf();
        return START_NOT_STICKY;
    }

    /** * Toggles the DroidPlannerService component * @param context * @param enable */
    public static void enableDroidPlannerService(Context context, boolean enable){ 
   
        final ComponentName serviceComp = new ComponentName(context, DroidPlannerService.class);
        final int newState = enable ? PackageManager.COMPONENT_ENABLED_STATE_ENABLED
                : PackageManager.COMPONENT_ENABLED_STATE_DISABLED;

        context.getPackageManager().setComponentEnabledSetting(serviceComp, newState, PackageManager.DONT_KILL_APP);
    }

}

После выполнения onbind он вернется

Тогда оно будет выполнено

Язык кода:javascript
копировать
    /** * Создать службу подключения */
    private final ServiceConnection o3drServicesConnection = new ServiceConnection() { 
   

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) { 
   
            isServiceConnecting.set(false);
            Log.i("lxw","onServiceConnected");
            o3drServices = IDroidPlannerServices.Stub.asInterface(service);
            try { 
   
                o3drServices.asBinder().linkToDeath(binderDeathRecipient, 0);
                Log.i("lxw","onServiceConnected try ");
                //Уведомление о соединении
                notifyTowerConnected();
            } catch (RemoteException e) { 
   
                //Сообщаем, что нет соединения
                notifyTowerDisconnected();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName name) { 
   
            Log.i("lxw","EEE777");
            isServiceConnecting.set(false);
            notifyTowerDisconnected();
        }
    };

Основная функция: notifyTowerConnected()

Язык кода:javascript
копировать
    /** * соединение для уведомлений */
    void notifyTowerConnected()
    { 
   
        Log.i("lxw","notifyTowerConnected");
        if (towerListener == null)
            return;
        //Соединение вызова
        Log.i("lxw","onTower");
        towerListener.onTowerConnected();
    }

Мы сосредоточимся на этой функции: TowerListener.onTowerConnected(); она в основном реализует регистрацию дронов и регистрацию слушателей.

Язык кода:javascript
копировать
    public void onTowerConnected() { 
   
        //Запрос об успешном подключении
        alertUser("DroneKit-Android Connected");
        this.controlTower.registerDrone(this.drone, this.handler);
        Log.i("lxw","onTowerConnected");
        this.drone.registerDroneListener(this);
        Log.i("lxw","registerDroneListener");
    }

3. Добавьте события щелчка соединения и подключитесь.

MainActivity реализует событие нажатия кнопки

Язык кода:javascript
копировать
    public void onBtnConnectTap(View view)
    { 
   
        //Определите, подключен ли статус соединения. Если он подключен, при нажатии будет предложено подключиться к порту.
        if (this.drone.isConnected())
        { 
   
            Log.i("lxw","Открыть соединение");
            this.drone.disconnect();
        } else
         { 
   
            Log.i("lxw","Подключиться");
            //Spinner предоставляет возможность быстрого выбора значения из коллекции данных. По умолчанию Spinner отображает текущее выбранное значение.
            // При нажатии на счетчик появится всплывающее меню, содержащее все доступные значения, из которого можно выбрать новое значение для счетчика.
            Spinner connectionSelector = (Spinner) findViewById(R.id.selectConnectionType);
            //Получаем свойства раскрывающегося списка
            int selectedConnectionType = connectionSelector.getSelectedItemPosition();
            //Определяем, является ли текущее соединение USB-соединением, если да, то устанавливаем USB-соединение, иначе это UDP-соединение
            //conntParams:ConnectionParameter{connectionType=0, paramsBundle=[extra_usb_baud_rate=115200]}
            ConnectionParameter connectionParams = selectedConnectionType == ConnectionType.TYPE_USB
                ? ConnectionParameter.newUsbConnection(null)
                : ConnectionParameter.newUdpConnection(null);

                Log.i("lxw","connectionParams:"+connectionParams);
            //Внутренний класс реализует события подключения
            this.drone.connect(connectionParams, new LinkListener()
            { 
   

                @Override
                public void onLinkStateUpdated(@NonNull LinkConnectionStatus connectionStatus)
                { 
   
                    Log.i("lxw","connectionStatus:"+connectionStatus);
                }
            });
        }
    }

Основной код: анонимный внутренний класс реализует события подключения.

Язык кода:javascript
копировать
            //Анонимный внутренний класс реализует события подключения
            this.drone.connect(connectionParams, new LinkListener()
            { 
   

                @Override
                public void onLinkStateUpdated(@NonNull LinkConnectionStatus connectionStatus)
                { 
   
                    Log.i("lxw","connectionStatus:"+connectionStatus);
                }
            });
Язык кода:javascript
копировать
    public void connect(ConnectionParameter connParams, LinkListener linkListener) { 
   
        Log.i("lxw","Начать операцию подключения дрона");
        //USB-соединение
        VehicleApi.getApi(this).connect(connParams);
        this.connectionParameter = connParams;
        this.linkListener = linkListener;
        Log.i("lxw","this.connectionParameter:"+this.connectionParameter);
        Log.i("lxw","this.linkListener:"+this.linkListener);
    }

Продолжайте изучать основной код: VehicleApi.getApi(this).connect(connParams);

Язык кода:javascript
копировать
    public void connect(ConnectionParameter parameter) { 
   
        Bundle params = new Bundle();
        //params:Bundle[{}]
        Log.i("lxw","Bundle params:"+params);
        params.putParcelable(EXTRA_CONNECT_PARAMETER, parameter);
        Action connectAction = new Action(ACTION_CONNECT, params);
        //дрон выполняет асинхронные операции
        //com.o3dr.services.android.lib.model.action.Action
        Log.i("lxw","Bundle connectAction:"+connectAction);
        drone.performAsyncAction(connectAction);
    }

Формат:drone.performAsyncAction(connectAction);

Язык кода:javascript
копировать
    public boolean performAsyncAction(Action action) { 
   
        //Поток асинхронного выполнения
        return performAsyncActionOnDroneThread(action, null);
    }

Код анализа: PerformAsyncActionOnDroneThread.

Язык кода:javascript
копировать
    public boolean performAsyncActionOnDroneThread(Action action, AbstractCommandListener listener) { 
   
        //Выполняем асинхронную обработку поведения
        return performAsyncActionOnHandler(action, this.handler, listener);
    }

Процедура выполненияAsyncActionOnHandler

Язык кода:javascript
копировать
    public boolean performAsyncActionOnHandler(Action action, Handler handler, AbstractCommandListener listener) { 
   
        boolean value=false;
        Log.i("lxw","performAsyncActionOnHandler");
        final IDroneApi droneApi = droneApiRef.get();
        //org.droidplanner.services.android.impl.api.DroneApi
        Log.i("lxw","droneApi:"+droneApi);
        value=isStarted(droneApi);
        Log.i("lxw","isStarted value:"+value);
        if (value)
        { 
   
            try { 
   
                //Выполнение поведения
                Log.i("lxw","executeAsyncAction:");
                droneApi.executeAsyncAction(action, wrapListener(handler, listener));
                return true;
            } catch (RemoteException e)
            { 
   
                Log.i("lxw","Выполнить RemoteException");
                Toast.makeText(getContext(), «Асинхронное выполнение 2», Toast.LENGTH_LONG).show();
                handleRemoteException(e);
            }
        }
        Log.i("lxw","disconnect7");
        return false;
    }

Основной код, вызывающий беспокойство:droneApi.executeAsyncAction(action, WrapListener(handler, Listener));

Язык кода:javascript
копировать
    /** * поведение выполнения */
    public void executeAction(Action action, ICommandListener listener) throws RemoteException { 
   
        //com.o3dr.services.android.lib.model.action.Action
        Log.i("lxw","executeAction action:"+action);
        if (action == null)
        { 
   
            Log.i("lxw","action null:");
            return;
        }

        String type = action.getType();
        //type com.o3dr.services.android.action.CONNECT
        Log.i("lxw","type:"+type);
        if (type == null)
        { 
   
            Log.i("lxw","type null:");
            return;
        }

        Bundle data = action.getData();
        Log.i("lxw","Bundle data:"+data);
        if (data != null) { 
   
            Log.i("lxw","data not null:");
            data.setClassLoader(context.getClassLoader());
        }
        //Получаем устройство
        Drone drone = getDrone();
        Log.i("lxw","drone1:"+drone);
        Log.i("lxw","mytype:"+type);
        switch (type) { 
   
            // CONNECTION ACTIONS
            case ConnectionActions.ACTION_CONNECT:
                ConnectionParameter parameter = data.getParcelable(ConnectionActions.EXTRA_CONNECT_PARAMETER);
                Log.i("lxw","ConnectionActions:");
                //устанавливаем соединение
                connect(parameter);
                break;

            case ConnectionActions.ACTION_DISCONNECT:
                disconnect();
                break;

            // CAMERA ACTIONS
            case CameraActions.ACTION_START_VIDEO_STREAM: { 
   
                Surface videoSurface = data.getParcelable(CameraActions.EXTRA_VIDEO_DISPLAY);
                String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");

                Bundle videoProps = data.getBundle(CameraActions.EXTRA_VIDEO_PROPERTIES);
                if (videoProps == null) { 
   
                    //Only case where it's null is when interacting with a deprecated client version.
                    //In this case, we assume that the client is attempting to start a solo stream, since that's
                    //the only api that was exposed.
                    videoProps = new Bundle();
                    videoProps.putInt(CameraActions.EXTRA_VIDEO_PROPS_UDP_PORT, VideoManager.ARTOO_UDP_PORT);
                }

                CommonApiUtils.startVideoStream(drone, videoProps, ownerId, videoTag, videoSurface, listener);
                break;
            }

            case ExperimentalActions.ACTION_START_VIDEO_STREAM_FOR_OBSERVER: { 
   
                String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
                CommonApiUtils.startVideoStreamForObserver(drone, ownerId, videoTag, listener);
                break;
            }

            case CameraActions.ACTION_STOP_VIDEO_STREAM: { 
   
                String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
                CommonApiUtils.stopVideoStream(drone, ownerId, videoTag, listener);
                break;
            }

            case ExperimentalActions.ACTION_STOP_VIDEO_STREAM_FOR_OBSERVER: { 
   
                String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
                CommonApiUtils.stopVideoStreamForObserver(drone, ownerId, videoTag, listener);
                break;
            }

            // MISSION ACTIONS
            case MissionActions.ACTION_BUILD_COMPLEX_MISSION_ITEM:
                if (drone instanceof MavLinkDrone || drone == null) { 
   
                    CommonApiUtils.buildComplexMissionItem((MavLinkDrone) drone, data);
                } else { 
   
                    CommonApiUtils.postErrorEvent(CommandExecutionError.COMMAND_UNSUPPORTED, listener);
                }
                break;

            case MissionActions.ACTION_SAVE_MISSION: { 
   
                Mission mission = data.getParcelable(MissionActions.EXTRA_MISSION);
                Uri saveUri = data.getParcelable(MissionActions.EXTRA_SAVE_MISSION_URI);
                if (saveUri == null) { 
   
                    CommonApiUtils.postErrorEvent(CommandExecutionError.COMMAND_FAILED, listener);
                } else { 
   
                    MissionUtils.saveMission(context, mission, saveUri, listener);
                }
                break;
            }

            case MissionActions.ACTION_LOAD_MISSION: { 
   
                Uri loadUri = data.getParcelable(MissionActions.EXTRA_LOAD_MISSION_URI);
                boolean setMission = data.getBoolean(MissionActions.EXTRA_SET_LOADED_MISSION, false);
                if (loadUri != null) { 
   
                    Mission mission = MissionUtils.loadMission(context, loadUri);
                    if(mission != null){ 
   
                        // Going back to the caller.
                        data.putParcelable(MissionActions.EXTRA_MISSION, mission);

                        if(setMission){ 
   
                            Bundle params = new Bundle();
                            params.putParcelable(EXTRA_MISSION, mission);
                            params.putBoolean(EXTRA_PUSH_TO_DRONE, false);
                            executeAction(new Action(ACTION_SET_MISSION, params), listener);
                        }
                    }
                }
                break;
            }

            default:
                if (droneMgr != null) { 
   
                    droneMgr.executeAsyncAction(clientInfo, action, listener);
                } else { 
   
                    CommonApiUtils.postErrorEvent(CommandExecutionError.COMMAND_FAILED, listener);
                }
                break;
        }
    }

Здесь выбираем USB-соединение и ориентируемся на выполняемые функции.

Язык кода:javascript
копировать
    /** * Выполнение задач по подключению * @param connParams */
    public void connect(ConnectionParameter connParams) { 
   
        try { 
   
            Log.i("lxw","connParams:"+connParams);
            //Validate the given connection parameter
            connParams = checkConnectionParameter(connParams);
            Log.i("lxw","connParams1:"+connParams);
            //Validate the current connection parameter for the drone
            ConnectionParameter currentConnParams = this.connectionParams == null
                ? this.connectionParams
                : checkConnectionParameter(this.connectionParams);
            Log.i("lxw","currentConnParams:"+currentConnParams);
            if (!connParams.equals(currentConnParams))
            { 
   
                Log.i("lxw","droneMgr:"+droneMgr);
                if (this.droneMgr != null)
                { 
   
                    Log.i("lxw","droneMgr is not null:");
                    LinkConnectionStatus connectionStatus = LinkConnectionStatus
                        .newFailedConnectionStatus(LinkConnectionStatus.ADDRESS_IN_USE,
                            "Connection already started with different connection parameters");
                    onConnectionStatus(connectionStatus);
                    return;
                }

                this.connectionParams = connParams;
                Log.i("lxw","this.connectionParams:"+this.connectionParams);
                //ownerId:com.o3dr.sample.hellodrone
                Log.i("lxw","ownerId:"+ownerId);
                //устанавливаем вызов управления соединением
                this.droneMgr = service.connectDroneManager(this.connectionParams, ownerId, this);
                Log.i("lxw","this.droneMgr:"+this.droneMgr);
                Log.i("lxw","bufdata:"+isEventsBufferingEnabled());
                if(isEventsBufferingEnabled())
                { 
   
                    Log.i("lxw","isEventsBufferingEnabled:");
                    eventsBuffer.clear();
                    handler.postDelayed(eventsDispatcher, this.connectionParams.getEventsDispatchingPeriod());
                }
            }
        } catch (ConnectionException e)
        { 
   
            Log.i("lxw","link connect:");
            LinkConnectionStatus connectionStatus = LinkConnectionStatus
                .newFailedConnectionStatus(LinkConnectionStatus.INVALID_CREDENTIALS, e.getMessage());
            onConnectionStatus(connectionStatus);
            disconnect();
        }
    }

Пример: this.droneMgr = service.connectDroneManager(this.connectionParams, OwnerId, this);

Язык кода:javascript
копировать
    /** * Установите соединение с автомобилем, используя заданные параметры соединения. * Establish a connection with a vehicle using the given connection parameter. * * @param connParams Параметры для подключения к автомобилю --- Параметры used to connect to the vehicle. * @param appId Идентификатор клиента подключающегося приложения --- Приложение id of the connecting client. * @param listener Обратный вызов для получения событий дрона. Перезвонить to receive drone events. * @return Экземпляр DroneManager, который действует как маршрутизатор между подключенными транспортными средствами и прослушивающими клиентами. * A DroneManager instance which acts as router between the connected vehicle and the listeneing client(s). */
    DroneManager connectDroneManager(ConnectionParameter connParams, String appId, DroneApi listener) { 
   

        if (connParams == null || TextUtils.isEmpty(appId) || listener == null)
        { 
   
            Log.i("lxw","connectDroneManager NULL");
            return null;
        }


        DroneManager droneMgr = droneManagers.get(connParams);
        Log.i("lxw","MYdroneMgr:"+droneMgr);
        if (droneMgr == null) { 
   
            final DroneManager temp = DroneManager.generateDroneManager(getApplicationContext(), connParams, new Handler(Looper.getMainLooper()));
            Log.i("lxw","temp:"+temp);
            droneMgr = droneManagers.putIfAbsent(connParams, temp);
            Log.i("lxw","MYdroneMgr2:"+droneMgr);
            if(droneMgr == null){ 
   
                Log.i("lxw","Generating:");
                Timber.d("Generating new drone manager.");
                droneMgr = temp;
            }
            else{ 
   
                temp.destroy();
            }
        }

        Timber.d("Drone manager connection for " + appId);
        Log.i("lxw","Generating connect");
        //устанавливаем соединение
        droneMgr.connect(appId, listener, connParams);
        return droneMgr;
    }

Код анализа:droneMgr.connect(appId, прослушиватель, connParams);

Язык кода:javascript
копировать
    public synchronized void connect(String appId, DroneApi listener, ConnectionParameter connParams) { 
   
        //org.droidplanner.services.android.impl.api.DroneApi
        Log.i("lxw","listener:"+listener);
        Log.i("lxw","listener2:"+TextUtils.isEmpty(appId));
        if (listener == null || TextUtils.isEmpty(appId)) { 
   
            Log.i("lxw","listener return:");
            return;
        }

        connectedApps.put(appId, listener);
        //Начало подключения
        Log.i("lxw","doConnect:");
        doConnect(appId, listener, connParams);
    }

Пример: doConnect(appId, прослушиватель, connParams);

Язык кода:javascript
копировать
    protected void doConnect(String appId, DroneApi listener, ConnectionParameter connParams) { 
   
        Log.i("lxw"," MavlinkDrone Manager doConnect:");
        if (mavClient.isDisconnected()) { 
   
            Timber.i("Opening connection for %s", appId);
            mavClient.openConnection();
            Log.i("lxw"," mavClientdoConnect:1");
        } else { 
   
            if (isConnected()) { 
   
                listener.onDroneEvent(DroneInterfaces.DroneEventsType.CONNECTED, drone);
                if (!drone.isConnectionAlive())
                    listener.onDroneEvent(DroneInterfaces.DroneEventsType.HEARTBEAT_TIMEOUT, drone);
            }
            Log.i("lxw"," mavClientdoConnect:2");
        }

        mavClient.registerForTLogLogging(appId, connParams.getTLogLoggingUri());
        Log.i("lxw"," updateDroneStreamRate:");
        updateDroneStreamRate(connParams);
    }

Сосредоточьтесь на: mavClient.openConnection();

Язык кода:javascript
копировать
    public synchronized void openConnection()
    { 
   
        if(isConnected() || isConnecting())
        { 
   
            Log.i("lxw"," openConnection:1");
            return;
        }


        final String tag = toString();

        //Создаем Mavlink----Создать the mavlink connection
        final int connectionType = connParams.getConnectionType();
        final Bundle paramsBundle = connParams.getParamsBundle();
        Log.i("lxw"," connectionType:"+connectionType); //0
        Log.i("lxw"," paramsBundle:"+paramsBundle); //скорость передачи данных
        Log.i("lxw"," mavlinkConn:"+mavlinkConn); //UsbConnection
        if(mavlinkConn == null) { 
   
            switch (connectionType) { 
   
                //Тип USB
                case ConnectionType.TYPE_USB:  //0
                    final int baudRate = paramsBundle.getInt(ConnectionType.EXTRA_USB_BAUD_RATE,
                            ConnectionType.DEFAULT_USB_BAUD_RATE);
                    mavlinkConn = new UsbConnection(context, baudRate);
                    Log.i("lxw"," mavlinkConn:6"+mavlinkConn);
                    Timber.i("Connecting over usb.");
                    break;
                //Тип Bluetooth
                case ConnectionType.TYPE_BLUETOOTH:
                    //Retrieve the bluetooth address to connect to
                    final String bluetoothAddress = paramsBundle.getString(ConnectionType.EXTRA_BLUETOOTH_ADDRESS);
                    mavlinkConn = new BluetoothConnection(context, bluetoothAddress);
                    Timber.i("Connecting over bluetooth.");
                    break;
                //Тип TCP
                case ConnectionType.TYPE_TCP:
                    //Retrieve the server ip and port
                    final String tcpServerIp = paramsBundle.getString(ConnectionType.EXTRA_TCP_SERVER_IP);
                    final int tcpServerPort = paramsBundle.getInt(ConnectionType
                            .EXTRA_TCP_SERVER_PORT, ConnectionType.DEFAULT_TCP_SERVER_PORT);
                    mavlinkConn = new AndroidTcpConnection(context, tcpServerIp, tcpServerPort, new WifiConnectionHandler(context));
                    Timber.i("Connecting over tcp.");
                    break;
                //тип UDP
                case ConnectionType.TYPE_UDP:
                    final int udpServerPort = paramsBundle
                            .getInt(ConnectionType.EXTRA_UDP_SERVER_PORT, ConnectionType.DEFAULT_UDP_SERVER_PORT);
                    mavlinkConn = new AndroidUdpConnection(context, udpServerPort, new WifiConnectionHandler(context));
                    Timber.i("Connecting over udp.");
                    break;
                //Тип СОЛО
                case ConnectionType.TYPE_SOLO: { 
   
                    Timber.i("Creating solo connection");
                    final String soloLinkId = paramsBundle.getString(ConnectionType.EXTRA_SOLO_LINK_ID, null);
                    final String linkPassword = paramsBundle.getString(ConnectionType.EXTRA_SOLO_LINK_PASSWORD, null);
                    mavlinkConn = new SoloConnection(context, soloLinkId, linkPassword);
                    break;
                }

                default:
                    Timber.e("Unrecognized connection type: %s", connectionType);
                    return;
            }
        }
        Log.i("lxw"," mavlinkConn:7");
        Log.i("lxw"," mConnectionListener:"+mConnectionListener);
        mavlinkConn.addMavLinkConnectionListener(tag, mConnectionListener);

        //Check if we need to ping a server to receive UDP data stream.
        if (connectionType == ConnectionType.TYPE_UDP)
        { 
   
            final String pingIpAddress = paramsBundle.getString(ConnectionType.EXTRA_UDP_PING_RECEIVER_IP);
            if (!TextUtils.isEmpty(pingIpAddress)) { 
   
                try { 
   
                    final InetAddress resolvedAddress = InetAddress.getByName(pingIpAddress);

                    final int pingPort = paramsBundle.getInt(ConnectionType.EXTRA_UDP_PING_RECEIVER_PORT);
                    final long pingPeriod = paramsBundle.getLong(ConnectionType.EXTRA_UDP_PING_PERIOD,
                            ConnectionType.DEFAULT_UDP_PING_PERIOD);
                    final byte[] pingPayload = paramsBundle.getByteArray(ConnectionType.EXTRA_UDP_PING_PAYLOAD);

                    ((AndroidUdpConnection) mavlinkConn).addPingTarget(resolvedAddress, pingPort, pingPeriod, pingPayload);

                } catch (UnknownHostException e) { 
   
                    Timber.e(e, "Unable to resolve UDP ping server ip address.");
                }
            }
        }
        Log.i("lxw"," mavlinkConn:888");
        if (mavlinkConn.getConnectionStatus() == MavLinkConnection.MAVLINK_DISCONNECTED)
        { 
   
            Log.i("lxw"," Начать выполнение Mavlink соединять");
            mavlinkConn.connect(null);
        }
    }

Ключевые функции: выполнить связанный с mavlink mavlinkConn.connect(null);

Язык кода:javascript
копировать
    public void connect(Bundle extras) { 
   
        Log.i("lxw"," mavlinkConn:999");
        if (mConnectionStatus.compareAndSet(MAVLINK_DISCONNECTED, MAVLINK_CONNECTING)) { 
   
            extrasHolder.set(extras);
            mLogger.logInfo(TAG, "Starting connection thread.");
            Log.i("lxw"," создание задачи подключения mavlink");
            mConnectThread = new Thread(mConnectingTask, "MavLinkConnection-Connecting Thread");
            //Начало соединения
            mConnectThread.start();
            //Сообщаем о состоянии ссылки
            reportConnecting();
        }
    }
Язык кода:javascript
копировать
    /** * Запустите процесс подключения. Начинать the connection process. */
    private final Runnable mConnectingTask = new Runnable() { 
   
        @Override
        public void run() { 
   
            // Load the connection specific preferences
            loadPreferences();
            // Open the connection
            try { 
   
                Log.i("lxw"," Запустите выполнение, чтобы открыть openConnection. run");
                openConnection(extrasHolder.get());
                Log.i("lxw"," mavlinkConn:111");
            } catch (IOException e) { 
   
                Log.i("lxw"," mavlinkConn:2222");
                // Ignore errors while shutting down
                if (mConnectionStatus.get() != MAVLINK_DISCONNECTED) { 
   
                    reportIOException(e);

                    mLogger.logErr(TAG, e);
                }

                disconnect();
            }

            mLogger.logInfo(TAG, "Exiting connecting thread.");
        }
    };

Продолжайте рассматривать код: openConnection(extrasHolder.get());

Язык кода:javascript
копировать
    @Override
    protected void openConnection(Bundle connectionExtras) throws IOException { 
   
        boolean id_device;
        Log.i("lxw"," mUsbConnection:"+mUsbConnection);
        if (mUsbConnection != null) { 
   
            try { 
   
                Log.i("lxw"," BBB123");
                mUsbConnection.openUsbConnection(connectionExtras);
                Log.d(TAG, "Reusing previous usb connection.");
                return;
            } catch (IOException e) { 
   
                Log.i("lxw"," BBB456");
                Log.e(TAG, "Previous usb connection is not usable.", e);
                mUsbConnection = null;
            }
        }
        id_device=isFTDIdevice(context);
        Log.i("lxw","id_device:"+id_device);
        if (id_device)
        { 
   
            final UsbConnectionImpl tmp = new UsbFTDIConnection(context, this, mBaudRate);
            try { 
   
                tmp.openUsbConnection(connectionExtras);

                // If the call above is successful, 'mUsbConnection' will be set.
                mUsbConnection = tmp;
                Log.d(TAG, "Using FTDI usb connection.");
            } catch (IOException e) { 
   
                Log.d(TAG, "Unable to open a ftdi usb connection. Falling back to the open "
                    + "usb-library.", e);
            }
        }

        // Fallback
        if (mUsbConnection == null) { 
   
            Log.i("lxw"," KKK");
            Log.i("lxw"," BT:"+mBaudRate);
            Log.i("lxw"," BT2:"+context);
            final UsbConnectionImpl tmp = new UsbCDCConnection(context, this, mBaudRate);
            Log.i("lxw"," BT3:"+tmp);
            // If an error happens here, let it propagate up the call chain since this is the fallback.
            tmp.openUsbConnection(connectionExtras);
            Log.i("lxw"," BT5:");
            mUsbConnection = tmp;
            Log.i("lxw"," BT6:"+mUsbConnection);
            Log.d(TAG, "Using open-source usb connection.");
        }
    }

Продолжайте рассматривать код: tmp.openUsbConnection(connectionExtras);

Язык кода:javascript
копировать
    protected void openUsbConnection(Bundle extras) throws IOException { 
   
        Log.i("lxw"," cdc");
        extrasHolder.set(extras);
        registerUsbPermissionBroadcastReceiver();

        // Get UsbManager from Android.
        UsbManager manager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);

        //Get the list of available devices
        List<UsbDevice> availableDevices = UsbSerialProber.getAvailableSupportedDevices(manager);
        Log.i("lxw"," availableDevices:"+availableDevices);
        if (availableDevices.isEmpty())
        { 
   
            Log.i("lxw"," CDC NULL");
            Log.d(TAG, "No Devices found");
            throw new IOException("No Devices found");
        }

        //Pick the first device
        UsbDevice device = availableDevices.get(0);
        if (manager.hasPermission(device))
        { 
   
            openUsbDevice(device, extras);
        } else { 
   
            removeWatchdog();

            scheduler = Executors.newSingleThreadScheduledExecutor();
            scheduler.schedule(permissionWatchdog, 15, TimeUnit.SECONDS);
            Log.d(TAG, "Requesting permission to access usb device " + device.getDeviceName());
            manager.requestPermission(device, usbPermissionIntent);
        }
    }

Продолжайте просматривать код: openUsbDevice(device, extras);

Язык кода:javascript
копировать
    private void openUsbDevice(UsbDevice device, Bundle extras) throws IOException { 
   
        //Получаем UsbManager с Android. --- Get UsbManager from Android.
        UsbManager manager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);

        //Находим первый доступный драйвер. --- Find the first available driver.
        final UsbSerialDriver serialDriver = UsbSerialProber.openUsbDevice(manager, device);

        if (serialDriver == null) { 
   
            Log.d(TAG, "No Devices found");
            throw new IOException("No Devices found");
        } else { 
   
            Log.d(TAG, "Opening using Baud rate " + mBaudRate);
            try { 
   
                serialDriver.open();
                serialDriver.setParameters(mBaudRate, 8, UsbSerialDriver.STOPBITS_1, UsbSerialDriver.PARITY_NONE);

                serialDriverRef.set(serialDriver);

                onUsbConnectionOpened(extras);
            } catch (IOException e)
            { 
   
                Log.e(TAG, "Error setting up device: " + e.getMessage(), e);
                try { 
   
                    serialDriver.close();
                } catch (IOException e2) { 
   
                    // Ignore.
                }
            }
        }
    }

продолжатьonUsbConnectionOpened (дополнительно);

Язык кода:javascript
копировать
       protected void onUsbConnectionOpened(Bundle extras) { 
   
            parentConnection.onConnectionOpened(extras);
        }
Язык кода:javascript
копировать
    /** * Откройте устройство и выполните задачу mManagerTask. * @param extras */
    protected void onConnectionOpened(Bundle extras) { 
   
        if (mConnectionStatus.compareAndSet(MAVLINK_CONNECTING, MAVLINK_CONNECTED)) { 
   
            extrasHolder.set(extras);
            mLogger.logInfo(TAG, "Starting manager thread.");
            Log.i("lxw"," mManagerTask в основном обрабатывает информацию mavlink");
            mTaskThread = new Thread(mManagerTask, "MavLinkConnection-Manager Thread");
            mTaskThread.start();
        }
    }

Анонимный внутренний класс реализует окончательную обработку протокола mavlink, связанную с

Язык кода:javascript
копировать
    private final Runnable mManagerTask = new Runnable() { 
   

        @Override
        public void run() { 
   
            Thread sendingThread = null;
            Thread loggingThread = null;

            try { 
   
                final long connectionTime = System.currentTimeMillis();
                mConnectionTime.set(connectionTime);
                reportConnect(connectionTime);

                // Launch the 'Sending' thread
                mLogger.logInfo(TAG, "Starting sender thread.");
                Log.i("lxw"," Starting new thread.");
                sendingThread = new Thread(mSendingTask, "MavLinkConnection-Sending Thread");
                sendingThread.start();

                //Launch the 'Logging' thread
                mLogger.logInfo(TAG, "Starting logging thread.");
                Log.i("lxw"," Starting sender thread.");
                loggingThread = new Thread(mLoggingTask, "MavLinkConnection-Logging Thread");
                loggingThread.start();
                //Анализ протокола
                final Parser parser = new Parser();
                //com.MAVLink.Parser
                Log.i("lxw"," parser:"+parser);
                //Статус парсинга сервиса
                parser.stats.resetStats();
                //Создаем массив разобранных данных
                final byte[] readBuffer = new byte[READ_BUFFER_SIZE];

                while (mConnectionStatus.get() == MAVLINK_CONNECTED)
                { 
   
                    //Читаем размер буфера
                    int bufferSize = readDataBlock(readBuffer);
                    //bufferSize=
                    Log.i("lxw"," bufferSize:"+bufferSize);
                    handleData(parser, bufferSize, readBuffer);
                }
            } catch (IOException e) { 
   
                // Ignore errors while shutting down
                if (mConnectionStatus.get() != MAVLINK_DISCONNECTED) { 
   
                    reportIOException(e);
                    mLogger.logErr(TAG, e);
                }
            } finally { 
   
                if (sendingThread != null && sendingThread.isAlive()) { 
   
                    sendingThread.interrupt();
                }

                if (loggingThread != null && loggingThread.isAlive()) { 
   
                    loggingThread.interrupt();
                }

                disconnect();
                mLogger.logInfo(TAG, "Exiting manager thread.");
            }
        }

        /** * Начать обработку данных * @param parser * @param bufferSize * @param buffer */
        private void handleData(Parser parser, int bufferSize, byte[] buffer) { 
   
            if (bufferSize < 1) { 
   
                return;
            }

            for (int i = 0; i < bufferSize; i++)
            { 
   
                //анализируем пакет
                MAVLinkPacket receivedPacket = parser.mavlink_parse_char(buffer[i] & 0x00ff);
                if (receivedPacket != null)
                { 
   
                    queueToLog(receivedPacket);
                    Log.i("lxw"," Начать разбор отчета: ");
                    reportReceivedPacket(receivedPacket);
                }
            }
        }
    };

Среди них наиболее важный контент, отправляемый приложением во внешний мир: Среди них наиболее важный контент, отправляемый приложением во внешний мир: Среди них наиболее важный контент, отправляемый приложением во внешний мир:

Язык кода:javascript
копировать
    private final Runnable mSendingTask = new Runnable() { 
   
        @Override
        public void run() { 
   
            try { 
   
                while (mConnectionStatus.get() == MAVLINK_CONNECTED) { 
   
                    byte[] buffer = mPacketsToSend.take();

                    try { 
   
                        sendBuffer(buffer);
                        queueToLog(buffer);
                    } catch (IOException e) { 
   
                        reportIOException(e);
                        mLogger.logErr(TAG, e);
                    }
                }
            } catch (InterruptedException e) { 
   
                mLogger.logVerbose(TAG, e.getMessage());
            } finally { 
   
                disconnect();
            }
        }
    };

Наше внимание здесь по-прежнему сосредоточено на приеме.

Основная обработка: handleData(parser, bufferSize, readBuffer); Основная обработка: handleData(parser, bufferSize, readBuffer); Основная обработка: handleData(parser, bufferSize, readBuffer);

Язык кода:javascript
копировать
        private void handleData(Parser parser, int bufferSize, byte[] buffer) { 
   
            if (bufferSize < 1) { 
   
                return;
            }

            for (int i = 0; i < bufferSize; i++)
            { 
   
                //анализируем пакет
                MAVLinkPacket receivedPacket = parser.mavlink_parse_char(buffer[i] & 0x00ff);
                if (receivedPacket != null)
                { 
   
                    queueToLog(receivedPacket);
                    Log.i("lxw"," Начать разбор отчета: ");
                    reportReceivedPacket(receivedPacket);
                }
            }
        }
Язык кода:javascript
копировать
    private void reportReceivedPacket(MAVLinkPacket packet)
    { 
   
        //Не является ли слушатель ненулевым
        if (mListeners.isEmpty())
        { 
   
            return;
        }

        for (MavLinkConnectionListener listener : mListeners.values())
        { 
   
            Log.i("lxw"," onReceivePacket:");
            //Получаем обработку данных
            listener.onReceivePacket(packet);
        }
    }
Язык кода:javascript
копировать
        public void onReceivePacket(final MAVLinkPacket packet) { 
   
            Log.i("lxw"," notifyReceivedData:");
            listener.notifyReceivedData(packet);
            Log.i("lxw"," receivedMsg END:");
        }
Язык кода:javascript
копировать
    public void notifyReceivedData(MAVLinkPacket packet) { 
   

        MAVLinkMessage receivedMsg = packet.unpack();
        Log.i("lxw"," receivedMsg:"+receivedMsg);
        if (receivedMsg == null)
            return;
        Log.i("lxw"," MSGID:"+receivedMsg.msgid);
        if (receivedMsg.msgid == msg_command_ack.MAVLINK_MSG_ID_COMMAND_ACK)
        { 
   
            Log.i("lxw"," command ack:");
            msg_command_ack commandAck = (msg_command_ack) receivedMsg;
            handleCommandAck(commandAck);
        } else
         { 
   
             Log.i("lxw"," Handler receiveData :");
            this.mavLinkMsgHandler.receiveData(receivedMsg);
            if (this.drone != null)
            { 
   
                Log.i("lxw"," Ядро обрабатывает информацию mavlink: ");
                this.drone.onMavLinkMessageReceived(receivedMsg);
            }
        }

        if (!connectedApps.isEmpty())
        { 
   
            Log.i("lxw"," spp :"+connectedApps.values());
            //org.droidplanner.services.android.impl.api.DroneApi
            for (DroneApi droneEventsListener : connectedApps.values())
            { 
   
                droneEventsListener.onReceivedMavLinkMessage(receivedMsg);
            }
        }
    }

Здесь есть две важные части кода: протокол с ответом и протокол без ответа. В основном мы фокусируемся на протоколе без ответа.

Сначала вызовите метод в Ardupilot:

Язык кода:javascript
копировать
    public void onMavLinkMessageReceived(MAVLinkMessage message) { 
   
        Log.i("lxw","ardupilot message.sysid :"+message.sysid );

        if ((message.sysid != this.getSysid()) && !isMavLinkMessageException(message)) { 
   
            // Reject Messages that are not for the system id
            return;
        }

        // Filter Components IDs to be specifically the IDs that can be processed
        int compId = message.compid;
        if (compId != AUTOPILOT_COMPONENT_ID
                && compId != ARTOO_COMPONENT_ID
                && compId != SiK_RADIO_FIXED_COMPID ){ 
   
            return;
        }

        if (!getParameterManager().processMessage(message)) { 
   

            getWaypointManager().processMessage(message);
            getCalibrationSetup().processMessage(message);

            switch (message.msgid) { 
   

                case msg_statustext.MAVLINK_MSG_ID_STATUSTEXT:
                    // These are any warnings sent from APM:Copter with
                    // gcs_send_text_P()
                    // This includes important thing like arm fails, prearm fails, low
                    // battery, etc.
                    // also less important things like "erasing logs" and
                    // "calibrating barometer"
                    msg_statustext msg_statustext = (msg_statustext) message;
                    processStatusText(msg_statustext);
                    break;

                case msg_vfr_hud.MAVLINK_MSG_ID_VFR_HUD:
                    processVfrHud((msg_vfr_hud) message);
                    break;

                case msg_raw_imu.MAVLINK_MSG_ID_RAW_IMU:
                    msg_raw_imu msg_imu = (msg_raw_imu) message;
                    mag.newData(msg_imu);
                    break;

                case msg_radio.MAVLINK_MSG_ID_RADIO:
                    msg_radio m_radio = (msg_radio) message;
                    processSignalUpdate(m_radio.rxerrors, m_radio.fixed, m_radio.rssi,
                            m_radio.remrssi, m_radio.txbuf, m_radio.noise, m_radio.remnoise);
                    break;

                case msg_rc_channels_raw.MAVLINK_MSG_ID_RC_CHANNELS_RAW:
                    rc.setRcInputValues((msg_rc_channels_raw) message);
                    break;

                case msg_servo_output_raw.MAVLINK_MSG_ID_SERVO_OUTPUT_RAW:
                    rc.setRcOutputValues((msg_servo_output_raw) message);
                    break;

                case msg_camera_feedback.MAVLINK_MSG_ID_CAMERA_FEEDBACK:
                    getCamera().newImageLocation((msg_camera_feedback) message);
                    break;

                case msg_mount_status.MAVLINK_MSG_ID_MOUNT_STATUS:
                    processMountStatus((msg_mount_status) message);
                    break;

                case msg_named_value_int.MAVLINK_MSG_ID_NAMED_VALUE_INT:
                    processNamedValueInt((msg_named_value_int) message);
                    break;

                //*************** Magnetometer calibration messages handling *************//
                case msg_mag_cal_progress.MAVLINK_MSG_ID_MAG_CAL_PROGRESS:
                case msg_mag_cal_report.MAVLINK_MSG_ID_MAG_CAL_REPORT:
                    getMagnetometerCalibration().processCalibrationMessage(message);
                    break;

                default:
                    break;
            }
        }
        Log.i("lxw","super message.sysid :"+message.sysid );
        //Вот метод вызова родительского класса
        super.onMavLinkMessageReceived(message);
    }
Язык кода:javascript
копировать
    public void onMavLinkMessageReceived(MAVLinkMessage message) { 
   
        Log.i("lxw","message.sysid :"+message.sysid );
        if ( (message.sysid != this.getSysid()) && !isMavLinkMessageException(message) )
        { 
   
            // Reject messages that are not for this drone's system id
            return;
        }

        onHeartbeat(message);

        switch (message.msgid) { 
   
            case msg_radio_status.MAVLINK_MSG_ID_RADIO_STATUS:
                msg_radio_status m_radio_status = (msg_radio_status) message;
                processSignalUpdate(m_radio_status.rxerrors, m_radio_status.fixed, m_radio_status.rssi,
                        m_radio_status.remrssi, m_radio_status.txbuf, m_radio_status.noise, m_radio_status.remnoise);
                break;

            case msg_attitude.MAVLINK_MSG_ID_ATTITUDE:
                msg_attitude m_att = (msg_attitude) message;
                processAttitude(m_att);
                break;

            case msg_heartbeat.MAVLINK_MSG_ID_HEARTBEAT:
                msg_heartbeat msg_heart = (msg_heartbeat) message;
                processHeartbeat(msg_heart);
                break;

            case msg_vibration.MAVLINK_MSG_ID_VIBRATION:
                msg_vibration vibrationMsg = (msg_vibration) message;
                processVibrationMessage(vibrationMsg);
                break;

            //*************** EKF State handling ******************//
            case msg_ekf_status_report.MAVLINK_MSG_ID_EKF_STATUS_REPORT:
                processEfkStatus((msg_ekf_status_report) message);
                break;

            case msg_sys_status.MAVLINK_MSG_ID_SYS_STATUS:
                msg_sys_status m_sys = (msg_sys_status) message;
                processSysStatus(m_sys);
                break;

            case msg_global_position_int.MAVLINK_MSG_ID_GLOBAL_POSITION_INT:
                processGlobalPositionInt((msg_global_position_int) message);
                break;

            case msg_gps_raw_int.MAVLINK_MSG_ID_GPS_RAW_INT:
                processGpsState((msg_gps_raw_int) message);
                break;

            case msg_mission_item.MAVLINK_MSG_ID_MISSION_ITEM:
                processHomeUpdate((msg_mission_item) message);
                break;

            case msg_mission_current.MAVLINK_MSG_ID_MISSION_CURRENT:
                missionStats.setWpno(((msg_mission_current) message).seq);
                break;

            case msg_mission_item_reached.MAVLINK_MSG_ID_MISSION_ITEM_REACHED:
                missionStats.setLastReachedWaypointNumber(((msg_mission_item_reached) message).seq);
                break;

            case msg_nav_controller_output.MAVLINK_MSG_ID_NAV_CONTROLLER_OUTPUT:
                msg_nav_controller_output m_nav = (msg_nav_controller_output) message;
                setDisttowpAndSpeedAltErrors(m_nav.wp_dist, m_nav.alt_error, m_nav.aspd_error);
                break;
        }
    }

Это общий процесс обработки протокола, который будет подробно проанализирован позже.

Издатель: Full stack программист и руководитель стека, укажите источник для перепечатки: https://javaforall.cn/170568.html Исходная ссылка: https://javaforall.cn

boy illustration
Неразрушающее увеличение изображений одним щелчком мыши, чтобы сделать их более четкими артефактами искусственного интеллекта, включая руководства по установке и использованию.
boy illustration
Копикодер: этот инструмент отлично работает с Cursor, Bolt и V0! Предоставьте более качественные подсказки для разработки интерфейса (создание навигационного веб-сайта с использованием искусственного интеллекта).
boy illustration
Новый бесплатный RooCline превосходит Cline v3.1? ! Быстрее, умнее и лучше вилка Cline! (Независимое программирование AI, порог 0)
boy illustration
Разработав более 10 проектов с помощью Cursor, я собрал 10 примеров и 60 подсказок.
boy illustration
Я потратил 72 часа на изучение курсорных агентов, и вот неоспоримые факты, которыми я должен поделиться!
boy illustration
Идеальная интеграция Cursor и DeepSeek API
boy illustration
DeepSeek V3 снижает затраты на обучение больших моделей
boy illustration
Артефакт, увеличивающий количество очков: на основе улучшения характеристик препятствия малым целям Yolov8 (SEAM, MultiSEAM).
boy illustration
DeepSeek V3 раскручивался уже три дня. Сегодня я попробовал самопровозглашенную модель «ChatGPT».
boy illustration
Open Devin — инженер-программист искусственного интеллекта с открытым исходным кодом, который меньше программирует и больше создает.
boy illustration
Эксклюзивное оригинальное улучшение YOLOv8: собственная разработка SPPF | SPPF сочетается с воспринимаемой большой сверткой ядра UniRepLK, а свертка с большим ядром + без расширения улучшает восприимчивое поле
boy illustration
Популярное и подробное объяснение DeepSeek-V3: от его появления до преимуществ и сравнения с GPT-4o.
boy illustration
9 основных словесных инструкций по доработке академических работ с помощью ChatGPT, эффективных и практичных, которые стоит собрать
boy illustration
Вызовите deepseek в vscode для реализации программирования с помощью искусственного интеллекта.
boy illustration
Познакомьтесь с принципами сверточных нейронных сетей (CNN) в одной статье (суперподробно)
boy illustration
50,3 тыс. звезд! Immich: автономное решение для резервного копирования фотографий и видео, которое экономит деньги и избавляет от беспокойства.
boy illustration
Cloud Native|Практика: установка Dashbaord для K8s, графика неплохая
boy illustration
Краткий обзор статьи — использование синтетических данных при обучении больших моделей и оптимизации производительности
boy illustration
MiniPerplx: новая поисковая система искусственного интеллекта с открытым исходным кодом, спонсируемая xAI и Vercel.
boy illustration
Конструкция сервиса Synology Drive сочетает проникновение в интрасеть и синхронизацию папок заметок Obsidian в облаке.
boy illustration
Центр конфигурации————Накос
boy illustration
Начинаем с нуля при разработке в облаке Copilot: начать разработку с минимальным использованием кода стало проще
boy illustration
[Серия Docker] Docker создает мультиплатформенные образы: практика архитектуры Arm64
boy illustration
Обновление новых возможностей coze | Я использовал coze для создания апплета помощника по исправлению домашних заданий по математике
boy illustration
Советы по развертыванию Nginx: практическое создание статических веб-сайтов на облачных серверах
boy illustration
Feiniu fnos использует Docker для развертывания личного блокнота Notepad
boy illustration
Сверточная нейронная сеть VGG реализует классификацию изображений Cifar10 — практический опыт Pytorch
boy illustration
Начало работы с EdgeonePages — новым недорогим решением для хостинга веб-сайтов
boy illustration
[Зона легкого облачного игрового сервера] Управление игровыми архивами
boy illustration
Развертывание SpringCloud-проекта на базе Docker и Docker-Compose