aboutsummaryrefslogtreecommitdiff
path: root/src/plugins/MangaCombined.cpp
blob: 5b8087201045200c1f33dcbe9400f078fab4ab9b (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
99
100
101
102
103
104
105
106
107
108
109
110
#include "../../plugins/MangaCombined.hpp"
#include "../../include/AsyncTask.hpp"

namespace QuickMedia {
    static const int SEARCH_TIMEOUT_MILLISECONDS = 6000;

    MangaCombinedSearchPage::MangaCombinedSearchPage(Program *program, std::vector<MangaPlugin> search_pages) :
        Page(program), search_pages(std::move(search_pages))
    {
        
    }

    static void result_items_add_thread_results(std::vector<std::pair<MangaPlugin*, AsyncTask<BodyItems>>> &search_threads, BodyItems &result_items) {
        std::vector<std::pair<MangaPlugin*, bool>> plugin_finished_state(search_threads.size());
        for(size_t i = 0; i < plugin_finished_state.size(); ++i) {
            plugin_finished_state[i].first = search_threads[i].first;
        }

        int accumulated_sleep_time = 0;
        while(true) {
            size_t finshed_plugin_index = 0;
            for(auto &f : plugin_finished_state) {
                f.second = false;
            }

            for(auto &search_thread : search_threads) {
                if(!search_thread.second.valid()) {
                    plugin_finished_state[finshed_plugin_index].second = true;
                    ++finshed_plugin_index;
                    continue;
                }

                if(!search_thread.second.ready())
                    continue;

                BodyItems search_page_body_items = search_thread.second.get();
                plugin_finished_state[finshed_plugin_index].second = true;
                ++finshed_plugin_index;
                if(search_page_body_items.empty())
                    continue;

                auto title_item = BodyItem::create("");
                title_item->set_author("======================== " + search_thread.first->title + " ========================");
                result_items.push_back(std::move(title_item));

                for(auto &new_body_item : search_page_body_items) {
                    new_body_item->userdata = search_thread.first->page.get();
                }
                result_items.insert(result_items.end(), std::move_iterator(search_page_body_items.begin()), std::move_iterator(search_page_body_items.end()));
            }

            size_t num_finished_plugins = 0;
            for(auto &f : plugin_finished_state) {
                if(f.second)
                    ++num_finished_plugins;
            }

            if(num_finished_plugins == search_threads.size())
                break;

            int sleep_time_ms = 200;
            std::this_thread::sleep_for(std::chrono::milliseconds(sleep_time_ms));
            accumulated_sleep_time += sleep_time_ms;
            if(accumulated_sleep_time >= SEARCH_TIMEOUT_MILLISECONDS) {
                for(auto &f : plugin_finished_state) {
                    if(!f.second) {
                        auto title_item = BodyItem::create("");
                        title_item->set_author("======================== " + f.first->title + " timed out ========================");
                        result_items.push_back(std::move(title_item));
                    }
                }
                break;
            }
        }
    }

    SearchResult MangaCombinedSearchPage::search(const std::string &str, BodyItems &result_items) {
        std::vector<std::pair<MangaPlugin*, AsyncTask<BodyItems>>> search_threads;
        for(auto &search_page : search_pages) {
            search_threads.push_back(std::make_pair(&search_page, AsyncTask<BodyItems>([&str, &search_page]() {
                BodyItems search_page_body_items;
                search_page.page->search(str, search_page_body_items);
                return search_page_body_items;
            })));
        }

        result_items_add_thread_results(search_threads, result_items);
        return SearchResult::OK;
    }

    PluginResult MangaCombinedSearchPage::get_page(const std::string &str, int page, BodyItems &result_items) {
        std::vector<std::pair<MangaPlugin*, AsyncTask<BodyItems>>> search_threads;
        for(auto &search_page : search_pages) {
            search_threads.push_back(std::make_pair(&search_page, AsyncTask<BodyItems>([&str, page, &search_page]() {
                BodyItems search_page_body_items;
                search_page.page->get_page(str, page, search_page_body_items);
                return search_page_body_items;
            })));
        }

        result_items_add_thread_results(search_threads, result_items);
        return PluginResult::OK;
    }

    PluginResult MangaCombinedSearchPage::submit(const std::string &title, const std::string &url, std::vector<Tab> &result_tabs) {
        Page *page = (Page*)submit_body_item->userdata;
        if(!page) return PluginResult::OK;
        return page->submit(title, url, result_tabs);
    }
}