C++/WinRT implementation inheritance: Notes on winrt::implements, part 6

Using CRTP to delegate the method. The post C++/WinRT implementation inheritance: Notes on winrt::implements, part 6 appeared first on The Old New Thing.

Mar 4, 2025 - 20:52
 0
C++/WinRT implementation inheritance: Notes on winrt::implements, part 6

Last time, we were looking for a way to allow a winrt::implements-based base class to defer implementation of a method to its derived class. The problem is that if you use winrt::implements to implement an interface, you have to implement all the methods. But you might also want to leave some of the methods for the derived class to implement in a manner of its choosing (for example, choosing whether the parameters are references or values, or allowing the parameters to be templated), which means that the virtual method approach doesn’t work. Something has to give.

There are multiple solutions, depending on which requirement you want to weaken.

You can weaken the “leave some of the methods for the derived class to implement” requirement by implementing it yourself but forwarding the call to the derived class via CRTP.

// A simple copy drop target provides no custom feedback
// and accepts anything by copy.
template
struct SimpleCopyDropTarget :
    winrt::implements,
    winrt::ICoreDropOperationTarget>
{
    winrt::IAsyncOperation
        EnterAsync(winrt::CoreDragInfo const& info,
                   winrt::CoreDragUIOverride const&)
    {
        co_return GetOperation(info);
    }

    winrt::IAsyncOperation
        OverAsync(winrt::CoreDragInfo const& info,
                  winrt::CoreDragUIOverride const&)
    {
        co_return GetOperation(info);
    }

    winrt::IAsyncAction
        LeaveAsync(winrt::CoreDragInfo const&)
    {
        co_return;
    }

    winrt::IAsyncOperation
        DropAsync(winrt::CoreDragInfo const& info)     
    {                                                  
        return static_cast(this)->                 
            DropAsyncImpl(info);                       
    }                                                  

protected:
    winrt::DataPackageOperation GetOperation(
        winrt::CoreDragInfo const& info)
    {
        return info.AllowedOperations() &
               winrt::DataPackageOperation::Copy;
    }
};

struct Derived : winrt::implements<
    Derived,
    SimpleCopyDropTarget>
{
    winrt::IAsyncOperation
        DropAsyncImpl(winrt::CoreDragInfo info)
    {
        auto lifetime = get_strong();

        auto operation = GetOperation(info);
        if (!(operation & winrt::DataPackageOperation::Copy)) {
            co_return winrt::DataPackageOperation::None;
        }

        ⟦ process the drop ⟧

        co_return winrt::DataPackageOperation::Copy;
    }
};

We implement DropAsync in the base class but immediately forward the call out to the derived class’s Drop­Async­Impl method. This is done via CRTP so that the derived class has full flexibility in deciding how to accept the parameters.

If you have access to “deducing this”, then you can let the “this” deduction do the work instead of CRTP.

// A simple copy drop target provides no custom feedback
// and accepts anything by copy.
struct SimpleCopyDropTarget :
    winrt::implements
{
    winrt::IAsyncOperation
        EnterAsync(winrt::CoreDragInfo const& info,
                   winrt::CoreDragUIOverride const&)
    {
        co_return GetOperation(info);
    }

    winrt::IAsyncOperation
        OverAsync(winrt::CoreDragInfo const& info,
                  winrt::CoreDragUIOverride const&)
    {
        co_return GetOperation(info);
    }

    winrt::IAsyncAction
        LeaveAsync(winrt::CoreDragInfo const&)
    {
        co_return;
    }

    winrt::IAsyncOperation
        DropAsync(this auto&& self,                    
                  winrt::CoreDragInfo const& info)     
    {                                                  
        return self.DropAsyncImpl(info);               
    }                                                  

protected:
    winrt::DataPackageOperation GetOperation(
        winrt::CoreDragInfo const& info)
    {
        return info.AllowedOperations() &
               winrt::DataPackageOperation::Copy;
    }
};

struct Derived : winrt::implements<
    Derived,
    SimpleCopyDropTarget>
{
    winrt::IAsyncOperation
        DropAsyncImpl(winrt::CoreDragInfo info)
    {
        auto lifetime = get_strong();

        auto operation = GetOperation(info);
        if (!(operation & winrt::DataPackageOperation::Copy)) {
            co_return winrt::DataPackageOperation::None;
        }

        ⟦ process the drop ⟧

        co_return winrt::DataPackageOperation::Copy;
    }
};

Another option is to weaken the “implement an interface” part of “use winrt::implements to implement an interface”. We can simply omit ICore­Drop­Operation­Target from the list of interfaces implemented by the base class, since our base class doesn’t contain a full implementation. Instead, let the derived class finish the implementation and declare the interface there.

// A simple copy drop target provides no custom feedback
// and accepts anything by copy.
struct SimpleCopyDropTarget :
    winrt::implementsIInspectable> // no ICoreDropOperationTarget
{
    winrt::IAsyncOperation
        EnterAsync(winrt::CoreDragInfo const& info,
                   winrt::CoreDragUIOverride const&)
    {
        co_return GetOperation(info);
    }

    winrt::IAsyncOperation
        OverAsync(winrt::CoreDragInfo const& info,
                  winrt::CoreDragUIOverride const&)
    {
        co_return GetOperation(info);
    }

    winrt::IAsyncAction
        LeaveAsync(winrt::CoreDragInfo const&)
    {
        co_return;
    }

    // DropAsync must be implemented by derived class

protected:
    winrt::DataPackageOperation GetOperation(
        winrt::CoreDragInfo const& info)
    {
        return info.AllowedOperations() &
               winrt::DataPackageOperation::Copy;
    }
};

struct Derived : winrt::implements<
    Derived,
    SimpleCopyDropTarget,
    winrt::ICoreDropOperationTarget>
{
    winrt::IAsyncOperation
        DropAsync(winrt::CoreDragInfo info)
    {
        auto lifetime = get_strong();

        auto operation = GetOperation(info);
        if (!(operation & winrt::DataPackageOperation::Copy)) {
            co_return winrt::DataPackageOperation::None;
        }

        ⟦ process the drop ⟧

        co_return winrt::DataPackageOperation::Copy;
    }
};

Our Simple­Copy­Drop­Target no longer implements I­Core­Drop­Operation­Target. Instead, it is Derived which implements I­Core­Drop­Operation­Target. The Simple­Copy­Drop­Target happens to provide some really handy implementations of I­Core­Drop­Operation­Target methods, but they aren’t actually hooked up to the I­Core­Drop­Operation­Target interface until the Derived class says, “And I implement I­Core­Drop­Operation­Target.”

The upside of this is that you don’t have to use CRTP or forwarders. The downside of this is that the Derived class has to remember to say “And I implement I­Core­Drop­Operation­Target,” because if nobody says it, then the interface isn’t implemented by anybody!

But wait, the Simple­Copy­Drop­Target doesn’t implement anything interesting any more. Why are we even bothering?

Next time, we’ll solve the problem a third way: Don’t even bother.

The post C++/WinRT implementation inheritance: Notes on winrt::implements, part 6 appeared first on The Old New Thing.