After updating to API v2, the old code stops compiling, and it takes some contemplation to fix the errors. It would be good to have some guidelines for common problems.
The caps.set_value() is neither intended nor a bug, but it is known. It is one of few (if not the only) adjustment that had to be made in the examples when moving to v2 API. There is no real alternative, but the examples use e.g. namespace GObject_ = gi::repository::GObject (and then GObject_::Value), which makes it not all that verbose.
In file included from /opt/rh/gcc-toolset-13/root/usr/include/c++/13/bits/stl_pair.h:60, from /opt/rh/gcc-toolset-13/root/usr/include/c++/13/utility:69, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qcompilerdetection.h:864, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qgenericatomic.h:8, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qatomic_cxx11.h:8, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qbasicatomic.h:8, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qatomic.h:8, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qrefcount.h:7, from /usr/local/desktop-app/Qt-6.6.1/include/QtCore/qbytearray.h:8:/opt/rh/gcc-toolset-13/root/usr/include/c++/13/type_traits: In instantiation of ‘struct std::is_convertible<_GHashTable, std::pair<gi::detail::cstring_d<>, gi::detail::cstring_d<> > >’:/usr/src/tdesktop/cmake/external/glib/cppgir/gi/container.hpp:1548:63: required by substitution of ‘template<class InputT, class Check> gi::detail::Collection<_GHashTable, std::pair<char*, char*>, gi::transfer_full_t>::Collection(InputT*, size_t) [with InputT = _GHashTable; Check = <missing>]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/container.hpp:1723:15: required from ‘static CppType gi::detail::Collection<ListType, T, Transfer, Notify, ExtraBase>::_wrap(const ListType*, const ReqTransfer&) [with CppType = gi::detail::Collection<_GHashTable, std::pair<char*, char*>, gi::transfer_full_t>; ReqTransfer = gi::transfer_full_t; typename std::enable_if<(std::is_same<Transfer, ReqTransfer>::value && (! _detail::is_span))>::type* <anonymous> = 0; ListType = _GHashTable; T = std::pair<char*, char*>; Transfer = gi::transfer_full_t; Notify = gi::detail::Notifier<void>; ExtraBase = gi::detail::Container]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/wrap.hpp:134:48: required from ‘TargetType gi::wrap_to(CType, const Transfer&) [with TargetType = detail::Collection<_GHashTable, std::pair<char*, char*>, transfer_full_t>; CType = _GHashTable*; Transfer = transfer_full_t; typename TargetType::_detail::DataType* <anonymous> = 0]’/usr/src/tdesktop/out/gen/glib/uri_impl.hpp:446:95: required from here/opt/rh/gcc-toolset-13/root/usr/include/c++/13/type_traits:1417:30: error: invalid use of incomplete type ‘struct _GHashTable’ [-fpermissive] 1417 | : public __bool_constant<__is_convertible(_From, _To)> | ^~~~~~~~~~~~~~~~~~~~~~~~~~~~In file included from /usr/local/include/glib-2.0/glib.h:52, from /usr/local/include/glib-2.0/gobject/gbinding.h:30, from /usr/local/include/glib-2.0/glib-object.h:24, from /usr/src/tdesktop/cmake/external/glib/cppgir/gi/boxed.hpp:6, from /usr/src/tdesktop/cmake/external/glib/cppgir/gi/base.hpp:29, from /usr/src/tdesktop/cmake/external/glib/cppgir/gi/gi.hpp:30:/usr/local/include/glib-2.0/glib/ghash.h:40:16: note: forward declaration of ‘struct _GHashTable’ 40 | typedef struct _GHashTable GHashTable; | ^~~~~~~~~~~/opt/rh/gcc-toolset-13/root/usr/include/c++/13/type_traits: In instantiation of ‘struct std::is_convertible<_GHashTable, std::pair<gi::detail::cstring_d<>, gi::repository::GLib::Variant> >’:/usr/src/tdesktop/cmake/external/glib/cppgir/gi/container.hpp:1548:63: required by substitution of ‘template<class InputT, class Check> gi::detail::Collection<_GHashTable, std::pair<char*, _GVariant*>, gi::transfer_full_t>::Collection(InputT*, size_t) [with InputT = _GHashTable; Check = <missing>]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/container.hpp:1723:15: required from ‘static CppType gi::detail::Collection<ListType, T, Transfer, Notify, ExtraBase>::_wrap(const ListType*, const ReqTransfer&) [with CppType = gi::detail::Collection<_GHashTable, std::pair<char*, _GVariant*>, gi::transfer_full_t>; ReqTransfer = gi::transfer_full_t; typename std::enable_if<(std::is_same<Transfer, ReqTransfer>::value && (! _detail::is_span))>::type* <anonymous> = 0; ListType = _GHashTable; T = std::pair<char*, _GVariant*>; Transfer = gi::transfer_full_t; Notify = gi::detail::Notifier<void>; ExtraBase = gi::detail::Container]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/wrap.hpp:134:48: required from ‘TargetType gi::wrap_to(CType, const Transfer&) [with TargetType = detail::Collection<_GHashTable, std::pair<char*, _GVariant*>, transfer_full_t>; CType = _GHashTable*; Transfer = transfer_full_t; typename TargetType::_detail::DataType* <anonymous> = 0]’/usr/src/tdesktop/out/gen/gio/menumodel_impl.hpp:183:107: required from here/opt/rh/gcc-toolset-13/root/usr/include/c++/13/type_traits:1417:30: error: invalid use of incomplete type ‘struct _GHashTable’ [-fpermissive] 1417 | : public __bool_constant<__is_convertible(_From, _To)> | ^~~~~~~~~~~~~~~~~~~~~~~~~~~~/usr/local/include/glib-2.0/glib/ghash.h:40:16: note: forward declaration of ‘struct _GHashTable’ 40 | typedef struct _GHashTable GHashTable; | ^~~~~~~~~~~/opt/rh/gcc-toolset-13/root/usr/include/c++/13/type_traits: In instantiation of ‘struct std::is_convertible<_GHashTable, std::pair<gi::detail::cstring_d<>, gi::repository::Gio::MenuModel> >’:/usr/src/tdesktop/cmake/external/glib/cppgir/gi/container.hpp:1548:63: required by substitution of ‘template<class InputT, class Check> gi::detail::Collection<_GHashTable, std::pair<char*, _GMenuModel*>, gi::transfer_full_t>::Collection(InputT*, size_t) [with InputT = _GHashTable; Check = <missing>]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/container.hpp:1723:15: required from ‘static CppType gi::detail::Collection<ListType, T, Transfer, Notify, ExtraBase>::_wrap(const ListType*, const ReqTransfer&) [with CppType = gi::detail::Collection<_GHashTable, std::pair<char*, _GMenuModel*>, gi::transfer_full_t>; ReqTransfer = gi::transfer_full_t; typename std::enable_if<(std::is_same<Transfer, ReqTransfer>::value && (! _detail::is_span))>::type* <anonymous> = 0; ListType = _GHashTable; T = std::pair<char*, _GMenuModel*>; Transfer = gi::transfer_full_t; Notify = gi::detail::Notifier<void>; ExtraBase = gi::detail::Container]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/wrap.hpp:134:48: required from ‘TargetType gi::wrap_to(CType, const Transfer&) [with TargetType = detail::Collection<_GHashTable, std::pair<char*, _GMenuModel*>, transfer_full_t>; CType = _GHashTable*; Transfer = transfer_full_t; typename TargetType::_detail::DataType* <anonymous> = 0]’/usr/src/tdesktop/out/gen/gio/menumodel_impl.hpp:209:104: required from here/opt/rh/gcc-toolset-13/root/usr/include/c++/13/type_traits:1417:30: error: invalid use of incomplete type ‘struct _GHashTable’ [-fpermissive] 1417 | : public __bool_constant<__is_convertible(_From, _To)> | ^~~~~~~~~~~~~~~~~~~~~~~~~~~~/usr/local/include/glib-2.0/glib/ghash.h:40:16: note: forward declaration of ‘struct _GHashTable’ 40 | typedef struct _GHashTable GHashTable; | ^~~~~~~~~~~In file included from /usr/src/tdesktop/cmake/external/glib/cppgir/gi/gi.hpp:36:/usr/src/tdesktop/out/gen/gio/dbusinterfaceskeleton.hpp: In instantiation of ‘static constexpr bool gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::TypeInitData::has_definition(const get_info_t*, const SubClass*) [with SubClass = base::Platform::SystemMediaControls::Private::Player; get_info_t = gi::detail::member_type<get_info_tag>]’:/usr/src/tdesktop/cmake/external/glib/cppgir/gi/objectclass.hpp:239:26: required from ‘static constexpr gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::TypeInitData gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::TypeInitData::factory() [with SubClass = base::Platform::SystemMediaControls::Private::Player]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/objectclass.hpp:209:10: required from ‘void (* gi::detail::make_type_init_data())() [with BaseDef = gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef; SubClass = base::Platform::SystemMediaControls::Private::Player; typename std::enable_if<(! std::is_same<SubClass, void>::value)>::type* <anonymous> = 0; type_init_data_factory_t = void (*)()]’/usr/src/tdesktop/cmake/external/glib/cppgir/gi/objectclass.hpp:577:76: required from ‘gi::detail::ClassTemplate<ClassDef, BaseClass, Interfaces>::ClassTemplate(const SubClass*, GType, gi::detail::cstring_v, const gi::detail::ObjectClass::ClassInitNode&, const gi::repository::GObject::construct_params&, const gi::detail::properties&) [with SubClass = base::Platform::SystemMediaControls::Private::Player; ClassDef = gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef; BaseClass = gi::detail::ObjectClass; Interfaces = {gi::repository::Gio::impl::internal::DBusInterfaceIfaceClassImpl}; GType = long unsigned int; gi::detail::cstring_v = gi::detail::cstring_v_d<>; gi::repository::GObject::construct_params = std::vector<gi::detail::Parameter>; gi::detail::properties = std::map<std::__cxx11::basic_string<char>, std::pair<gi::detail::PropertyBase*, gi::repository::GObject::ParamSpec> >]’/usr/src/tdesktop/out/gen/gio/dbusinterfaceskeleton.hpp:215:14: required from ‘gi::detail::ClassTemplate<ClassDef, BaseClass, Interfaces>::ClassTemplate(const SubClass*, const gi::repository::GObject::construct_params&, const gi::detail::properties&) [with SubClass = base::Platform::SystemMediaControls::Private::Player; ClassDef = gi::repository::Mpris::impl::internal::MediaPlayer2PlayerSkeletonClassDef; BaseClass = gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClass; Interfaces = {gi::repository::Mpris::impl::internal::MediaPlayer2PlayerIfaceClassImpl}; gi::repository::GObject::construct_params = std::vector<gi::detail::Parameter>; gi::detail::properties = std::map<std::__cxx11::basic_string<char>, std::pair<gi::detail::PropertyBase*, gi::repository::GObject::ParamSpec> >]’/usr/src/tdesktop/out/gen/mpris/mediaplayer2playerskeleton.hpp:125:14: required from ‘gi::detail::ObjectImpl<ObjectT, ClassT>::ObjectImpl(const SubClass*, const gi::repository::GObject::construct_params&, const gi::detail::properties&) [with SubClass = base::Platform::SystemMediaControls::Private::Player; ObjectT = gi::repository::Mpris::MediaPlayer2PlayerSkeleton; ClassT = gi::repository::Mpris::impl::internal::MediaPlayer2PlayerSkeletonClass; gi::repository::GObject::construct_params = std::vector<gi::detail::Parameter>; gi::detail::properties = std::map<std::__cxx11::basic_string<char>, std::pair<gi::detail::PropertyBase*, gi::repository::GObject::ParamSpec> >]’/usr/src/tdesktop/Telegram/lib_base/base/platform/linux/base_system_media_controls_linux.cpp:133:19: required from here/usr/src/tdesktop/out/gen/gio/dbusinterfaceskeleton.hpp:266:5: in ‘constexpr’ expansion of ‘gi::detail::Combine<gi::detail::ObjectClass::DefinitionData, gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::TypeInitData>::defines<gi::detail::member_type<gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::TypeInitData::get_info_tag>, base::Platform::SystemMediaControls::Private::Player>(((const gi::detail::member_type<gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::TypeInitData::get_info_tag>*)0), ((const base::Platform::SystemMediaControls::Private::Player*)0))’/usr/src/tdesktop/out/gen/gio/dbusinterfaceskeleton.hpp:255:3: error: lookup of ‘_check_member_conflict_get_info’ in ‘base::Platform::SystemMediaControls::Private::Player’ is ambiguous 255 | GI_MEMBER_DEFINE(DBusInterfaceSkeletonClass, get_info) | ^~~~~~~~~~~~~~~~/usr/src/tdesktop/out/gen/gio/dbusinterface.hpp:116:7: note: candidates are: ‘using gi::repository::Gio::impl::internal::DBusInterfaceIfaceDef::_check_member_conflict_get_info = gi::repository::Gio::impl::internal::DBusInterfaceIfaceDef::self’ 116 | using GI_MEMBER_CHECK_CONFLICT(get_info) = self; | ^~~~~~~~~~~~~~~~~~~~~~~~/usr/src/tdesktop/out/gen/gio/dbusinterfaceskeleton.hpp:172:7: note: ‘using gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::_check_member_conflict_get_info = gi::repository::Gio::impl::internal::DBusInterfaceSkeletonClassDef::self’ 172 | using GI_MEMBER_CHECK_CONFLICT(get_info) = self; | ^~~~~~~~~~~~~~~~~~~~~~~~
And it seems something happened to nullable strings?
/usr/src/tdesktop/Telegram/lib_base/base/platform/linux/base_system_media_controls_linux.cpp: In lambda function:/usr/src/tdesktop/Telegram/lib_base/base/platform/linux/base_system_media_controls_linux.cpp:241:80: error: ‘using gi::detail::cstring_v = class gi::detail::cstring_v_d<>’ {aka ‘class gi::detail::cstring_v_d<>’} has no member named ‘value_or’; did you mean ‘value_type’? 241 | LoopStatusToCommand(player().get_loop_status().value_or(""))); | ^~~~~~~~ | value_type
How is this supposed to be instantiated implicitly? Correct me if I am wrong, but it will hypothetically require the compiler to try all possible types available in the translation unit as a Destination.
When Qt is not involved, the only operator+ candidate would be inline cstring operator+(const _string_view x, const _string_view y) noexcept so both the C string (char*) literal and the gi::cstring variable will be converted to a _string_view (using _string_view = cstr<transfer_none_t>;) for concatenation and the result is a gi::cstring. Given that std::out_of_range accepts a std::string, the templated operator Destination will get instantiated into a operator std::string so it will get implicitly converted to std::string in the end.
Do you have an MWE demonstrating such a templated conversion can be implicitly instantiated? I have an MWE showing it cannot: https://godbolt.org/z/jPGqj1Pvn. Feel free to modify it to illustrate your point.
Not minimal, no. But this cppgir code works if I include it in a file not using Qt headers. Your example lacks the operator+ implementation cppgir has which is important.
If it works, it doesn't mean your assumption of an implicit instantiation holds any water. Prove your assumption with facts expressed in an MWE. So far, it doesn't make sense to me.
I'm here to report the bug with cppgir+Qt headers combo, not to argue. You have the cppgir code, you can explore it to find how it works despite it doesn't make sense to you.
This is indeed due to a surprising incompatibility with Qt headers, as might happen with a conversion function template. So the allowed cases will be restricted (quite) some more to avoid this ambiguity.
The std::optional<std::string> was introduced as std::string could not express/convey the NULL pointer case. Now gi::cstring is used as a more tight/simple wrapper of a char* and it also accepts the NULL case. As such, there is no more need for std::optional<std::string> (though it still supports some conversion to that case).
No idea yet on the uri_impl.hpp case (?), suffice it to say there is no such error in my builds. Differences in compiler and/or in GLib version (TBD)? The other could be a conflict check on "duplicate members". That is, some situation where inherited interfaces/classes have a conflicting member definition. The SUBCLASS IMPLEMENTATION API in documentation elaborates a bit on that.
GI_MEMBER_CHECK_CONFLICT checks if some method member is defined in multiple superclasses and arranges for hard failure if that is the case. In this case it looks like there is a get_info member in DBusInterfaceSkeletonClass(Def) and DBusInterfaceIface(Def). This could/should be resolved by manual specification of DefinitionData, e.g. as in (dummy example) https://gitlab.com/mnauw/cppgir/-/blob/master/examples/gst.cpp?ref_type=heads#L329.
I am guessing in this case it might be in base::Platform::SystemMediaControls::Private::Player? But there seems be user code and then also glib-generated skeleton code in play here (which I do not have at hand here).
I eventually rolled back to e856d763. I had a queue of VideoFrame, which worked quite well. At the moment, I don't have time to explore why it doesn't work and what changes I have to make. And there is no guarantee it's the last one. Maybe sometime in the future, I will switch to v2 of cppgir.
Thanks for the update, even though a somewhat unfortunate one.
AFAIK, there has only been a minor change in that CBoxed type area; the allocate method has been rename to allocate_.
Besides/beyond that, a CBoxed type is and always has been a tricky kind. In this case, e.g. [gst_video_frame_]map is even more tricky, as the instance parameter is actually (sort-of) an output parameter (though not so annotated). In any case, it would have to be explicitly allocated (using that allocate_ method) before calling map. And it would also have to be explicitly unmaped again prior to destruction.
I moved on with API v2. It turned out that the problem with VideoFrame was that it was not copyable. I used boost::lockfree::spsc_queue, which will support non-copyable objects only in Boost 1.86. I don't know why it worked in v1. I changed to storing plain C GstVideoFrame in the queue. map/unmap anyway take care of reference counting. An additional benefit is that GstVideoFrame is now not allocated on the heap.
Since many methods return now non-standard strings, you must change all of your instances of std::format, fmt::print, fmt::format. For example, this worked before: