PortableImageMapLoader.h 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. /*
  2. * Copyright (c) 2020, Hüseyin ASLITÜRK <asliturk@hotmail.com>
  3. * Copyright (c) 2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/BufferedStream.h>
  9. #include <AK/MemoryStream.h>
  10. #include <AK/RefPtr.h>
  11. #include <AK/StringView.h>
  12. #include <AK/Types.h>
  13. #include <LibGfx/Bitmap.h>
  14. #include <LibGfx/ImageFormats/PortableImageLoaderCommon.h>
  15. namespace Gfx {
  16. template<class TFormatDetails>
  17. struct PortableImageMapLoadingContext {
  18. using FormatDetails = TFormatDetails;
  19. enum class Type {
  20. Unknown,
  21. ASCII,
  22. RAWBITS
  23. };
  24. enum class State {
  25. NotDecoded = 0,
  26. Error,
  27. MagicNumber,
  28. Width,
  29. Height,
  30. Maxval,
  31. Bitmap,
  32. Decoded
  33. };
  34. Type type { Type::Unknown };
  35. State state { State::NotDecoded };
  36. size_t width { 0 };
  37. size_t height { 0 };
  38. FormatDetails format_details {};
  39. RefPtr<Gfx::Bitmap> bitmap;
  40. NonnullOwnPtr<SeekableStream> stream;
  41. PortableImageMapLoadingContext(NonnullOwnPtr<SeekableStream> stream)
  42. : stream(move(stream))
  43. {
  44. }
  45. };
  46. template<typename TContext>
  47. class PortableImageDecoderPlugin final : public ImageDecoderPlugin {
  48. public:
  49. static bool sniff(ReadonlyBytes);
  50. static ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> create(ReadonlyBytes);
  51. virtual ~PortableImageDecoderPlugin() override = default;
  52. virtual IntSize size() override;
  53. virtual ErrorOr<void> initialize() override { return {}; }
  54. virtual bool is_animated() override;
  55. virtual size_t loop_count() override;
  56. virtual size_t frame_count() override;
  57. virtual size_t first_animated_frame_index() override;
  58. virtual ErrorOr<ImageFrameDescriptor> frame(size_t index, Optional<IntSize> ideal_size = {}) override;
  59. virtual ErrorOr<Optional<ReadonlyBytes>> icc_data() override;
  60. private:
  61. PortableImageDecoderPlugin(NonnullOwnPtr<SeekableStream> stream);
  62. OwnPtr<TContext> m_context;
  63. };
  64. template<typename TContext>
  65. PortableImageDecoderPlugin<TContext>::PortableImageDecoderPlugin(NonnullOwnPtr<SeekableStream> stream)
  66. {
  67. m_context = make<TContext>(move(stream));
  68. }
  69. template<typename TContext>
  70. IntSize PortableImageDecoderPlugin<TContext>::size()
  71. {
  72. if (m_context->state == TContext::State::Error)
  73. return {};
  74. if (m_context->state < TContext::State::Decoded) {
  75. if (decode(*m_context).is_error()) {
  76. m_context->state = TContext::State::Error;
  77. // FIXME: We should propagate errors
  78. return {};
  79. }
  80. }
  81. return { m_context->width, m_context->height };
  82. }
  83. template<typename TContext>
  84. ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> PortableImageDecoderPlugin<TContext>::create(ReadonlyBytes data)
  85. {
  86. auto stream = TRY(try_make<FixedMemoryStream>(data));
  87. return adopt_nonnull_own_or_enomem(new (nothrow) PortableImageDecoderPlugin<TContext>(move(stream)));
  88. }
  89. template<typename TContext>
  90. bool PortableImageDecoderPlugin<TContext>::sniff(ReadonlyBytes data)
  91. {
  92. using Context = TContext;
  93. if (data.size() < 2)
  94. return false;
  95. if (data.data()[0] == 'P' && data.data()[1] == Context::FormatDetails::ascii_magic_number)
  96. return true;
  97. if (data.data()[0] == 'P' && data.data()[1] == Context::FormatDetails::binary_magic_number)
  98. return true;
  99. return false;
  100. }
  101. template<typename TContext>
  102. bool PortableImageDecoderPlugin<TContext>::is_animated()
  103. {
  104. return false;
  105. }
  106. template<typename TContext>
  107. size_t PortableImageDecoderPlugin<TContext>::loop_count()
  108. {
  109. return 0;
  110. }
  111. template<typename TContext>
  112. size_t PortableImageDecoderPlugin<TContext>::frame_count()
  113. {
  114. return 1;
  115. }
  116. template<typename TContext>
  117. size_t PortableImageDecoderPlugin<TContext>::first_animated_frame_index()
  118. {
  119. return 0;
  120. }
  121. template<typename TContext>
  122. ErrorOr<ImageFrameDescriptor> PortableImageDecoderPlugin<TContext>::frame(size_t index, Optional<IntSize>)
  123. {
  124. if (index > 0)
  125. return Error::from_string_literal("PortableImageDecoderPlugin: Invalid frame index");
  126. if (m_context->state == TContext::State::Error)
  127. return Error::from_string_literal("PortableImageDecoderPlugin: Decoding failed");
  128. if (m_context->state < TContext::State::Decoded) {
  129. if (decode(*m_context).is_error()) {
  130. m_context->state = TContext::State::Error;
  131. return Error::from_string_literal("PortableImageDecoderPlugin: Decoding failed");
  132. }
  133. }
  134. VERIFY(m_context->bitmap);
  135. return ImageFrameDescriptor { m_context->bitmap, 0 };
  136. }
  137. template<typename TContext>
  138. ErrorOr<Optional<ReadonlyBytes>> PortableImageDecoderPlugin<TContext>::icc_data()
  139. {
  140. return OptionalNone {};
  141. }
  142. }