summaryrefslogtreecommitdiff
path: root/Userland/Services/WindowServer/MultiScaleBitmaps.cpp
blob: e6434ee21e7b523cdb50b2fd2a924902d2830d59 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/*
 * Copyright (c) 2020, the SerenityOS developers.
 *
 * SPDX-License-Identifier: BSD-2-Clause
 */

#include "MultiScaleBitmaps.h"
#include "Screen.h"

namespace WindowServer {

const Gfx::Bitmap& MultiScaleBitmaps::bitmap(int scale_factor) const
{
    auto it = m_bitmaps.find(scale_factor);
    if (it == m_bitmaps.end()) {
        it = m_bitmaps.find(1);
        if (it == m_bitmaps.end())
            it = m_bitmaps.begin();
    }
    // We better found *something*
    if (it == m_bitmaps.end()) {
        dbgln("Could not find any bitmap in this MultiScaleBitmaps");
        VERIFY_NOT_REACHED();
    }
    return it->value;
}

Gfx::Bitmap const* MultiScaleBitmaps::find_bitmap(int scale_factor) const
{
    auto it = m_bitmaps.find(scale_factor);
    return it != m_bitmaps.end() ? it->value.ptr() : nullptr;
}

RefPtr<MultiScaleBitmaps> MultiScaleBitmaps::create_empty()
{
    return adopt_ref(*new MultiScaleBitmaps());
}

RefPtr<MultiScaleBitmaps> MultiScaleBitmaps::create(StringView const& filename, StringView const& default_filename)
{
    auto per_scale_bitmap = adopt_ref(*new MultiScaleBitmaps());
    if (per_scale_bitmap->load(filename, default_filename))
        return per_scale_bitmap;
    return {};
}

bool MultiScaleBitmaps::load(StringView const& filename, StringView const& default_filename)
{
    Optional<Gfx::BitmapFormat> bitmap_format;
    bool did_load_any = false;

    m_bitmaps.clear(); // If we're reloading the bitmaps get rid of the old ones

    auto add_bitmap = [&](StringView const& path, int scale_factor) {
        auto bitmap_or_error = Gfx::Bitmap::try_load_from_file(path, scale_factor);
        if (bitmap_or_error.is_error())
            return;
        auto bitmap = bitmap_or_error.release_value_but_fixme_should_propagate_errors();
        auto bitmap_format = bitmap->format();
        if (m_format == Gfx::BitmapFormat::Invalid || m_format == bitmap_format) {
            if (m_format == Gfx::BitmapFormat::Invalid)
                m_format = bitmap_format;

            did_load_any = true;
            m_bitmaps.set(scale_factor, move(bitmap));
        } else {
            // Gracefully ignore, we have at least one bitmap already
            dbgln("Bitmap {} (scale {}) has format inconsistent with the other per-scale bitmaps", path, bitmap->scale());
        }
    };

    Screen::for_each_scale_factor_in_use([&](int scale_factor) {
        add_bitmap(filename, scale_factor);
        return IterationDecision::Continue;
    });
    if (!did_load_any && !default_filename.is_null() && !default_filename.is_empty()) {
        Screen::for_each_scale_factor_in_use([&](int scale_factor) {
            add_bitmap(default_filename, scale_factor);
            return IterationDecision::Continue;
        });
    }
    return did_load_any;
}

void MultiScaleBitmaps::add_bitmap(int scale_factor, NonnullRefPtr<Gfx::Bitmap>&& bitmap)
{
    auto bitmap_format = bitmap->format();
    if (m_format == Gfx::BitmapFormat::Invalid || m_format == bitmap_format) {
        if (m_format == Gfx::BitmapFormat::Invalid)
            m_format = bitmap_format;
        m_bitmaps.set(scale_factor, move(bitmap));
    } else {
        dbgln("MultiScaleBitmaps::add_bitmap (scale {}) has format inconsistent with the other per-scale bitmaps", bitmap->scale());
        VERIFY_NOT_REACHED(); // The caller of this function should have made sure it is consistent!
    }
}

}