Фонарик в Android-фрагмент - SurfaceView
Я пытаюсь разработать приложение для фонарика для местного концерта. Это часть большого приложения, поэтому оно фрагментарно. Это код:
Сначала я объявил класс вместе с его переменными:
public class ConcertFragment extends Fragment {
ToggleButton btnFlashlight;
View rootView;
private Camera cam;
private boolean hasFlash;
boolean hasCamera;
boolean isFlashOn;
Parameters params;
public ConcertFragment() {
}
Далее идет onActivityCreated
метод, который возвращает метод getCamera (объявлен ниже):
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getCamera();
}
Затем я создаю onCreateView
метод, который строит макет:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
rootView = inflater.inflate(R.layout.fragment_concert, container, false);
hasFlash = getActivity().getApplicationContext().getPackageManager().hasSystemFeature(PackageManager.FEATURE_CAMERA_FLASH);
if (!hasFlash) {
// device doesn't support flash
// Show alert message and close the application
AlertDialog alert = new AlertDialog.Builder(getActivity()).create();
alert.setTitle("No Flash");
alert.setMessage("Sorry, device is not flash supported.");
alert.setButton(DialogInterface.BUTTON_POSITIVE, "OK", new android.content.DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
}
});
alert.show();
}
btnFlashlight = (ToggleButton) rootView.findViewById(R.id.toggleButton);
btnFlashlight.setOnCheckedChangeListener(new OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView,
boolean isChecked) {
if (isFlashOn) {
// turn off flash
turnOffFlash();
} else {
// turn on flash
turnOnFlash();
}
}
});
return rootView;
}
Способ включения вспышки:
private void turnOnFlash() {
if (!isFlashOn) {
if (cam == null || params == null) {
return;
}
params = cam.getParameters();
params.setFlashMode(Parameters.FLASH_MODE_TORCH);
cam.setParameters(params);
cam.startPreview();
isFlashOn = true;
// changing button/switch image
toggleButtonImage();
}
}
Способ выключения вспышки:
private void turnOffFlash() {
if (isFlashOn) {
if (cam == null || params == null) {
return;
}
params = cam.getParameters();
params.setFlashMode(Parameters.FLASH_MODE_OFF);
cam.setParameters(params);
cam.stopPreview();
isFlashOn = false;
// changing button/switch image
toggleButtonImage();
}
}
Способ установки параметров камеры:
private void getCamera() {
if (cam != null) {
try {
cam = Camera.open();
params = cam.getParameters();
cam.startPreview();
hasCamera = true;
} catch (RuntimeException e) {
Log.e("Camera Error. Failed to Open. Error: ", e.getMessage());
}
}
}
private void toggleButtonImage(){
if(isFlashOn){
btnFlashlight.setBackgroundResource(R.drawable.btn_switch_on);
}else{
btnFlashlight.setBackgroundResource(R.drawable.btn_switch_off);
}
}
}
}
Тумблер включения фонарика включается и выключается, но фонарик никогда не включается на задней панели моего Nexus 5.
Я использую следующие разрешения:
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.FLASHLIGHT" />
<uses-permission android:name="android.permission.VIBRATE" />
Может кто-нибудь помочь, пожалуйста?
1 ответ
Я тоже борюсь с той же проблемой. Вам нужно добавить SurfacePreview над кнопкой в макете:
<SurfaceView
android:id="@+id/PREVIEW"
android:layout_width="1dp"
android:layout_height="1dp"/>
и в настоящее время мой фрагмент выглядит так:
public class FlashlightFragment extends Fragment implements SurfaceHolder.Callback {
ImageButton btnSwitch;
//flag to detect flash is on or off
private boolean isLighOn = false;
private Camera camera;
Parameters params;
@Override
public void onStart() {
super.onStart();
SurfaceView preview = (SurfaceView)getView().findViewById(R.id.PREVIEW);
SurfaceHolder mHolder = preview.getHolder();
mHolder.addCallback(this);
}
@Override
public void onPause(){
super.onPause();
turnOffFlash();
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.flashlight_fragment,
container, false);
// flash switch button
btnSwitch = (ImageButton) view.findViewById(R.id.flashlight_button);
// displaying button image
toggleButtonImage();
// Switch button click event to toggle flash on/off
btnSwitch.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (isLighOn) {
// turn off flash
turnOffFlash();
} else {
// turn on flash
turnOnFlash();
}
}
});
return view;
}
// Turning On flash
private void turnOnFlash() {
if (!isLighOn) {
if (camera == null || params == null) {
return;
}
params = camera.getParameters();
params.setFlashMode(Parameters.FLASH_MODE_TORCH);
camera.setParameters(params);
camera.startPreview();
isLighOn = true;
// changing button/switch image
toggleButtonImage();
}
}
// Turning Off flash
private void turnOffFlash() {
if (isLighOn) {
if (camera == null || params == null) {
return;
}
params = camera.getParameters();
params.setFlashMode(Parameters.FLASH_MODE_OFF);
camera.setParameters(params);
camera.stopPreview();
isLighOn = false;
// changing button/switch image
toggleButtonImage();
}
}
/*
* Toggle switch button images
* changing image states to on / off
* */
private void toggleButtonImage(){
if(isLighOn){
btnSwitch.setImageResource(R.drawable.icon_bulb_on);
}else{
btnSwitch.setImageResource(R.drawable.icon_bulb_off);
}
}
@Override
public void surfaceChanged(SurfaceHolder holder,int format,int width,int height) {
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
if (camera != null) {
camera.stopPreview();
camera.setPreviewCallback(null);
camera.release();
camera = null;
}
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
if (camera == null) {
camera = Camera.open();
params = camera.getParameters();
try {
camera.setPreviewDisplay(holder);
} catch (IOException e) {
camera.release();
camera = null;
}
}
}
Похоже, это работает нормально на моем Nexus 5, но принудительно закрывается на Xperia Z в camera.setPreviewDisplay(mHolder); Я еще не смог решить эту проблему.
Надеюсь это поможет.