mirror of
https://github.com/juce-framework/JUCE.git
synced 2026-01-11 23:54:18 +00:00
220 lines
7.5 KiB
C++
220 lines
7.5 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE examples.
|
|
Copyright (c) 2017 - ROLI Ltd.
|
|
|
|
The code included in this file is provided under the terms of the ISC license
|
|
http://www.isc.org/downloads/software-support-policy/isc-license. Permission
|
|
To use, copy, modify, and/or distribute this software for any purpose with or
|
|
without fee is hereby granted provided that the above copyright notice and
|
|
this permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES,
|
|
WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR
|
|
PURPOSE, ARE DISCLAIMED.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
/*******************************************************************************
|
|
The block below describes the properties of this PIP. A PIP is a short snippet
|
|
of code that can be read by the Projucer and used to generate a JUCE project.
|
|
|
|
BEGIN_JUCE_PIP_METADATA
|
|
|
|
name: CameraDemo
|
|
version: 1.0.0
|
|
vendor: JUCE
|
|
website: http://juce.com
|
|
description: Showcases camera features.
|
|
|
|
dependencies: juce_core, juce_cryptography, juce_data_structures, juce_events,
|
|
juce_graphics, juce_gui_basics, juce_gui_extra, juce_video
|
|
exporters: xcode_mac, vs2017, linux_make
|
|
|
|
moduleFlags: JUCE_USE_CAMERA=1
|
|
|
|
type: Component
|
|
mainClass: CameraDemo
|
|
|
|
useLocalCopy: 1
|
|
|
|
END_JUCE_PIP_METADATA
|
|
|
|
*******************************************************************************/
|
|
|
|
#pragma once
|
|
|
|
#include "../Assets/DemoUtilities.h"
|
|
|
|
//==============================================================================
|
|
class CameraDemo : public Component,
|
|
private CameraDevice::Listener,
|
|
private AsyncUpdater
|
|
{
|
|
public:
|
|
CameraDemo()
|
|
{
|
|
setOpaque (true);
|
|
|
|
addAndMakeVisible (cameraSelectorComboBox);
|
|
updateCameraList();
|
|
cameraSelectorComboBox.setSelectedId (1);
|
|
cameraSelectorComboBox.onChange = [this] { cameraChanged(); };
|
|
|
|
addAndMakeVisible (snapshotButton);
|
|
snapshotButton.onClick = [this] { takeSnapshot(); };
|
|
snapshotButton.setEnabled (false);
|
|
|
|
addAndMakeVisible (recordMovieButton);
|
|
recordMovieButton.onClick = [this] { startRecording(); };
|
|
recordMovieButton.setEnabled (false);
|
|
|
|
addAndMakeVisible (lastSnapshot);
|
|
|
|
cameraSelectorComboBox.setSelectedId (2);
|
|
|
|
setSize (500, 500);
|
|
}
|
|
|
|
//==============================================================================
|
|
void paint (Graphics& g) override
|
|
{
|
|
g.fillAll (Colours::black);
|
|
}
|
|
|
|
void resized() override
|
|
{
|
|
auto r = getLocalBounds().reduced (5);
|
|
|
|
auto top = r.removeFromTop (25);
|
|
cameraSelectorComboBox.setBounds (top.removeFromLeft (250));
|
|
|
|
r.removeFromTop (4);
|
|
top = r.removeFromTop (25);
|
|
|
|
snapshotButton.changeWidthToFitText (24);
|
|
snapshotButton.setBounds (top.removeFromLeft (snapshotButton.getWidth()));
|
|
top.removeFromLeft (4);
|
|
recordMovieButton.changeWidthToFitText (24);
|
|
recordMovieButton.setBounds (top.removeFromLeft (recordMovieButton.getWidth()));
|
|
|
|
r.removeFromTop (4);
|
|
auto previewArea = r.removeFromTop (r.getHeight() / 2);
|
|
|
|
if (cameraPreviewComp.get() != nullptr)
|
|
cameraPreviewComp->setBounds (previewArea);
|
|
|
|
r.removeFromTop (4);
|
|
lastSnapshot.setBounds (r);
|
|
}
|
|
|
|
|
|
private:
|
|
//==============================================================================
|
|
ScopedPointer<CameraDevice> cameraDevice;
|
|
ScopedPointer<Component> cameraPreviewComp;
|
|
ImageComponent lastSnapshot;
|
|
|
|
ComboBox cameraSelectorComboBox { "Camera" };
|
|
TextButton snapshotButton { "Take a snapshot" };
|
|
TextButton recordMovieButton { "Record a movie (to your desktop)..." };
|
|
bool recordingMovie = false;
|
|
|
|
void updateCameraList()
|
|
{
|
|
cameraSelectorComboBox.clear();
|
|
cameraSelectorComboBox.addItem ("No camera", 1);
|
|
cameraSelectorComboBox.addSeparator();
|
|
|
|
auto cameras = CameraDevice::getAvailableDevices();
|
|
|
|
for (int i = 0; i < cameras.size(); ++i)
|
|
cameraSelectorComboBox.addItem (cameras[i], i + 2);
|
|
}
|
|
|
|
void cameraChanged()
|
|
{
|
|
// This is called when the user chooses a camera from the drop-down list.
|
|
cameraDevice .reset();
|
|
cameraPreviewComp.reset();
|
|
recordingMovie = false;
|
|
|
|
if (cameraSelectorComboBox.getSelectedId() > 1)
|
|
{
|
|
// Try to open the user's choice of camera..
|
|
cameraDevice.reset (CameraDevice::openDevice (cameraSelectorComboBox.getSelectedId() - 2));
|
|
|
|
// and if it worked, create a preview component for it..
|
|
if (cameraDevice.get() != nullptr)
|
|
{
|
|
cameraPreviewComp.reset (cameraDevice->createViewerComponent());
|
|
addAndMakeVisible (cameraPreviewComp.get());
|
|
}
|
|
}
|
|
|
|
snapshotButton .setEnabled (cameraDevice.get() != nullptr);
|
|
recordMovieButton.setEnabled (cameraDevice.get() != nullptr);
|
|
resized();
|
|
}
|
|
|
|
void startRecording()
|
|
{
|
|
if (cameraDevice.get() != nullptr)
|
|
{
|
|
// The user has clicked the record movie button..
|
|
if (! recordingMovie)
|
|
{
|
|
// Start recording to a file on the user's desktop..
|
|
recordingMovie = true;
|
|
|
|
auto file = File::getSpecialLocation (File::userDesktopDirectory)
|
|
.getNonexistentChildFile ("JuceCameraDemo", CameraDevice::getFileExtension());
|
|
|
|
cameraDevice->startRecordingToFile (file);
|
|
recordMovieButton.setButtonText ("Stop Recording");
|
|
}
|
|
else
|
|
{
|
|
// Already recording, so stop...
|
|
recordingMovie = false;
|
|
cameraDevice->stopRecording();
|
|
recordMovieButton.setButtonText ("Start recording (to a file on your desktop)");
|
|
}
|
|
}
|
|
}
|
|
|
|
void takeSnapshot()
|
|
{
|
|
// When the user clicks the snapshot button, we'll attach ourselves to
|
|
// the camera as a listener, and wait for an image to arrive...
|
|
cameraDevice->addListener (this);
|
|
}
|
|
|
|
// This is called by the camera device when a new image arrives
|
|
void imageReceived (const Image& image) override
|
|
{
|
|
// In this app we just want to take one image, so as soon as this happens,
|
|
// we'll unregister ourselves as a listener.
|
|
if (cameraDevice.get() != nullptr)
|
|
cameraDevice->removeListener (this);
|
|
|
|
// This callback won't be on the message thread, so to get the image back to
|
|
// the message thread, we'll stash a pointer to it (which is reference-counted in
|
|
// a thead-safe way), and trigger an async callback which will then display the
|
|
// new image..
|
|
incomingImage = image;
|
|
triggerAsyncUpdate();
|
|
}
|
|
|
|
Image incomingImage;
|
|
|
|
void handleAsyncUpdate() override
|
|
{
|
|
if (incomingImage.isValid())
|
|
lastSnapshot.setImage (incomingImage);
|
|
}
|
|
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (CameraDemo)
|
|
};
|