Swiftで動画撮影機能を簡単に実装!手順とサンプルコード15選

Swiftでの動画撮影機能の実装方法とサンプルコードSwift
この記事は約56分で読めます。

 

【サイト内のコードはご自由に個人利用・商用利用いただけます】

この記事では、プログラムの基礎知識を前提に話を進めています。

説明のためのコードや、サンプルコードもありますので、もちろん初心者でも理解できるように表現してあります。

基本的な知識があればカスタムコードを使って機能追加、目的を達成できるように作ってあります。

※この記事は、一般的にプロフェッショナルの指標とされる『実務経験10,000時間以上』を凌駕する現役のプログラマチームによって監修されています。

サイト内のコードを共有する場合は、参照元として引用して下さいますと幸いです

※Japanシーモアは、常に解説内容のわかりやすさや記事の品質に注力しております。不具合、分かりにくい説明や不適切な表現、動かないコードなど気になることがございましたら、記事の品質向上の為にお問い合わせフォームにてご共有いただけますと幸いです。
(送信された情報は、プライバシーポリシーのもと、厳正に取扱い、処分させていただきます。)

はじめに

この記事を読めば、Swiftでの動画撮影機能の実装がスムーズにできるようになります。

動画撮影機能は、SNSアプリや動画編集アプリなど、多岐にわたるアプリで求められる機能の一つです。

しかし、初心者の方々にとっては、その実装方法が一筋縄ではいかないことも多いでしょう。

そこで、この記事ではSwiftを用いた動画撮影機能の基本から、実際の使用例を交えたサンプルコードまでを、初心者の方々でも理解しやすいように詳しく解説していきます。

●動画撮影機能とは

動画撮影機能とは、スマートフォンやタブレットのカメラを利用して、動画を撮影し、その動画をアプリ内で保存・再生・共有することができる機能のことを指します。

この機能は、ユーザーがアプリ内で自由に動画を撮影できることから、ユーザーエンゲージメントの向上や、アプリの利用頻度の向上に寄与することが期待されます。

○動画撮影の基本的な流れ

動画撮影の基本的な流れは次の通りです。

  1. ユーザーが動画撮影ボタンを押下します。
  2. カメラアクセスの権限確認が行われます。
  3. 権限が許可された場合、カメラが起動します。
  4. ユーザーが撮影開始ボタンを押下すると、動画の撮影が開始されます。
  5. ユーザーが撮影終了ボタンを押下すると、動画の撮影が終了し、動画が保存されます。
  6. 保存された動画は、アプリ内での再生や、他のユーザーとの共有など、様々な用途で利用することができます。

この基本的な流れを理解した上で、具体的な実装方法やサンプルコードを学んでいくことで、効率よく動画撮影機能の実装が進められます。

●Swiftにおける動画撮影の実装手順

Swiftで動画撮影機能をアプリに実装する際の手順は、緻密な計画と正確なコーディングが求められます。

初心者にとっても分かりやすく進めていくため、必要なライブラリのインストールからアクセス権限の設定までを詳細に解説します。

○必要なライブラリと設定

動画撮影機能をSwiftで実装する前に、必要なライブラリをインストールし、適切に設定することが重要です。

主にAVFoundationというフレームワークを使用します。

AVFoundationは、音声、動画、写真などのメディアに関する豊富なAPIを提供しており、これによって動画撮影機能の実装が可能になります。

まずは、Xcode上で新しいプロジェクトを開始し、次のようにAVFoundationフレームワークをインポートします。

// AVFoundationフレームワークをインポート
import AVFoundation

このフレームワークのインポートにより、カメラやマイクなどのハードウェアへのアクセスや、メディアの録音、再生、編集などが可能になります。

このコードは、動画撮影機能の実装の基盤となる部分です。

○アクセス権限の設定

動画撮影機能を利用するためには、ユーザーからカメラとマイクへのアクセス許可を得る必要があります。

これは、プライバシー保護の観点から非常に重要なプロセスです。

アクセス権限の設定は、アプリのinfo.plistファイルにて行います。

具体的には、info.plistに次の2つのキーを追加します。

  1. Privacy – Camera Usage Description
  2. Privacy – Microphone Usage Description

それぞれのキーに対して、ユーザーに表示される説明文を設定します。

例えば、「カメラは動画撮影のために使用されます」といった具体的な説明が必要です。

これによって、ユーザーはどのような目的でアクセス権限が要求されているのかを明確に理解し、許可するかどうかを判断することができます。

●サンプルコード紹介

Swiftを利用した動画撮影機能の実装は、非常に多岐にわたります。

ここでは、基本的な動画撮影からカメラの切り替えまでのサンプルコードを2つ紹介します。

○サンプルコード1:基本的な動画撮影

初めに、Swiftでの基本的な動画撮影を実現する方法を見ていきましょう。

このコードでは、AVFoundationフレームワークを使用して、シンプルな動画撮影機能を実装します。

import UIKit
import AVFoundation

class VideoCaptureViewController: UIViewController, AVCaptureFileOutputRecordingDelegate {

    var captureSession: AVCaptureSession!
    var videoFileOutput: AVCaptureMovieFileOutput!

    override func viewDidLoad() {
        super.viewDidLoad()

        captureSession = AVCaptureSession()
        captureSession.sessionPreset = .hd1920x1080

        guard let videoDevice = AVCaptureDevice.default(for: .video) else { return }
        guard let videoInput = try? AVCaptureDeviceInput(device: videoDevice) else { return }
        captureSession.addInput(videoInput)

        videoFileOutput = AVCaptureMovieFileOutput()
        captureSession.addOutput(videoFileOutput)

        let previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
        previewLayer.frame = view.bounds
        view.layer.addSublayer(previewLayer)

        captureSession.startRunning()
    }

    @IBAction func startRecording(_ sender: Any) {
        let outputDirectory = FileManager.default.temporaryDirectory
        let outputFilePath = outputDirectory.appendingPathComponent(UUID().uuidString).appendingPathExtension("mov")
        videoFileOutput.startRecording(to: outputFilePath, recordingDelegate: self)
    }

    @IBAction func stopRecording(_ sender: Any) {
        videoFileOutput.stopRecording()
    }

    func fileOutput(_ output: AVCaptureFileOutput, didFinishRecordingTo outputFileURL: URL, from connections: [AVCaptureConnection], error: Error?) {
        if let error = error {
            print("録画エラー: \(error.localizedDescription)")
        } else {
            print("録画完了: \(outputFileURL)")
        }
    }
}

このコードでは、AVCaptureSessionを使って動画の撮影セッションを設定しています。

撮影開始の際は、startRecordingメソッドを呼び出し、撮影終了の際は、stopRecordingメソッドを呼び出します。

動画は一時的なディレクトリに保存され、撮影が完了したらファイルパスがコンソールに出力されます。

○サンプルコード2:フロントカメラとバックカメラの切り替え

多くのモバイルデバイスには、フロントカメラとバックカメラの2つのカメラが搭載されています。

ユーザーがこれらのカメラを簡単に切り替えられるようにするためのサンプルコードを紹介します。

import UIKit
import AVFoundation

class CameraSwitchViewController: UIViewController {

    var captureSession: AVCaptureSession!
    var currentCameraPosition: AVCaptureDevice.Position = .back

    override func viewDidLoad() {
        super.viewDidLoad()
        setupCaptureSession(cameraPosition: currentCameraPosition)
    }

    func setupCaptureSession(cameraPosition: AVCaptureDevice.Position) {
        captureSession = AVCaptureSession()

        guard let videoDevice = AVCaptureDevice.default(.builtInWideAngleCamera, for: .video, position: cameraPosition) else { return }
        guard let videoInput = try? AVCaptureDeviceInput(device: videoDevice) else { return }
        captureSession.addInput(videoInput)

        let previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
        previewLayer.frame = view.bounds
        view.layer.addSublayer(previewLayer)

        captureSession.startRunning()
    }

    @IBAction func switchCamera(_ sender: Any) {
        captureSession.stopRunning()
        captureSession = nil
        currentCameraPosition = (currentCameraPosition == .back) ? .front : .back
        setupCaptureSession(cameraPosition: currentCameraPosition)
    }
}

このコードでは、setupCaptureSessionメソッドを使用してカメラのセッションを設定しています。カメラの切り替えは、switchCameraメソッドを呼び出すことで行います。

このメソッドでは、現在のカメラの位置を判定し、それに基づいてフロントカメラとバックカメラを切り替えます。

○サンプルコード3:動画の保存と再生

アプリで動画を撮影した後、その動画を保存して再生する機能はユーザーエクスペリエンスを向上させる要素として非常に重要です。

ここでは、Swiftで動画を保存し、再生するための実装手順とサンプルコードをご紹介します。

まず、動画をローカルに保存する方法を見ていきましょう。

このコードでは、AVAssetExportSessionを使用して、動画をアルバムに保存します。

import UIKit
import AVFoundation
import Photos

class SaveVideoViewController: UIViewController {

    var videoURL: URL?

    @IBAction func saveVideoToAlbum(_ sender: Any) {
        guard let url = videoURL else { return }

        // アルバムに動画を保存
        PHPhotoLibrary.shared().performChanges({
            PHAssetChangeRequest.creationRequestForAssetFromVideo(atFileURL: url)
        }) { (success, error) in
            DispatchQueue.main.async {
                if success {
                    print("動画をアルバムに保存しました。")
                } else {
                    print("動画の保存に失敗しました。")
                }
            }
        }
    }
}

このコードではPHPhotoLibrary.shared().performChangesを使って動画をアルバムに保存しています。

保存が成功した場合や失敗した場合の処理も含めています。

次に、撮影した動画を再生する方法を見ていきましょう。

このコードでは、AVPlayerViewControllerを使って動画を再生します。

import UIKit
import AVKit
import AVFoundation

class PlayVideoViewController: UIViewController {

    var videoURL: URL?

    @IBAction func playVideo(_ sender: Any) {
        guard let url = videoURL else { return }

        let player = AVPlayer(url: url)
        let playerViewController = AVPlayerViewController()
        playerViewController.player = player
        present(playerViewController, animated: true) {
            playerViewController.player?.play()
        }
    }
}

このコードでは、ボタンを押すと、playVideoメソッドが呼び出され、AVPlayerViewControllerを使って動画を再生します。

AVPlayerViewControllerは再生コントロールやフルスクリーン表示などの機能を持っています。

○サンプルコード4:撮影時のカスタムUI設定

アプリに動画撮影機能を追加する際、ユーザーの操作性やアプリのブランドイメージに合わせてUIをカスタマイズすることがよくあります。

ここでは、撮影画面のUIをカスタマイズする方法について説明します。

このコードでは、AVCaptureVideoPreviewLayerを利用して、カメラのプレビュー表示を行い、上にカスタムのUIを配置しています。

import UIKit
import AVFoundation

class CustomUIVideoCaptureViewController: UIViewController {

    var captureSession: AVCaptureSession!
    var previewLayer: AVCaptureVideoPreviewLayer!

    override func viewDidLoad() {
        super.viewDidLoad()

        captureSession = AVCaptureSession()

        guard let videoDevice = AVCaptureDevice.default(for: .video) else { return }
        guard let videoInput = try? AVCaptureDeviceInput(device: videoDevice) else { return }
        captureSession.addInput(videoInput)

        // カメラのプレビュー表示
        previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
        previewLayer.frame = view.bounds
        view.layer.addSublayer(previewLayer)

        // カスタムのUIを配置
        let customButton = UIButton(frame: CGRect(x: 150, y: 400, width: 70, height: 30))
        customButton.setTitle("撮影", for: .normal)
        customButton.backgroundColor = .red
        customButton.addTarget(self, action: #selector(startRecording), for: .touchUpInside)
        view.addSubview(customButton)

        captureSession.startRunning()
    }

    @objc func startRecording() {
        // ここで録画の開始などの処理を行う
    }
}

このコードでは、撮影画面の中心部にカスタムの「撮影」ボタンを配置しています。

ボタンが押されたときには、startRecordingメソッドが呼び出されます。

○サンプルコード5:フィルターやエフェクトの追加

動画撮影アプリを開発する際、撮影した動画にフィルターやエフェクトを追加する機能はユーザーからの要望が高いものの一つです。

Swiftでの実装方法を詳しく見ていきましょう。

このコードでは、CIFilterを用いて動画にフィルターを適用します。

CIFilterは様々なエフェクトやフィルターを簡単に実装するためのツールです。

import UIKit
import AVFoundation
import CoreImage

class FilterVideoViewController: UIViewController, AVCaptureFileOutputRecordingDelegate {

    var captureSession: AVCaptureSession!
    var videoOutput: AVCaptureMovieFileOutput!
    var previewLayer: AVCaptureVideoPreviewLayer!
    var filter: CIFilter!

    override func viewDidLoad() {
        super.viewDidLoad()

        captureSession = AVCaptureSession()

        guard let videoDevice = AVCaptureDevice.default(for: .video) else { return }
        guard let videoInput = try? AVCaptureDeviceInput(device: videoDevice) else { return }
        captureSession.addInput(videoInput)

        videoOutput = AVCaptureMovieFileOutput()
        captureSession.addOutput(videoOutput)

        // カメラのプレビュー表示
        previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
        previewLayer.frame = view.bounds
        view.layer.addSublayer(previewLayer)

        // フィルターの設定
        filter = CIFilter(name: "CISepiaTone")
        previewLayer.filters = [filter]

        captureSession.startRunning()
    }

    @IBAction func startRecording(_ sender: Any) {
        let outputFilePath = NSTemporaryDirectory() + "output.mov"
        let outputURL = URL(fileURLWithPath: outputFilePath)
        videoOutput.startRecording(to: outputURL, recordingDelegate: self)
    }
}

この例では、セピア調のフィルター「CISepiaTone」を動画に適用しています。

CIFilterには多くのフィルターやエフェクトが用意されているので、好みに応じて選択できます。

○サンプルコード6:動画のトリミング

動画撮影後、特定の部分だけを取り出して保存したいという要望は一般的です。

こちらでは、Swiftを使って動画をトリミングする方法を紹介します。

このコードでは、AVAssetExportSessionを利用して、指定された時間範囲の動画をトリミングします。

import UIKit
import AVFoundation

class TrimVideoViewController: UIViewController {

    var videoURL: URL?

    func trimVideo(startTime: Float64, endTime: Float64) {
        guard let url = videoURL else { return }
        let asset = AVAsset(url: url)
        let timeRange = CMTimeRange(start: CMTimeMakeWithSeconds(startTime, preferredTimescale: 600), end: CMTimeMakeWithSeconds(endTime, preferredTimescale: 600))

        let exportSession = AVAssetExportSession(asset: asset, presetName: AVAssetExportPresetHighestQuality)!
        exportSession.timeRange = timeRange
        exportSession.outputURL = URL(fileURLWithPath: NSTemporaryDirectory() + "trimmed.mov")
        exportSession.outputFileType = .mov

        exportSession.exportAsynchronously {
            if exportSession.status == .completed {
                DispatchQueue.main.async {
                    print("動画をトリミングしました。")
                }
            } else if let error = exportSession.error {
                DispatchQueue.main.async {
                    print("エラー発生: \(error)")
                }
            }
        }
    }
}

このコードでは、startTimeendTimeの間の動画をトリミングして保存します。

これにより、必要なシーンだけを簡単に取り出すことができます。

○サンプルコード7:動画の合成

動画の合成は、複数の動画クリップや静止画を一つの動画にまとめる技術です。

例えば、動画の冒頭にタイトル画面を追加したり、エンドロールを追加する際などに使用します。

Swiftでは、AVMutableCompositionを使用して動画の合成を行います。

このコードでは、AVMutableCompositionを使って2つの動画クリップを一つの動画に合成する手法を紹介します。

この例では、clip1clip2の2つの動画を続けて再生する一つの動画を生成しています。

import AVFoundation

func mergeVideos(clip1: URL, clip2: URL, outputURL: URL) {
    let mixComposition = AVMutableComposition()

    guard let firstAsset = AVAsset(url: clip1), let secondAsset = AVAsset(url: clip2) else { return }

    let firstTrack = mixComposition.addMutableTrack(withMediaType: .video, preferredTrackID: Int32(kCMPersistentTrackID_Invalid))
    let secondTrack = mixComposition.addMutableTrack(withMediaType: .video, preferredTrackID: Int32(kCMPersistentTrackID_Invalid))

    do {
        try firstTrack?.insertTimeRange(CMTimeRangeMake(start: CMTime.zero, duration: firstAsset.duration), of: firstAsset.tracks(withMediaType: .video)[0], at: CMTime.zero)
        try secondTrack?.insertTimeRange(CMTimeRangeMake(start: CMTime.zero, duration: secondAsset.duration), of: secondAsset.tracks(withMediaType: .video)[0], at: firstAsset.duration)
    } catch {
        print("エラー発生: \(error.localizedDescription)")
    }

    let exporter = AVAssetExportSession(asset: mixComposition, presetName: AVAssetExportPresetHighestQuality)!
    exporter.outputURL = outputURL
    exporter.outputFileType = .mov
    exporter.shouldOptimizeForNetworkUse = true

    exporter.exportAsynchronously {
        switch exporter.status {
        case .completed:
            print("動画を合成しました。")
        case .failed, .cancelled:
            print("動画の合成に失敗しました: \(String(describing: exporter.error))")
        default:
            break
        }
    }
}

この関数を使用することで、指定した2つの動画を続けて再生する一つの動画を生成できます。

出力先のURLを指定することで、動画を任意の場所に保存することも可能です。

○サンプルコード8:BGMの追加

動画に背景音楽を追加することで、よりプロフェッショナルな仕上がりになります。

SwiftでBGMを動画に追加する方法を見ていきましょう。

このコードでは、AVMutableCompositionを使って、動画に背景音楽を追加しています。

この例では、videoURLの動画にaudioURLの音楽を背景音楽として追加しています。

import AVFoundation

func addBackgroundMusicToVideo(videoURL: URL, audioURL: URL, outputURL: URL) {
    let mixComposition = AVMutableComposition()
    guard let asset = AVAsset(url: videoURL), let audioAsset = AVAsset(url: audioURL) else { return }

    let videoTrack = mixComposition.addMutableTrack(withMediaType: .video, preferredTrackID: Int32(kCMPersistentTrackID_Invalid))
    let audioTrack = mixComposition.addMutableTrack(withMediaType: .audio, preferredTrackID: Int32(kCMPersistentTrackID_Invalid))

    do {
        try videoTrack?.insertTimeRange(CMTimeRangeMake(start: CMTime.zero, duration: asset.duration), of: asset.tracks(withMediaType: .video)[0], at: CMTime.zero)
        try audioTrack?.insertTimeRange(CMTimeRangeMake(start: CMTime.zero, duration: asset.duration), of: audioAsset.tracks(withMediaType: .audio)[0], at: CMTime.zero)
    } catch {
        print("エラー発生: \(error.localizedDescription)")
    }

    let exporter = AVAssetExportSession(asset: mixComposition, presetName: AVAssetExportPresetHighestQuality)!
    exporter.outputURL = outputURL
    exporter.outputFileType = .mov

    exporter.exportAsynchronously {
        switch exporter.status {
        case .completed:
            print("BGMを動画に追加しました。")
        case .failed, .cancelled:
            print("BGMの追加に失敗しました: \(String(describing: exporter.error))")
        default:
            break
        }
    }
}

この関数を使用することで、指定した動画に指定した音楽を背景音楽として追加することができます。

出力先のURLを指定することで、編集後の動画を任意の場所に保存することも可能です。

○サンプルコード9:動画のアップロード機能

動画のアップロード機能は、特定のサーバーやクラウドストレージに動画を送信するための機能です。

アプリで撮影した動画を共有する際や、オンラインでの編集を目的とした時にこの機能を利用します。

Swiftでは、URLSessionを使用して動画をアップロードすることができます。

このコードでは、URLSessionを使って指定したURLへ動画をアップロードする方法を表しています。

この例では、指定したuploadURLvideoFileURLの動画をアップロードする処理を行っています。

import Foundation

func uploadVideo(videoFileURL: URL, uploadURL: URL) {
    var request = URLRequest(url: uploadURL)
    request.httpMethod = "POST"
    request.setValue("video/mp4", forHTTPHeaderField: "Content-Type")

    let task = URLSession.shared.uploadTask(with: request, fromFile: videoFileURL) { data, response, error in
        if let error = error {
            print("アップロードエラー: \(error.localizedDescription)")
            return
        }
        if let data = data, let responseText = String(data: data, encoding: .utf8) {
            print("アップロード成功: \(responseText)")
        }
    }
    task.resume()
}

上記の関数を使用すると、指定したURLへ動画を簡単にアップロードすることができます。

動画のアップロードが成功すると、サーバーからのレスポンスを取得して表示します。

○サンプルコード10:動画の編集機能

動画の編集機能により、撮影した動画の内容をより魅力的に、または目的に合わせて変更することができます。

Swiftでは、AVMutableCompositionを使用して動画の編集を行うことができます。

このコードでは、AVMutableCompositionを使って動画の一部をカットして新しい動画を生成する方法を紹介します。

この例では、originalVideoURLの動画の指定した時間範囲をカットして新しい動画を生成しています。

import AVFoundation

func trimVideo(originalVideoURL: URL, outputURL: URL, startTime: Double, endTime: Double) {
    let asset = AVAsset(url: originalVideoURL)
    let exportSession = AVAssetExportSession(asset: asset, presetName: AVAssetExportPresetHighestQuality)!
    let timeRange = CMTimeRange(start: CMTime(seconds: startTime, preferredTimescale: 600), end: CMTime(seconds: endTime, preferredTimescale: 600))

    exportSession.outputURL = outputURL
    exportSession.outputFileType = .mov
    exportSession.timeRange = timeRange

    exportSession.exportAsynchronously {
        switch exportSession.status {
        case .completed:
            print("動画の編集が完了しました。")
        case .failed, .cancelled:
            print("動画の編集に失敗しました: \(String(describing: exportSession.error))")
        default:
            break
        }
    }
}

この関数を使用することで、指定した時間範囲の動画を簡単にカットして新しい動画を生成することができます。

出力先のURLを指定することで、編集後の動画を任意の場所に保存することも可能です。

○サンプルコード11:動画のシェア機能

動画のシェア機能は、撮影した動画をSNSやメッセージアプリなどといった外部のサービスを利用して共有するための機能です。

SwiftとiOSプラットフォームは、UIActivityViewControllerというクラスを提供しており、これを使用することで簡単に動画やその他のコンテンツをシェアすることができます。

このコードでは、UIActivityViewControllerを使って動画をシェアする方法を紹介します。

この例では、指定したvideoFileURLの動画をシェアするための画面を表示します。

import UIKit

func shareVideo(videoFileURL: URL, fromViewController: UIViewController) {
    let activityViewController = UIActivityViewController(activityItems: [videoFileURL], applicationActivities: nil)
    fromViewController.present(activityViewController, animated: true, completion: nil)
}

この関数を使用すると、アプリ内で指定した動画をSNSやメッセージアプリなどといった外部のサービスと共有するための画面が表示されます。

ユーザーは、この画面から好きなサービスを選択し、動画をシェアすることができます。

シェア画面を表示する際には、呼び出し元となるViewControllerの参照も渡す必要があります。

これは、シェア画面をそのViewController上にモーダルとして表示するためです。

動画をシェアする際には、ユーザーのプライバシーに関する懸念も考慮する必要があります。

特に、位置情報が含まれる動画など、機密性の高い情報を含む動画をシェアする際には、ユーザーに明確に情報を伝え、了解を得た上でシェアを実行するようにしてください。

○サンプルコード12:動画のダウンロード機能

動画のダウンロード機能は、サーバーやクラウド上に保存されている動画をアプリ内にダウンロードするための機能です。

Swiftでは、URLSessionを使用して、指定したURLから動画をダウンロードすることができます。

このコードでは、URLSessionを使って指定したURLから動画をダウンロードし、その結果を保存する方法を表しています。

この例では、指定したdownloadURLから動画をダウンロードし、destinationURLに保存する処理を行っています。

import Foundation

func downloadVideo(downloadURL: URL, destinationURL: URL, completion: @escaping (Bool) -> Void) {
    let task = URLSession.shared.downloadTask(with: downloadURL) { temporaryURL, response, error in
        guard let temporaryURL = temporaryURL else {
            completion(false)
            return
        }
        do {
            try FileManager.default.moveItem(at: temporaryURL, to: destinationURL)
            completion(true)
        } catch {
            completion(false)
        }
    }
    task.resume()
}

この関数を使用すると、指定したURLから動画をダウンロードし、指定した場所に保存することができます。

ダウンロードが成功したかどうかは、コンプリーションハンドラを通じて取得することができます。

ダウンロード機能を利用する際には、インターネット接続の品質やダウンロードに要する時間を考慮し、ユーザーに進捗情報やエラーメッセージを適切に伝えるように心掛けてください。

○サンプルコード13:撮影モードの選択

動画撮影の際、撮影モードの選択は非常に重要です。

例えば、普通の撮影モードやスローモーション、タイムラプスなど、様々な撮影モードが考えられます。

Swiftを使用すると、これらの撮影モードを簡単に切り替えることができます。

このコードでは、AVCaptureMovieFileOutputを使用して撮影モードを切り替える方法を紹介します。

この例では、通常の撮影モードとスローモーション撮影の切り替えを行っています。

import AVFoundation

class VideoCaptureManager: NSObject {
    var captureSession: AVCaptureSession?
    var movieOutput: AVCaptureMovieFileOutput?

    init(sessionPreset: AVCaptureSession.Preset) {
        super.init()
        captureSession = AVCaptureSession()
        captureSession?.sessionPreset = sessionPreset
        movieOutput = AVCaptureMovieFileOutput()

        if let captureSession = captureSession {
            if captureSession.canAddOutput(movieOutput!) {
                captureSession.addOutput(movieOutput!)
            }
        }
    }

    func setSlowMotionMode(isSlowMotion: Bool) {
        if let format = movieOutput?.availableMovieCodecTypes.first {
            if isSlowMotion {
                movieOutput?.setMaxRecordedDuration(CMTime(value: 2, timescale: 1))
                movieOutput?.setOutputSettings([AVVideoCodecKey: format,
                                                AVVideoScalingModeKey: AVVideoScalingModeResizeAspectFill], 
                                               for: movieOutput!.connection(with: .video)!)
            } else {
                movieOutput?.setMaxRecordedDuration(CMTime(value: 0, timescale: 1))
            }
        }
    }
}

上記のVideoCaptureManagerクラスでは、setSlowMotionModeメソッドを使用することで、スローモーションの撮影モードと通常の撮影モードを簡単に切り替えることができます。

このメソッドは、スローモーションモードを利用する場合にはisSlowMotiontrueに設定し、通常の撮影モードを利用する場合にはfalseに設定します。

このコードを利用することで、動画の撮影中にも簡単に撮影モードの切り替えが可能となります。

また、他の撮影モードへの拡張も容易に行える構造になっています。

実際にアプリを実行すると、撮影ボタンや設定ボタンをタップすることで、瞬時に撮影モードの切り替えが行えます。

これにより、ユーザーはシチュエーションに合わせて撮影モードを選択することができ、よりクリエイティブな動画撮影が可能となります。

○サンプルコード14:解像度の設定

動画を撮影する際、解像度は画質やファイルサイズに大きな影響を及ぼします。

高解像度の動画は詳細で鮮明な画像を提供しますが、ファイルサイズが大きくなるというデメリットもあります。

逆に、低解像度の動画はファイルサイズは小さくなりますが、画質が低下します。

Swiftでは、AVCaptureSession.Presetを利用することで、動画の解像度を簡単に設定することができます。

このコードでは、AVCaptureSession.Presetを使って動画の解像度を変更する方法を表しています。

この例では、HDの解像度と4Kの解像度の切り替えを行っています。

import AVFoundation

extension VideoCaptureManager {

    func setResolution(isHD: Bool) {
        guard let captureSession = captureSession else { return }

        if isHD {
            if captureSession.canSetSessionPreset(.hd1920x1080) {
                captureSession.sessionPreset = .hd1920x1080
            }
        } else {
            if captureSession.canSetSessionPreset(.hd4K3840x2160) {
                captureSession.sessionPreset = .hd4K3840x2160
            }
        }
    }
}

このコードを使用することで、アプリ内で簡単に動画の解像度を変更することができます。

ユーザーインターフェースに解像度切り替えのボタンを追加して、上記のsetResolutionメソッドを呼び出すことで、動的に解像度の変更が可能です。

実際にアプリを実行すると、設定メニューや解像度切り替えのボタンをタップすることで、瞬時に動画の解像度の変更が行えます。

これにより、ユーザーはストレージの空き容量や画質のニーズに応じて、適切な解像度を選択することができます。

○サンプルコード15:動画のメタデータ編集

動画ファイルには、撮影日時やカメラの設定などの情報がメタデータとして埋め込まれています。

このメタデータを編集することで、動画の詳細情報をカスタマイズしたり、特定の情報を隠蔽することが可能です。

このコードでは、AVAssetExportSessionAVMutableMetadataItemを使用して、動画のメタデータを編集する方法を紹介します。

この例では、動画のタイトルと説明を編集しています。

import AVFoundation

func editVideoMetadata(inputURL: URL, outputURL: URL, title: String, description: String, completion: @escaping (Bool) -> Void) {
    let asset = AVAsset(url: inputURL)
    let exportSession = AVAssetExportSession(asset: asset, presetName: AVAssetExportPresetPassthrough)

    let titleItem = AVMutableMetadataItem()
    titleItem.keySpace = .iTunes
    titleItem.key = AVMetadataiTunesMetadataKeySongName as (NSCopying & NSObjectProtocol)
    titleItem.value = title as (NSCopying & NSObjectProtocol)

    let descriptionItem = AVMutableMetadataItem()
    descriptionItem.keySpace = .iTunes
    descriptionItem.key = AVMetadataiTunesMetadataKeyUserComment as (NSCopying & NSObjectProtocol)
    descriptionItem.value = description as (NSCopying & NSObjectProtocol)

    exportSession?.metadata = [titleItem, descriptionItem]
    exportSession?.outputURL = outputURL
    exportSession?.outputFileType = .mp4

    exportSession?.exportAsynchronously(completionHandler: {
        switch exportSession?.status {
        case .completed:
            completion(true)
        default:
            completion(false)
        }
    })
}

上記のeditVideoMetadata関数を使用することで、指定した動画ファイルのタイトルと説明を編集することができます。

この関数は、元の動画ファイルを変更せず、新しい動画ファイルとして出力します。

これにより、元の動画ファイルを安全に保持することができます。

このコードを使用すると、ユーザーは撮影した動画のタイトルや説明を簡単にカスタマイズすることができます。

また、他のメタデータの項目も同様の方法で編集することが可能です。

●動画撮影機能の応用例

動画撮影機能の基本的な実装を学んだ後、さらなる機能拡張やカスタマイズを追求したくなるのは自然なことです。

Swiftでの動画撮影を更に進化させるための応用的な実装方法を紹介します。

○リアルタイムフィルターの実装

動画を撮影する際、リアルタイムでフィルターやエフェクトを適用することで、撮影結果を即時確認しながらクリエイティブな映像を作成することができます。

ここでは、リアルタイムでフィルターを適用する方法をSwiftでのサンプルコードを交えて紹介します。

このコードでは、CIFilterを使ってリアルタイムのフィルター処理を実装しています。

この例では、セピアフィルターを適用しています。

import AVFoundation
import CoreImage

class RealTimeFilterCapture: NSObject, AVCaptureVideoDataOutputSampleBufferDelegate {
    var captureSession: AVCaptureSession!
    var videoPreviewLayer: AVCaptureVideoPreviewLayer!
    var context: CIContext!

    func startSession() {
        captureSession = AVCaptureSession()
        captureSession.sessionPreset = .hd1920x1080

        guard let backCamera = AVCaptureDevice.default(for: .video) else { return }
        do {
            let input = try AVCaptureDeviceInput(device: backCamera)
            if captureSession.canAddInput(input) {
                captureSession.addInput(input)

                let videoOutput = AVCaptureVideoDataOutput()
                videoOutput.setSampleBufferDelegate(self, queue: DispatchQueue.main)
                if captureSession.canAddOutput(videoOutput) {
                    captureSession.addOutput(videoOutput)

                    videoPreviewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
                    videoPreviewLayer.videoGravity = .resizeAspectFill

                    context = CIContext()
                    captureSession.startRunning()
                }
            }
        } catch {
            print("Error: \(error.localizedDescription)")
        }
    }

    func captureOutput(_ output: AVCaptureOutput, didOutput sampleBuffer: CMSampleBuffer, from connection: AVCaptureConnection) {
        guard let pixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer) else { return }

        let ciImage = CIImage(cvPixelBuffer: pixelBuffer)
        if let sepiaFilter = CIFilter(name: "CISepiaTone") {
            sepiaFilter.setValue(ciImage, forKey: kCIInputImageKey)
            sepiaFilter.setValue(1.0, forKey: kCIInputIntensityKey)

            if let outputImage = sepiaFilter.outputImage {
                let cgImage = context.createCGImage(outputImage, from: outputImage.extent)
                DispatchQueue.main.async {
                    self.videoPreviewLayer.contents = cgImage
                }
            }
        }
    }
}

上記のコードは、撮影中の映像にリアルタイムでセピアフィルターを適用する例です。

このような方法で、様々なフィルターやエフェクトを動画撮影中にリアルタイムで適用することができます。

○マルチカメラの同時撮影

最近のスマートフォンには複数のカメラが搭載されており、これを利用して複数のカメラからの映像を同時に撮影することが考えられます。

ここでは、複数のカメラを利用して同時に動画を撮影する方法を紹介します。

このコードでは、フロントカメラとバックカメラの映像を同時にキャプチャする方法を表しています。

import AVFoundation

class MultiCameraCapture: NSObject {
    var captureSession: AVCaptureSession!

    func startSession() {
        captureSession = AVCaptureSession()

        guard let frontCamera = AVCaptureDevice.default(.builtInWideAngleCamera, for: .video, position: .front),
              let backCamera = AVCaptureDevice.default(.builtInWideAngleCamera, for: .video, position: .back) else { return }

        do {
            let frontInput = try AVCaptureDeviceInput(device: frontCamera)
            let backInput = try AVCaptureDeviceInput(device: backCamera)

            if captureSession.canAddInput(frontInput) && captureSession.canAddInput(backInput) {
                captureSession.addInput(frontInput)
                captureSession.addInput(backInput)

                captureSession.startRunning()
            }
        } catch {
            print("Error: \(error.localizedDescription)")
        }
    }
}

上記のコードを使用すると、フロントカメラとバックカメラからの映像を同時に取得することができます。

これにより、例えばインタビューなどのシーンで、質問者と回答者の反応を同時にキャッチするなどの応用が考えられます。

●注意点と対処法

動画撮影機能をSwiftで実装する際には、様々な注意点やトラブルが考えられます。

初心者向けに動画撮影機能を簡単に実装する方法を紹介する中で、具体的なトラブルやそれに対する対処法を詳しく解説します。

○アクセス権限の取得失敗時の対処法

アプリがユーザーのカメラやマイクにアクセスするためには、ユーザーからの許可が必要です。

しかし、許可の取得がうまくいかない場合や、許可が拒否された場合の対処が必要となります。

このコードでは、カメラのアクセス権限をリクエストし、許可されていない場合にはユーザーに通知する処理を行っています。

import AVFoundation

func requestCameraPermission() {
    let status = AVCaptureDevice.authorizationStatus(for: .video)
    switch status {
    case .authorized:
        // 許可されている場合の処理
        break
    case .denied, .restricted:
        // 許可されていない場合の通知
        notifyUserPermissionDenied()
    case .notDetermined:
        AVCaptureDevice.requestAccess(for: .video) { granted in
            if !granted {
                self.notifyUserPermissionDenied()
            }
        }
    default:
        break
    }
}

func notifyUserPermissionDenied() {
    // ユーザーにカメラのアクセス権限がないことを通知する処理
}

この例では、まず現在のアクセス権限の状態を確認します。

もし、アクセス権限が確定していない場合は、リクエストを行います。

その結果、アクセス権限が拒否された場合、ユーザーに通知するnotifyUserPermissionDeniedメソッドを呼び出します。

○動画の保存失敗時の対処法

動画の保存に失敗する原因としては、デバイスのストレージが不足している、ファイルの形式やエンコードの問題などが考えられます。

失敗した際には適切なエラーメッセージを表示してユーザーに情報を提供することが重要です。

このコードでは、動画の保存時にエラーが発生した場合、エラー内容に応じてエラーメッセージを表示する処理を行っています。

import Photos

func saveVideoToPhotoLibrary(url: URL) {
    PHPhotoLibrary.shared().performChanges({
        PHAssetChangeRequest.creationRequestForAssetFromVideo(atFileURL: url)
    }) { (success, error) in
        if !success {
            if let err = error as NSError? {
                switch err.code {
                case -1: // 一般的なエラー
                    self.displayErrorMessage(message: "動画の保存に失敗しました。")
                case -5000: // 保存の権限がない
                    self.displayErrorMessage(message: "動画の保存権限がありません。")
                case -5005: // ストレージ不足
                    self.displayErrorMessage(message: "ストレージが不足しています。不要なデータを削除してください。")
                default:
                    self.displayErrorMessage(message: "不明なエラーが発生しました。")
                }
            }
        }
    }
}

func displayErrorMessage(message: String) {
    // エラーメッセージを表示する処理
}

この例では、動画の保存を行うsaveVideoToPhotoLibraryメソッド内でエラーが発生した際、エラーコードに応じて異なるエラーメッセージを表示します。

これにより、ユーザーは具体的なトラブルの原因を把握しやすくなります。

○動画の読み込み失敗時の対処法

動画の読み込みに失敗する原因としては、動画ファイルが破損している、サポートされていないファイル形式である、デバイスの性能の問題などが考えられます。

エラーが発生した際の対処法としては、適切なエラーメッセージの表示や、サポートしているファイル形式の案内などが考えられます。

このコードでは、動画の読み込み時にエラーが発生した場合の処理を行っています。

import AVFoundation

func loadVideo(url: URL) {
    let asset = AVAsset(url: url)
    let playerItem = AVPlayerItem(asset: asset)
    playerItem.addObserver(self, forKeyPath: "status", options: .new, context: nil)
}

override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
    if keyPath == "status" {
        if let item = object as? AVPlayerItem {
            switch item.status {
            case .failed:
                displayErrorMessage(message: "動画の読み込みに失敗しました。サポートされているファイル形式をご確認ください。")
            default:
                break
            }
        }
    }
}

func displayErrorMessage(message: String) {
    // エラーメッセージを表示する処理
}

上記のコードでは、AVPlayerItemのステータスを監視し、読み込み失敗時にはエラーメッセージを表示しています。

このようにしてユーザーに動画の読み込み問題を通知し、適切な対処を促すことができます。

●カスタマイズの方法

Swiftでの動画撮影機能を実装する際に、そのデザインや動画フォーマットをカスタマイズして、独自のUIや特定の品質要件を満たす動画を生成する方法を学びましょう。

ここでは、UIデザインのカスタマイズと動画フォーマットのカスタマイズについて具体的なサンプルコードとともに解説します。

○UIデザインのカスタマイズ

デフォルトのカメラUIではなく、独自のデザインや機能を持ったカメラUIを実装する場合があります。

例えば、特定のボタンの位置や色、サイズを変更したり、新しいボタンや機能を追加することが考えられます。

このコードでは、カメラのシャッターボタンのデザインをカスタマイズしています。

import UIKit
import AVFoundation

class CustomCameraViewController: UIViewController {

    var captureSession: AVCaptureSession!
    var shutterButton: UIButton!

    override func viewDidLoad() {
        super.viewDidLoad()
        setupCamera()
        setupUI()
    }

    func setupCamera() {
        // カメラのセットアップ処理
    }

    func setupUI() {
        shutterButton = UIButton(frame: CGRect(x: (view.frame.width - 80) / 2, y: view.frame.height - 100, width: 80, height: 80))
        shutterButton.layer.cornerRadius = 40
        shutterButton.backgroundColor = UIColor.red
        shutterButton.addTarget(self, action: #selector(shutterButtonPressed), for: .touchUpInside)
        view.addSubview(shutterButton)
    }

    @objc func shutterButtonPressed() {
        // シャッターボタンが押されたときの処理
    }
}

この例では、赤い円形のシャッターボタンを画面の下部に表示しています。

このようにして、標準のシャッターボタンではなく、オリジナルのデザインや位置にカスタマイズすることができます。

○動画フォーマットのカスタマイズ

動画の品質やサイズ、圧縮率をカスタマイズする際には、動画のフォーマットやエンコーディング設定を変更する必要があります。

これにより、保存される動画のサイズや品質を調整することができます。

このコードでは、動画のエンコーディング設定をカスタマイズしています。

import AVFoundation

func setupVideoOutput() {
    let videoOutput = AVCaptureMovieFileOutput()

    let compressionProperties = [
        AVVideoAverageBitRateKey: 3000000, // 3Mbps
        AVVideoMaxKeyFrameIntervalKey: 30,
        AVVideoProfileLevelKey: AVVideoProfileLevelH264High40
    ]

    let videoSettings: [String: Any] = [
        AVVideoCodecKey: AVVideoCodecType.h264,
        AVVideoWidthKey: 1920,
        AVVideoHeightKey: 1080,
        AVVideoCompressionPropertiesKey: compressionProperties
    ]

    videoOutput.movieFragmentInterval = .invalid
    videoOutput.setOutputSettings(videoSettings, for: videoOutput.connection(with: .video)!)

    // 上記設定を使用して、動画出力をセットアップします
}

この例では、H.264コーデックを使用して、Full HD(1920×1080)の解像度で動画をエンコードします。

さらに、ビットレートやキーフレームの間隔など、詳細なエンコーディング設定もカスタマイズしています。

これにより、動画の品質やファイルサイズを調整することができます。

まとめ

Swiftを使用して動画撮影機能をアプリに実装する際には、基本的な実装手順から高度なカスタマイズ方法まで、さまざまな技術や手法が存在します。

この記事を通じて、動画撮影の基本的な流れや、Swiftでの動画撮影機能の実装手順、さらには実際のサンプルコードの紹介や、応用的な機能の実装方法、さらには注意点や対処法、カスタマイズの方法など、動画撮影機能に関する幅広い内容を学ぶことができたかと思います。

これらの知識とサンプルコードを基に、独自の動画撮影アプリや機能を開発する際の参考にしていただければと思います。

今後、動画撮影技術や関連する機能はさらに進化し続けることでしょう。

そのため、常に最新の情報や技術動向を追いかけ、新しい知識を取り入れることで、より質の高いアプリケーションを開発することが可能になります。

Swiftを用いた動画撮影機能の実装は、アプリ開発者にとって重要なスキルの一つと言えるでしょう。

この記事が皆様の開発活動の一助となることを心より願っています。