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
|
#pragma once
#include "../include/Storage.hpp"
#include "../include/MessageQueue.hpp"
#include "../include/FileAnalyzer.hpp"
#include "../include/AsyncTask.hpp"
#include <SFML/System/Vector2.hpp>
#include <SFML/Graphics/Texture.hpp>
#include <SFML/System/Clock.hpp>
#include <string>
#include <memory>
#include <unordered_map>
namespace QuickMedia {
enum class LoadingState {
NOT_LOADED,
LOADING,
FINISHED_LOADING,
APPLIED_TO_TEXTURE
};
struct ThumbnailData {
LoadingState loading_state = LoadingState::NOT_LOADED;
sf::Texture texture;
std::unique_ptr<sf::Image> image; // Set in another thread. This should be .reset after loading it into |texture|, to save memory
size_t counter = 0;
};
// If |symlink_if_no_resize| is false then a copy is made from |thumbnail_path| to |thumbnail_path_resized| instead of a symlink if |thumbnail_path| is not larger than |resize_target_size|.
// One example of why you might not want a symlink is if |thumbnail_path| is a temporary file.
bool create_thumbnail(const Path &thumbnail_path, const Path &thumbnail_path_resized, sf::Vector2i resize_target_size, ContentType content_type, bool symlink_if_no_resize);
constexpr int NUM_IMAGE_LOAD_THREADS = 4;
class AsyncImageLoader {
public:
static AsyncImageLoader& get_instance();
// Never returns nullptr. Instead check the |loading_state| of the thumbnail data to see if it has finished loading.
// This function should be called every frame for the objects that need to display this thumbnail, otherwise it can be unloaded.
// set |resize_target_size| to {0, 0} to disable resizing.
// Note: this method is not thread-safe
std::shared_ptr<ThumbnailData> get_thumbnail(const std::string &url, bool local, sf::Vector2i resize_target_size);
// Note: this should only be called once every frame.
// Note: this method is not thread-safe
void update();
private:
AsyncImageLoader();
~AsyncImageLoader();
AsyncImageLoader(AsyncImageLoader &other) = delete;
AsyncImageLoader& operator=(AsyncImageLoader &other) = delete;
struct ThumbnailLoadData {
Path path;
Path thumbnail_path;
bool local;
std::shared_ptr<ThumbnailData> thumbnail_data;
sf::Vector2i resize_target_size;
};
// set |resize_target_size| to {0, 0} to disable resizing.
// Note: this method is not thread-safe
void load_thumbnail(const std::string &url, bool local, sf::Vector2i resize_target_size, std::shared_ptr<ThumbnailData> thumbnail_data);
// Returns -1 if all threads are busy
int get_free_load_index() const;
private:
bool loading_image[NUM_IMAGE_LOAD_THREADS];
// TODO: Use curl single-threaded multi-download feature instead
AsyncTask<void> download_image_thread[NUM_IMAGE_LOAD_THREADS];
AsyncTask<void> load_image_thread;
MessageQueue<ThumbnailLoadData> image_load_queue;
std::unordered_map<std::string, std::shared_ptr<ThumbnailData>> thumbnails;
size_t counter = 0;
};
}
|