mirror of
https://github.com/RGBCube/serenity
synced 2025-07-24 22:37:34 +00:00
ChanViewer: Fetch the list of boards and allow switching the board
We put the list of boards in a combo box and allow the user to switch boards that way. :^)
This commit is contained in:
parent
6311a617be
commit
7a63277115
6 changed files with 145 additions and 1 deletions
90
Applications/ChanViewer/BoardListModel.cpp
Normal file
90
Applications/ChanViewer/BoardListModel.cpp
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
#include "BoardListModel.h"
|
||||||
|
#include <AK/JsonArray.h>
|
||||||
|
#include <AK/JsonObject.h>
|
||||||
|
#include <AK/JsonValue.h>
|
||||||
|
#include <LibCore/CHttpRequest.h>
|
||||||
|
#include <LibCore/CNetworkJob.h>
|
||||||
|
#include <LibCore/CNetworkResponse.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
BoardListModel::BoardListModel()
|
||||||
|
{
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
BoardListModel::~BoardListModel()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void BoardListModel::update()
|
||||||
|
{
|
||||||
|
CHttpRequest request;
|
||||||
|
request.set_hostname("a.4cdn.org");
|
||||||
|
request.set_path("/boards.json");
|
||||||
|
|
||||||
|
auto* job = request.schedule();
|
||||||
|
|
||||||
|
job->on_finish = [job, this](bool success) {
|
||||||
|
auto* response = job->response();
|
||||||
|
dbg() << "Board list download finished, success=" << success << ", response=" << response;
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
return;
|
||||||
|
|
||||||
|
dbg() << "Board list payload size: " << response->payload().size();
|
||||||
|
|
||||||
|
auto json = JsonValue::from_string(response->payload());
|
||||||
|
|
||||||
|
if (json.is_object()) {
|
||||||
|
auto new_boards = json.as_object().get("boards");
|
||||||
|
if (new_boards.is_array())
|
||||||
|
m_boards = move(new_boards.as_array());
|
||||||
|
}
|
||||||
|
|
||||||
|
did_update();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
int BoardListModel::row_count(const GModelIndex&) const
|
||||||
|
{
|
||||||
|
return m_boards.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
String BoardListModel::column_name(int column) const
|
||||||
|
{
|
||||||
|
switch (column) {
|
||||||
|
case Column::Board:
|
||||||
|
return "Board";
|
||||||
|
default:
|
||||||
|
ASSERT_NOT_REACHED();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GModel::ColumnMetadata BoardListModel::column_metadata([[maybe_unused]] int column) const
|
||||||
|
{
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
GVariant BoardListModel::data(const GModelIndex& index, Role role) const
|
||||||
|
{
|
||||||
|
auto& board = m_boards.at(index.row()).as_object();
|
||||||
|
if (role == Role::Display) {
|
||||||
|
switch (index.column()) {
|
||||||
|
case Column::Board:
|
||||||
|
return String::format("/%s/ - %s",
|
||||||
|
board.get("board").to_string().characters(),
|
||||||
|
board.get("title").to_string().characters());
|
||||||
|
default:
|
||||||
|
ASSERT_NOT_REACHED();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (role == Role::Custom) {
|
||||||
|
switch (index.column()) {
|
||||||
|
case Column::Board:
|
||||||
|
return board.get("board").to_string();
|
||||||
|
default:
|
||||||
|
ASSERT_NOT_REACHED();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
27
Applications/ChanViewer/BoardListModel.h
Normal file
27
Applications/ChanViewer/BoardListModel.h
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/JsonArray.h>
|
||||||
|
#include <LibGUI/GModel.h>
|
||||||
|
|
||||||
|
class BoardListModel final : public GModel {
|
||||||
|
public:
|
||||||
|
enum Column {
|
||||||
|
Board,
|
||||||
|
__Count,
|
||||||
|
};
|
||||||
|
|
||||||
|
static NonnullRefPtr<BoardListModel> create() { return adopt(*new BoardListModel); }
|
||||||
|
virtual ~BoardListModel() override;
|
||||||
|
|
||||||
|
virtual int row_count(const GModelIndex& = GModelIndex()) const override;
|
||||||
|
virtual int column_count(const GModelIndex& = GModelIndex()) const override { return Column::__Count; }
|
||||||
|
virtual String column_name(int) const override;
|
||||||
|
virtual ColumnMetadata column_metadata(int) const override;
|
||||||
|
virtual GVariant data(const GModelIndex&, Role = Role::Display) const override;
|
||||||
|
virtual void update() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
BoardListModel();
|
||||||
|
|
||||||
|
JsonArray m_boards;
|
||||||
|
};
|
|
@ -2,6 +2,7 @@ include ../../Makefile.common
|
||||||
|
|
||||||
OBJS = \
|
OBJS = \
|
||||||
ThreadCatalogModel.o \
|
ThreadCatalogModel.o \
|
||||||
|
BoardListModel.o \
|
||||||
main.o
|
main.o
|
||||||
|
|
||||||
APP = ChanViewer
|
APP = ChanViewer
|
||||||
|
|
|
@ -16,11 +16,19 @@ ThreadCatalogModel::~ThreadCatalogModel()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ThreadCatalogModel::set_board(const String& board)
|
||||||
|
{
|
||||||
|
if (m_board == board)
|
||||||
|
return;
|
||||||
|
m_board = board;
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
void ThreadCatalogModel::update()
|
void ThreadCatalogModel::update()
|
||||||
{
|
{
|
||||||
CHttpRequest request;
|
CHttpRequest request;
|
||||||
request.set_hostname("a.4cdn.org");
|
request.set_hostname("a.4cdn.org");
|
||||||
request.set_path("/g/catalog.json");
|
request.set_path(String::format("/%s/catalog.json", m_board.characters()));
|
||||||
|
|
||||||
auto* job = request.schedule();
|
auto* job = request.schedule();
|
||||||
|
|
||||||
|
|
|
@ -25,8 +25,11 @@ public:
|
||||||
virtual GVariant data(const GModelIndex&, Role = Role::Display) const override;
|
virtual GVariant data(const GModelIndex&, Role = Role::Display) const override;
|
||||||
virtual void update() override;
|
virtual void update() override;
|
||||||
|
|
||||||
|
void set_board(const String&);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ThreadCatalogModel();
|
ThreadCatalogModel();
|
||||||
|
|
||||||
|
String m_board { "g" };
|
||||||
JsonArray m_catalog;
|
JsonArray m_catalog;
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
|
#include "BoardListModel.h"
|
||||||
#include "ThreadCatalogModel.h"
|
#include "ThreadCatalogModel.h"
|
||||||
#include <LibDraw/PNGLoader.h>
|
#include <LibDraw/PNGLoader.h>
|
||||||
#include <LibGUI/GApplication.h>
|
#include <LibGUI/GApplication.h>
|
||||||
#include <LibGUI/GBoxLayout.h>
|
#include <LibGUI/GBoxLayout.h>
|
||||||
|
#include <LibGUI/GComboBox.h>
|
||||||
#include <LibGUI/GTableView.h>
|
#include <LibGUI/GTableView.h>
|
||||||
#include <LibGUI/GWindow.h>
|
#include <LibGUI/GWindow.h>
|
||||||
|
|
||||||
|
@ -16,11 +18,24 @@ int main(int argc, char** argv)
|
||||||
|
|
||||||
auto* widget = new GWidget;
|
auto* widget = new GWidget;
|
||||||
window->set_main_widget(widget);
|
window->set_main_widget(widget);
|
||||||
|
widget->set_fill_with_background_color(true);
|
||||||
widget->set_layout(make<GBoxLayout>(Orientation::Vertical));
|
widget->set_layout(make<GBoxLayout>(Orientation::Vertical));
|
||||||
|
|
||||||
|
auto* board_combo = new GComboBox(widget);
|
||||||
|
board_combo->set_only_allow_values_from_model(true);
|
||||||
|
board_combo->set_size_policy(SizePolicy::Fill, SizePolicy::Fixed);
|
||||||
|
board_combo->set_preferred_size(0, 20);
|
||||||
|
board_combo->set_model(BoardListModel::create());
|
||||||
|
|
||||||
auto* catalog_view = new GTableView(widget);
|
auto* catalog_view = new GTableView(widget);
|
||||||
catalog_view->set_model(ThreadCatalogModel::create());
|
catalog_view->set_model(ThreadCatalogModel::create());
|
||||||
|
|
||||||
|
board_combo->on_change = [&] (auto&, const GModelIndex& index) {
|
||||||
|
auto selected_board = board_combo->model()->data(index, GModel::Role::Custom);
|
||||||
|
ASSERT(selected_board.is_string());
|
||||||
|
static_cast<ThreadCatalogModel*>(catalog_view->model())->set_board(selected_board.to_string());
|
||||||
|
};
|
||||||
|
|
||||||
window->show();
|
window->show();
|
||||||
|
|
||||||
return app.exec();
|
return app.exec();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue