AUI Framework  develop
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
APixelFormat.h
    1/*
    2 * AUI Framework - Declarative UI toolkit for modern C++20
    3 * Copyright (C) 2020-2025 Alex2772 and Contributors
    4 *
    5 * SPDX-License-Identifier: MPL-2.0
    6 *
    7 * This Source Code Form is subject to the terms of the Mozilla Public
    8 * License, v. 2.0. If a copy of the MPL was not distributed with this
    9 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
   10 */
   11
   12//
   13// Created by Alex2772 on 12/12/2022.
   14//
   15
   16#pragma once
   17
   18#include <cstdint>
   19#include <cstddef>
   20
   21#include <AUI/Common/AColor.h>
   22
   27class APixelFormat {
   28public:
   29    enum Value : std::uint32_t {
   30        UNKNOWN = 0,
   31        BYTE = 0b1,
   32        FLOAT = 0b10,
   33        R      = 0b000001'00,
   34        RG     = 0b000011'00,
   35        RGB    = 0b000111'00,
   36        RGBA   = 0b001111'00,
   37        ARGB   = 0b011111'00,
   38        BGRA   = 0b111111'00,
   39
   40        RGB_BYTE = RGB | BYTE,
   41        RGBA_BYTE = RGBA | BYTE,
   42        RGBA_FLOAT = RGBA | FLOAT,
   43
   44        // WARNING! please handle AImageView::visit cases when adding new enum values
   45
   46        DEFAULT = RGBA | BYTE,
   47    };
   48
   49    static constexpr std::uint32_t COMPONENT_BITS = BGRA;
   50    static constexpr std::uint32_t TYPE_BITS = 0b11;
   51
   52    constexpr APixelFormat(Value value) noexcept: mValue(value) {}
   53    constexpr APixelFormat(std::uint32_t value) noexcept: mValue((Value)value) {}
   54
   55    constexpr operator Value() const noexcept {
   56        return mValue;
   57    }
   58
   59    [[nodiscard]]
   60    uint8_t bytesPerPixel() const noexcept {
   61        std::uint8_t b;
   62        switch (static_cast<std::uint8_t>(mValue & COMPONENT_BITS)) {
   63            case R: b = 1;   break;
   64            case RG: b = 2;  break;
   65            case RGB: b = 3; break;
   66            default: b = 4;  break;
   67        }
   68        if (mValue & FLOAT)
   69        {
   70            b *= 4;
   71        }
   72        return b;
   73    }
   74
   75
   76private:
   77    Value mValue;
   78};
   79
   80namespace aui::pixel_format {
   81    namespace detail {
   82
   83        template<typename T, std::uint32_t componentBits>
   84        struct component_representation;
   85
   86        template<typename T> struct component_representation<T, APixelFormat::R> {
   87            T r;
   88        };
   89
   90        template<typename T> struct component_representation<T, APixelFormat::RG> {
   91            T r, g;
   92        };
   93
   94        template<typename T> struct component_representation<T, APixelFormat::RGB> {
   95            T r, g, b;
   96        };
   97
   98        template<typename T> struct component_representation<T, APixelFormat::RGBA> {
   99            T r, g, b, a;
  100        };
  101        template<          > struct component_representation<float, APixelFormat::RGBA>: AColor {
  102            using AColor::AColor;
  103        };
  104
  105        template<typename T> struct component_representation<T, APixelFormat::BGRA> {
  106            T b, g, r, a;
  107        };
  108
  109        template<typename T> struct component_representation<T, APixelFormat::ARGB> {
  110            T a, r, g, b;
  111        };
  112
  113        template<std::uint32_t componentBits>
  114        constexpr std::size_t component_count() {
  115            return sizeof(component_representation<std::uint8_t, componentBits>);
  116        }
  117
  118
  119        template<std::uint32_t typeBits>
  120        struct type;
  121
  122        template<>
  123        struct type<APixelFormat::FLOAT> {
  124            using value = float;
  125        };
  126
  127        template<>
  128        struct type<APixelFormat::BYTE> {
  129            using value = std::uint8_t;
  130        };
  131
  132
  133        template<typename T>
  134        constexpr auto format_of = (APixelFormat)T::FORMAT;
  135        template<>
  136        inline constexpr auto format_of<AColor> = APixelFormat::RGBA_FLOAT;
  137    }
  138
  139    template<std::uint32_t format>
  140    struct traits {
  141        static constexpr std::size_t FORMAT = format;
  142        static constexpr std::size_t COMPONENT_COUNT = detail::component_count<format & APixelFormat::COMPONENT_BITS>();
  143        using component_t = typename detail::type<format & APixelFormat::TYPE_BITS>::value;
  144
  145
  146    private:
  147        using representation_t_impl = detail::component_representation<component_t, format & APixelFormat::COMPONENT_BITS>;
  148
  149
  150
  151    public:
  152
  153        struct representation_t: representation_t_impl {
  154            using super = representation_t_impl;
  155            static constexpr std::size_t FORMAT = format;
  156            explicit operator AColor() const noexcept;
  157
  158            component_t* begin() {
  159                return reinterpret_cast<component_t*>(this);
  160            }
  161
  162            const component_t* begin() const {
  163                return reinterpret_cast<const component_t*>(this);
  164            }
  165
  166            component_t* end() {
  167                return begin() + COMPONENT_COUNT;
  168            }
  169
  170            const component_t* end() const {
  171                return begin() + COMPONENT_COUNT;
  172            }
  173
  174            representation_t& operator+=(representation_t rhs) noexcept {
  175                std::transform(begin(), end(), rhs, begin(), std::plus<component_t>{});
  176                return *this;
  177            }
  178            representation_t& operator-=(representation_t rhs) noexcept {
  179                std::transform(begin(), end(), rhs, begin(), std::minus<component_t>{});
  180                return *this;
  181            }
  182            representation_t& operator*=(representation_t rhs) noexcept {
  183                std::transform(begin(), end(), rhs, begin(), std::multiplies<component_t>{});
  184                return *this;
  185            }
  186            representation_t& operator/=(representation_t rhs) noexcept {
  187                std::transform(begin(), end(), rhs, begin(), std::divides<component_t>{});
  188                return *this;
  189            }
  190
  191            representation_t& operator+=(component_t rhs) noexcept {
  192                std::transform(begin(), end(), begin(), [&](auto lhs) { return lhs + rhs; });
  193                return *this;
  194            }
  195            representation_t& operator-=(component_t rhs) noexcept {
  196                std::transform(begin(), end(), begin(), [&](auto lhs) { return lhs - rhs; });
  197                return *this;
  198            }
  199            representation_t& operator*=(component_t rhs) noexcept {
  200                std::transform(begin(), end(), begin(), [&](auto lhs) { return lhs * rhs; });
  201                return *this;
  202            }
  203            representation_t& operator/=(component_t rhs) noexcept {
  204                std::transform(begin(), end(), begin(), [&](auto lhs) { return lhs / rhs; });
  205                return *this;
  206            }
  207
  208            representation_t operator+(auto rhs) const noexcept {
  209                representation_t copy = *this;
  210                copy += rhs;
  211                return copy;
  212            }
  213            representation_t operator-(auto rhs) const noexcept {
  214                representation_t copy = *this;
  215                copy -= rhs;
  216                return copy;
  217            }
  218            representation_t operator*(auto rhs) const noexcept {
  219                representation_t copy = *this;
  220                copy *= rhs;
  221                return copy;
  222            }
  223            representation_t operator/(auto rhs) const noexcept {
  224                representation_t copy = *this;
  225                copy /= rhs;
  226                return copy;
  227            }
  228        };
  229    };
  230
  231    template<typename From, typename To>
  233
  234    template<typename T>
  235    struct component_converter<T, T> {
  236        static constexpr T convert(T t) {
  237            return t;
  238        }
  239    };
  240
  241    template<>
  242    struct component_converter<std::uint8_t, float> {
  243        static constexpr float convert(std::uint8_t t) {
  244            return float(t) / 255.f;
  245        }
  246    };
  247
  248    template<>
  249    struct component_converter<float, std::uint8_t> {
  250        static constexpr std::uint8_t convert(float t) {
  251            return std::uint8_t(t * 255.f);
  252        }
  253    };
  254
  255
  256    template<APixelFormat::Value from, APixelFormat::Value to>
  258        using traits_from = aui::pixel_format::traits<from>;
  259        using traits_to = aui::pixel_format::traits<to>;
  260        constexpr std::size_t countFrom = traits_from::COMPONENT_COUNT;
  261        constexpr std::size_t countTo = traits_to::COMPONENT_COUNT;
  262
  263        typename traits_to::representation_t out;
  264
  266
  267        out.r = my_component_converter::convert(in.r);
  268        if constexpr(countTo > 1) {
  269            if constexpr(countFrom > 1) {
  270                out.g = my_component_converter::convert(in.g);
  271            } else {
  272                out.g = my_component_converter::convert(0);
  273            }
  274        }
  275
  276        if constexpr(countTo > 2) {
  277            if constexpr(countFrom > 2) {
  278                out.b = my_component_converter::convert(in.b);
  279            } else {
  280                out.b = my_component_converter::convert(0);
  281            }
  282        }
  283
  284        if constexpr(countTo > 3) {
  285            if constexpr(countFrom > 3) {
  286                out.a = my_component_converter::convert(in.a);
  287            } else {
  288                out.a = component_converter<float, typename traits_to::component_t>::convert(1);
  289            }
  290        }
  291
  292        return out;
  293    }
  294
  295    template<std::uint32_t format>
  296    traits<format>::representation_t::operator AColor() const noexcept {
  297        return convert<(APixelFormat::Value)format, (APixelFormat::Value)(APixelFormat::RGBA | APixelFormat::FLOAT)>(*this);
  298    }
  299}
  300
  301
  302
  306template<auto pixelFormat = APixelFormat::DEFAULT>
  307using AFormattedColor = typename aui::pixel_format::traits<pixelFormat>::representation_t;
  308
  309template<typename Source>
  310struct AFormattedColorConverter {
  311public:
  312    constexpr explicit AFormattedColorConverter(const Source& color) : mColor(color) {}
  313
  314    template<typename Destination>
  315    constexpr operator Destination() const noexcept {
  316        constexpr auto source = aui::pixel_format::detail::format_of<std::decay_t<Source>>;
  317        return aui::pixel_format::convert<source, aui::pixel_format::detail::format_of<std::decay_t<Destination>>>(
  318                reinterpret_cast<const typename aui::pixel_format::traits<source>::representation_t&>(mColor)
  319                );
  320    }
  321
  322private:
  323    Source mColor;
  324
  325};
Represents a 4-component floating point color (RGBA).
Definition AColor.h:26
Pixel in-memory format descriptor (type, count and order of subpixel components).
Definition APixelFormat.h:27
Definition APixelFormat.h:232
Definition APixelFormat.h:153
Definition APixelFormat.h:140