class IMPORT_VIDEOS_TASK

(source code)

Description

Import videos task

note
	description: "Import videos task"

	author: "Finnian Reilly"
	copyright: "Copyright (c) 2001-2022 Finnian Reilly"
	contact: "finnian at eiffel hyphen loop dot com"

	license: "MIT license (See: en.wikipedia.org/wiki/MIT_License)"
	date: "2023-01-09 9:55:12 GMT (Monday 9th January 2023)"
	revision: "17"

class
	IMPORT_VIDEOS_TASK

inherit
	RBOX_MANAGEMENT_TASK

	DATABASE_UPDATE_TASK

	EL_MODULE_AUDIO_COMMAND; EL_MODULE_VIDEO_COMMAND

	EL_MODULE_USER_INPUT

create
	make

feature -- Basic operations

	apply
			--
		local
			import_notes: like Video_import_notes; done: BOOLEAN
			song_count: INTEGER
		do
			lio.put_line ("VIDEO IMPORT NOTES")
			import_notes := Video_import_notes #$ [Video_extensions.joined_with_string (", ")]
			across import_notes.lines as line loop
				lio.put_line (line.item)
			end
			song_count := Database.songs.count
			lio.put_new_line
			across Video_extensions as extension loop
				across OS.file_list (Database.music_dir, "*." + extension.item) as video_path loop
					lio.put_path_field ("Found %S", video_path.item.relative_path (Database.music_dir))
					lio.put_new_line
					from done := False until done loop
						Database.extend (new_video_song (video_path.item))
						done := not video_contains_another_song
					end
					OS.delete_file (video_path.item)
				end
			end
			if Database.songs.count > song_count then
				Database.store_all
			end
		end

feature {NONE} -- Factory

	new_input_song_time (prompt: ZSTRING): TIME
		local
			time_str: STRING; time: EL_TIME_ROUTINES
		do
			time_str := User_input.line (prompt).to_string_8
			if not time_str.has ('.') then
				time_str.append (".000")
			end
			if time.is_valid_fine (time_str) then
				create Result.make_from_string (time_str, Fine_time_format)
			else
				create Result.make_by_seconds (0)
			end
		end

	new_song_info_input (duration_time: TIME_DURATION; default_title, lead_artist: ZSTRING): like SONG_INFO
		local
			zero: DOUBLE; song: RBOX_SONG; secs_silence: INTEGER
		do
			create Result
			Result.time_from := new_input_song_time ("From time")
			Result.time_to := new_input_song_time ("To time")
			if Result.time_to.fine_seconds ~ zero then
				Result.time_to := new_time (duration_time.fine_seconds_count)
			end

			song := Database.new_song

			secs_silence := User_input.integer ("Post play silence (secs)")
			if Database.silence_intervals.valid_index (secs_silence) then
				song.set_beats_per_minute (secs_silence)
			end
			song.set_title (User_input.line ("Title"))
			if song.title.is_empty then
				song.set_title (default_title)
			end
			song.set_album (User_input.line ("Album name"))
			if song.album ~ Ditto then
				song.set_album (last_album)
				lio.put_labeled_string ("Using album name", last_album)
				lio.put_new_line
			elseif song.album.is_empty then
				song.set_album (lead_artist)
			end
			last_album := song.album
			song.set_album_artists (User_input.line ("Album artists"))
			song.set_recording_year (User_input.integer ("Recording year"))

			Result.song := song
		end

	new_time (fine_seconds: DOUBLE): TIME
		do
			create Result.make_by_fine_seconds (fine_seconds)
		end

	new_video_song (video_path: FILE_PATH): RBOX_SONG
		local
			video_properties: like Audio_command.new_audio_properties
			video_to_mp3_command: like Video_command.new_video_to_mp3
			genre, artist: ZSTRING; l_info: like SONG_INFO
			duration_time: TIME_DURATION; mp3: EL_MP3_IDENTIFIER
		do
			if attached video_path.steps as steps then
				steps.remove_tail (1); artist := steps.base
				steps.remove_tail (1); genre := steps.base
			end

			video_properties := Audio_command.new_audio_properties (video_path)
			l_info := new_song_info_input (video_properties.duration, video_path.base_name, artist)
			Result := l_info.song
			Result.set_genre (genre)
			Result.set_artist (artist)
			Result.set_mp3_path (Result.unique_normalized_mp3_path)

			video_to_mp3_command := Video_command.new_video_to_mp3 (video_path, Result.mp3_path)

			if l_info.time_from.seconds > 0
				or l_info.time_to.fine_seconds /~ video_properties.duration.fine_seconds_count
			then
				video_to_mp3_command.set_offset_time (l_info.time_from)
				duration_time := l_info.time_to.relative_duration (l_info.time_from)
				-- duration has extra 0.001 secs added to prevent rounding error below the required duration
				duration_time.fine_second_add (0.001)
				video_to_mp3_command.set_duration (duration_time)
				Result.set_duration (duration_time.fine_seconds_count.rounded)
			end
			-- Increase bitrate by 64 for AAC -> MP3 conversion
			video_to_mp3_command.set_bit_rate (video_properties.standard_bit_rate + 64)
			lio.put_string ("Converting..")
			video_to_mp3_command.execute

			create mp3.make (Result.mp3_path)
			Result.set_audio_id_from_uuid (mp3.audio_id)

			Result.save_id3_info
			lio.put_new_line
		end

feature {NONE} -- Implementation

	video_contains_another_song: BOOLEAN
		do
			Result := User_input.approved_action_y_n ("Extract another song from this video?")
		end

feature {NONE} -- Internal attributes

	last_album: ZSTRING

feature {NONE} -- Type definitions

	SONG_INFO: TUPLE [time_from, time_to: TIME; song: RBOX_SONG]
		require
			never_called: False
		once
			create Result
		end

feature {NONE} -- Constants

	Ditto: ZSTRING
		once
			Result := "%""
		end

	Fine_time_format: STRING = "mi:ss.ff3"

	Video_extensions: EL_STRING_8_LIST
		once
			Result := "flac, flv, m4a, m4v, mp4, mov, wav"
		end

	Video_import_notes: ZSTRING
			-- '#' is the same as '%S'
		once
			Result := "[
				Place videos (or m4a audio files) in genre/artist folder.
				Imports files with extensions: #.
				Leave blank fields for the default. Default artist is the parent directory name.
				To duplicate previous album name enter " (for ditto).
				Input offset times are in mm:ss[.xxx] form. If recording year is unknown, enter 0.
			]"
		end

end