Files
foobar2000-sdk/foobar2000/SDK/metadb_handle.h
2021-12-14 00:28:25 -07:00

263 lines
17 KiB
C++

#pragma once
class titleformat_hook;
class titleformat_text_filter;
class titleformat_object;
class metadb_info_container : public service_base {
FB2K_MAKE_SERVICE_INTERFACE(metadb_info_container, service_base);
public:
virtual file_info const & info() = 0;
virtual t_filestats const & stats() = 0;
virtual bool isInfoPartial() = 0;
};
//! A metadb_handle object represents interface to reference-counted file_info cache entry for the specified location.\n
//! To obtain a metadb_handle to specific location, use metadb::handle_create(). To obtain a list of metadb_handle objects corresponding to specific path (directory, playlist, multitrack file, etc), use relevant playlist_incoming_item_filter methods (recommended), or call playlist_loader methods directly.\n
//! A metadb_handle is also the most efficient way of passing playable object locations around because it provides fast access to both location and infos, and is reference counted so duplicating it is as fast as possible.\n
//! To retrieve a path of a file from a metadb_handle, use metadb_handle::get_path() function. Note that metadb_handle is NOT just file path, some formats support multiple subsongs per physical file, which are signaled using subsong indexes.
class NOVTABLE metadb_handle : public service_base
{
public:
//! Retrieves location represented by this metadb_handle object. Returned reference is valid until calling context releases metadb_handle that returned it (metadb_handle_ptr is deallocated etc).
virtual const playable_location & get_location() const = 0;//never fails, returned pointer valid till the object is released
//! Renders information about item referenced by this metadb_handle object.
//! @param p_hook Optional callback object overriding fields and functions; set to NULL if not used.
//! @param p_out String receiving the output on success.
//! @param p_script Titleformat script to use. Use titleformat_compiler service to create one.
//! @param p_filter Optional callback object allowing input to be filtered according to context (i.e. removal of linebreak characters present in tags when rendering playlist lines). Set to NULL when not used.
//! @returns true on success, false when dummy file_info instance was used because actual info is was not (yet) known.
virtual bool format_title(titleformat_hook * p_hook,pfc::string_base & p_out,const service_ptr_t<titleformat_object> & p_script,titleformat_text_filter * p_filter) = 0;
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual void metadb_lock() = 0;
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual void metadb_unlock() = 0;
//! Returns last seen file stats, filestats_invalid if unknown.
virtual t_filestats get_filestats() const = 0;
//! Obsolete, use get_info_ref() family of methods instead. \n
//! Queries whether cached info about item referenced by this metadb_handle object is already available. Note that this function causes the metadb to be temporarily locked; you can not use it in context that where locking is forbidden.\n
//! Note that state of cached info changes only inside main thread, so you can safely assume that it doesn't change while some block of your code inside main thread is being executed.
virtual bool is_info_loaded() const = 0;
//! Obsolete, use get_info_ref() instead. \n
//! Queries cached info about item referenced by this metadb_handle object. Returns true on success, false when info is not yet known. Note that this function causes the metadb to be temporarily locked; you can not use it in context that where locking is forbidden. \n
//! Note that state of cached info changes only inside main thread, so you can safely assume that it doesn't change while some block of your code inside main thread is being executed.
virtual bool get_info(file_info & p_info) const = 0;
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual bool get_info_locked(const file_info * & p_info) const = 0;
//! Obsolete, use get_info_ref() family of methods instead. \n
//! Queries whether cached info about item referenced by this metadb_handle object is already available.\n
//! This is intended for use in special cases when you need to immediately retrieve info sent by metadb_io hint from another thread; state of returned data can be altered by any thread, as opposed to non-async methods.
virtual bool is_info_loaded_async() const = 0;
//! Obsolete, use get_info_ref() family of methods instead. \n
//! Queries cached info about item referenced by this metadb_handle object. Returns true on success, false when info is not yet known. Note that this function causes the metadb to be temporarily locked; you can not use it in context that where locking is forbidden.\n
//! This is intended for use in special cases when you need to immediately retrieve info sent by metadb_io hint from another thread; state of returned data can be altered by any thread, as opposed to non-async methods.
virtual bool get_info_async(file_info & p_info) const = 0;
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual bool get_info_async_locked(const file_info * & p_info) const = 0;
//! Renders information about item referenced by this metadb_handle object, using external file_info data.
virtual void format_title_from_external_info(const file_info & p_info,titleformat_hook * p_hook,pfc::string_base & p_out,const service_ptr_t<titleformat_object> & p_script,titleformat_text_filter * p_filter) = 0;
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual bool format_title_nonlocking(titleformat_hook * p_hook,pfc::string_base & p_out,const service_ptr_t<titleformat_object> & p_script,titleformat_text_filter * p_filter) = 0;
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual void format_title_from_external_info_nonlocking(const file_info & p_info,titleformat_hook * p_hook,pfc::string_base & p_out,const service_ptr_t<titleformat_object> & p_script,titleformat_text_filter * p_filter) = 0;
#if FOOBAR2000_TARGET_VERSION >= 76
//! \since 1.0
//! Returns browse info for this track. \n
//! Browse info comes from an external source - such as internet playlist metadata - not from the media file itself, and is maintained separately. \n
//! When title formatting calls are invoked on for a track having browse info present, data for title formatting is sourced from both primary and browse info. \n
//! Example: internet radio stream provides no metadata but its playlist XML has title (radio station name), %title% resolves to the radio station name from the playlist.
virtual bool get_browse_info(file_info & info, t_filetimestamp & ts) const = 0;
//! \since 1.0
//! OBSOLETE, DO NOT CALL
__declspec(deprecated) virtual bool get_browse_info_locked(const file_info * & p_info, t_filetimestamp & ts) const = 0;
#endif
#if FOOBAR2000_TARGET_VERSION >= 78
//! \since 1.3
//! Retrieve a reference to the primary info. \n
//! You can hold the reference to the info as long as you like, call the method in any context you like with no lock semantics involved. The info held by the returned reference will remain constant even if the metadb content changes. \n
//! Returns true and sets outInfo to a reference to this item's primary info on success, returns false on failure (no info known at this time).
virtual bool get_info_ref(metadb_info_container::ptr & outInfo) const = 0;
//! \since 1.3
//! Retrieve a reference to the async info (pending info update). If async info isn't set, a reference to the primary info is returned.\n
//! You can hold the reference to the info as long as you like, call the method in any context you like with no lock semantics involved. The info held by the returned reference will remain constant even if the metadb content changes. \n
//! Returns true and sets outInfo to a reference to this item's async or primary info on success, returns false on failure (no info known at this time).
virtual bool get_async_info_ref(metadb_info_container::ptr & outInfo) const = 0;
//! \since 1.3
//! Retrieve references to the item's primary and browse infos. If no info is set, NULL pointers are returned. For most local files, browse info is not available and you get a NULL for it.\n
//! Since browse info is usually used along with the primary info (as a fallback for missing metas), you can get the two with one call for better performance. \n
//! You can hold the reference to the info as long as you like, call the method in any context you like with no lock semantics involved. The info held by the returned reference will remain constant even if the metadb content changes. \n
//! See also: get_browse_info(), for browse info rationale.
virtual void get_browse_info_ref(metadb_info_container::ptr & outInfo, metadb_info_container::ptr & outBrowse) const = 0;
//! Simplified method, always returns non-null, dummy info if nothing to return.
virtual metadb_info_container::ptr get_info_ref() const = 0;
//! Simplified method, always returns non-null, dummy info if nothing to return.
virtual metadb_info_container::ptr get_async_info_ref() const = 0;
//! \since 1.3
//! Retrieve full info using available means - read actual file if not cached. \n
//! Throws exceptions on failure.
metadb_info_container::ptr get_full_info_ref( abort_callback & aborter ) const;
#endif
//! \since 1.3
//! Helper using get_browse_info_ref(). \n
//! Retrieves primary info + browse info merged together. \n
//! Returns true on success, false if neither info is available. \n
//! If neither info is avaialble, output data structure is emptied. \n
//! See also: get_browse_info() for browse info rationale.
bool get_browse_info_merged(file_info & infoMerged) const;
static bool g_should_reload(const t_filestats & p_old_stats,const t_filestats & p_new_stats,bool p_fresh);
bool should_reload(const t_filestats & p_new_stats,bool p_fresh) const;
//! Helper provided for backwards compatibility; takes formatting script as text string and calls relevant titleformat_compiler methods; returns false when the script could not be compiled.\n
//! See format_title() for descriptions of parameters.\n
//! Bottleneck warning: you should consider using precompiled titleformat script object and calling regular format_title() instead when processing large numbers of items.
bool format_title_legacy(titleformat_hook * p_hook,pfc::string_base & out,const char * p_spec,titleformat_text_filter * p_filter);
//! Retrieves path of item described by this metadb_handle instance. Returned string is valid until calling context releases metadb_handle that returned it (metadb_handle_ptr is deallocated etc).
inline const char * get_path() const {return get_location().get_path();}
//! Retrieves subsong index of item described by this metadb_handle instance (used for multiple playable tracks within single physical file).
inline t_uint32 get_subsong_index() const {return get_location().get_subsong_index();}
double get_length();//helper
t_filetimestamp get_filetimestamp();
t_filesize get_filesize();
//! Internal method, do not use
inline const char * _get_path() const { return get_path(); }
FB2K_MAKE_SERVICE_INTERFACE(metadb_handle,service_base);
};
typedef service_ptr_t<metadb_handle> metadb_handle_ptr;
typedef pfc::list_base_t<metadb_handle_ptr> & metadb_handle_list_ref;
typedef pfc::list_base_const_t<metadb_handle_ptr> const & metadb_handle_list_cref;
namespace metadb_handle_list_helper {
void sort_by_format(metadb_handle_list_ref p_list,const char * spec,titleformat_hook * p_hook);
void sort_by_format_get_order(metadb_handle_list_cref p_list,t_size* order,const char * spec,titleformat_hook * p_hook);
void sort_by_format(metadb_handle_list_ref p_list,const service_ptr_t<titleformat_object> & p_script,titleformat_hook * p_hook, int direction = 1);
void sort_by_format_get_order(metadb_handle_list_cref p_list,t_size* order,const service_ptr_t<titleformat_object> & p_script,titleformat_hook * p_hook,int p_direction = 1);
void sort_by_relative_path(metadb_handle_list_ref p_list);
void sort_by_relative_path_get_order(metadb_handle_list_cref p_list,t_size* order);
void remove_duplicates(pfc::list_base_t<metadb_handle_ptr> & p_list);
void sort_by_pointer_remove_duplicates(pfc::list_base_t<metadb_handle_ptr> & p_list);
void sort_by_path_quick(pfc::list_base_t<metadb_handle_ptr> & p_list);
void sort_by_pointer(pfc::list_base_t<metadb_handle_ptr> & p_list);
t_size bsearch_by_pointer(const pfc::list_base_const_t<metadb_handle_ptr> & p_list,const metadb_handle_ptr & val);
double calc_total_duration(const pfc::list_base_const_t<metadb_handle_ptr> & p_list);
void sort_by_path(pfc::list_base_t<metadb_handle_ptr> & p_list);
t_filesize calc_total_size(metadb_handle_list_cref list, bool skipUnknown = false);
t_filesize calc_total_size_ex(metadb_handle_list_cref list, bool & foundUnknown);
bool extract_single_path(metadb_handle_list_cref list, const char * &path);
bool extract_folder_path(metadb_handle_list_cref list, pfc::string_base & path);
void sort_by_format_get_order_v2( metadb_handle_list_cref p_list, size_t * order, const service_ptr_t<titleformat_object> & script, titleformat_hook * hook, int direction, abort_callback & aborter );
void sort_by_format_v2(metadb_handle_list_ref p_list, const service_ptr_t<titleformat_object> & script, titleformat_hook * hook, int direction, abort_callback & aborter);
};
template<template<typename> class t_alloc = pfc::alloc_fast >
class metadb_handle_list_t : public service_list_t<metadb_handle,t_alloc> {
private:
typedef metadb_handle_list_t<t_alloc> t_self;
typedef pfc::list_base_const_t<metadb_handle_ptr> t_interface;
public:
inline void sort_by_format(const char * spec,titleformat_hook * p_hook) {
return metadb_handle_list_helper::sort_by_format(*this, spec, p_hook);
}
inline void sort_by_format_get_order(t_size* order,const char * spec,titleformat_hook * p_hook) const {
metadb_handle_list_helper::sort_by_format_get_order(*this, order, spec, p_hook);
}
inline void sort_by_format(const service_ptr_t<titleformat_object> & p_script,titleformat_hook * p_hook, int direction = 1) {
metadb_handle_list_helper::sort_by_format(*this, p_script, p_hook, direction);
}
inline void sort_by_format_get_order(t_size* order,const service_ptr_t<titleformat_object> & p_script,titleformat_hook * p_hook) const {
metadb_handle_list_helper::sort_by_format_get_order(*this, order, p_script, p_hook);
}
inline void sort_by_relative_path() {
metadb_handle_list_helper::sort_by_relative_path(*this);
}
inline void sort_by_relative_path_get_order(t_size* order) const {
metadb_handle_list_helper::sort_by_relative_path_get_order(*this,order);
}
inline void remove_duplicates() {metadb_handle_list_helper::remove_duplicates(*this);}
inline void sort_by_pointer_remove_duplicates() {metadb_handle_list_helper::sort_by_pointer_remove_duplicates(*this);}
inline void sort_by_path_quick() {metadb_handle_list_helper::sort_by_path_quick(*this);}
inline void sort_by_pointer() {metadb_handle_list_helper::sort_by_pointer(*this);}
inline t_size bsearch_by_pointer(const metadb_handle_ptr & val) const {return metadb_handle_list_helper::bsearch_by_pointer(*this,val);}
inline double calc_total_duration() const {return metadb_handle_list_helper::calc_total_duration(*this);}
inline void sort_by_path() {metadb_handle_list_helper::sort_by_path(*this);}
const t_self & operator=(const t_self & p_source) { this->remove_all(); this->add_items(p_source);return *this;}
const t_self & operator=(const t_interface & p_source) {this->remove_all(); this->add_items(p_source);return *this;}
const t_self & operator=(t_self && p_source) {this->move_from(p_source); return *this; }
metadb_handle_list_t(const t_self & p_source) { this->add_items(p_source);}
metadb_handle_list_t(const t_interface & p_source) { this->add_items(p_source);}
metadb_handle_list_t() {}
metadb_handle_list_t(t_self && p_source) { this->move_from(p_source);}
t_self & operator+=(const t_interface & source) { this->add_items(source); return *this;}
t_self & operator+=(const metadb_handle_ptr & source) { this->add_item(source); return *this;}
bool extract_single_path(const char * &path) const {return metadb_handle_list_helper::extract_single_path(*this, path);}
};
typedef metadb_handle_list_t<> metadb_handle_list;
namespace metadb_handle_list_helper {
void sorted_by_pointer_extract_difference(metadb_handle_list const & p_list_1,metadb_handle_list const & p_list_2,metadb_handle_list & p_list_1_specific,metadb_handle_list & p_list_2_specific);
};
inline pfc::string_base & operator<<(pfc::string_base & p_fmt,const metadb_handle_ptr & p_location) {
if (p_location.is_valid())
return p_fmt << p_location->get_location();
else
return p_fmt << "[invalid location]";
}
namespace fb2k {
pfc::string_formatter formatTrackList( metadb_handle_list_cref );
pfc::string_formatter formatTrackTitle(metadb_handle_ptr item, const char * script = "%title%" );
pfc::string_formatter formatTrackTitle(metadb_handle_ptr item,service_ptr_t<class titleformat_object> script);
}