質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
Java

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

Kotlin

Kotlinは、ジェットブレインズ社のアンドリー・ブレスラフ、ドミトリー・ジェメロフが開発した、 静的型付けのオブジェクト指向プログラミング言語です。

Q&A

解決済

1回答

2503閲覧

カメラで撮影した動画ファイルをギャラリーに直接保存する方法について

m-mega

総合スコア56

Java

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

Kotlin

Kotlinは、ジェットブレインズ社のアンドリー・ブレスラフ、ドミトリー・ジェメロフが開発した、 静的型付けのオブジェクト指向プログラミング言語です。

0グッド

0クリップ

投稿2021/10/12 08:25

編集2021/10/13 09:53

やりたいこと

表題の通りですが、自作のカメラアプリで録画した動画ファイルを直接ギャラリーに保存できるようにしたいです。
つまり、以下の「やったこと」で記載している「setOutputFile」のパスとして、ギャラリーのパスを直接指定したいです。
ここで言うギャラリーとは、端末既存のカメラアプリで撮影したデータが保存されるデフォルトのフォルダのことを指します。

やったこと

現在、以下のコードでパスを取得し、アプリ独自の外部ストレージを指定して動画を保存することはできています。

kotlin

1// ビデオの出力ファイルパスを設定 2 private fun getVideoFilePath(context: Context?): String { 3 Log.i("録画", "ファイルパス設定") 4 val filename = "${System.currentTimeMillis()}.mp4" // 録画ファイル名を保存する際の時間にする 5 val dir = context?.getExternalFilesDir(null) // アプリ独自のパスを取得 6 return if (dir == null) { 7 filename 8 } else { 9 "${dir.absolutePath}/$filename" // 取得したパスにファイル名を繋げて指定 10 } 11 }

そして、取得したパスをMediaRecorderのsetOutputFileに渡して、ファイルの出力先として設定しています。

kotlin

1setOutputFile(nextVideoAbsolutePath) // 出力ファイルパスを設定

また、サイトを参考に、現在以下のように実装をしているのですが、エラーが発生してしまっております。

kotlin

1 // ギャラリーにデータを保存する 2 private fun saveFile(f: File) { 3 Log.i("録画", "ギャラリー保存準備") 4 val stream = FileOutputStream(f) 5 stream.close() 6 Log.i("録画", "ギャラリー保存設定") 7 val contentValues = ContentValues().apply { 8 put(MediaStore.Video.Media.MIME_TYPE, "video/mp4") 9 put("_data", f.absolutePath) 10 } 11 Log.i("録画", "ギャラリー保存開始") 12 ContentResolver.insert( 13 MediaStore.Video.Media.EXTERNAL_CONTENT_URI, contentValues) 14 }
エラー内容
Importを追記することを促されますが、提案通りにImportを選択しても何も改善されず、エラー(文字が赤くなっている)状態。

わからないこと

動画の保存先としてギャラリーを指定する方法と、ギャラリーのパスを取得する方法がわかりません。

調査すると、画像ファイルをギャラリーにコピーする方法などは見受けられましたが、動画ファイルをコピーしている例は見当たらず苦戦しています。

そもそも、ギャラリーにデータをコピーするといった周りくどい方法ではなく、ギャラリーのパスを指定する構文か何かで簡単にパスを指定できないものなのでしょうか。

以下の参考サイトをもとに実装を試みているのですが、うまくいっていない状態です。
また、エラーの内容ですが、lateinitに何を入れるべきなのかが分からず、このようなエラーが出てしまっています。
ですが、以下の記載を無くしてしまうと、存在しないプロパティとしてエラーが発生してしまいます。

kotlin

1lateinit var contentResolver: ContentResolver

引き続き調査と検証を続けますが、解決法をご存知の方がいらっしゃいましたらご教授いただけますと幸いです。

参考にしたサイト

note - ScopedStorageを使った例
Qiita - 画像をギャラリーに表示させる例
Storage Access Frameworkの活用例
Android developers - 共有ストレージからメディア ファイルにアクセスする
teratail - 画像をギャラリーに反映する方法

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

自己解決

【自己解決】
contextのcontentResolverを設定し忘れていたことが原因でした。
以下のように値を与えた後にギャラリーへコピーする処理を行うことで無事に解決しました。

kotlin

1<MainActivity.kt> 2... 3// 指定したファイルをギャラリーに追加する 4 private fun saveToGallery(path: String?) { 5 val contentResolver = requireContext().contentResolver // ←ここが抜けていました 6 val contentValues = ContentValues().apply { 7 put(MediaStore.Video.Media.MIME_TYPE, "video/mp4") 8 put("_data", File(path).absolutePath) 9 } 10 contentResolver.insert( 11 MediaStore.Video.Media.EXTERNAL_CONTENT_URI, contentValues) 12 }

また、調査したところ、やはりギャラリーに直接保存する方法はありませんでした。
そのため、録画の場合は質問本文のコードのように、一旦仮のフォルダにデータを保存する必要があるようです。

(注意)
この時、getExternalFilesDir()を使って取得したパスだとギャラリーへのコピーがうまくいきません。
そのため、私は以下のようにDCIMのパス("/storage/emulated/0/DCIM")を一旦の保存場所として直接指定しております。

kotlin

1<MainActivity.kt> 2... 3// ビデオの出力ファイルパスを設定 4 private fun getVideoFilePath(context: Context?): String { 5 Log.i("録画", "ファイルパス設定") 6 val filename = "${System.currentTimeMillis()}.mp4" // 録画ファイル名を保存する際の時間にする 7 val dir = "/storage/emulated/0/DCIM" // DCIMのパスを直接指定 8 return if (dir == null) { 9 filename 10 } else { 11 "$dir/$filename" // 取得したパスにファイル名を繋げて指定 12 } 13 }

また、このままでは一旦の保存場所とギャラリーの2箇所に同じ録画データが溜まっていってしまうので、以下のように一旦の保存場所のデータを削除するコードも追記しました。

kotlin

1<MainActivity.kt> 2... 3// 指定したファイルを削除する 4 @RequiresApi(Build.VERSION_CODES.O) 5 private fun deletedata() { 6 val deletepath = Paths.get("nextVideoAbsolutePath") 7 try { 8 // コピー元の録画データを削除 9 Files.deleteIfExists(deletepath) 10 } catch (e: IOException) { 11 e.printStackTrace() 12 } 13 }

以上となります。
この質問をご覧いただき、解決方法を調査いただいた方々には心より感謝いたします。
また、同じような問題に直面した方に、今回の回答が少しでも役立つことができれば幸いです。

投稿2021/10/14 04:00

m-mega

総合スコア56

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問