По работе пришлось сделать несколько своих компонентов на Wt: визард, диалог выбора каталогов и файлов на устройстве. Решил выложить на GitHub, может кому-нибудь понадобится. Под катом будет простой диалог выбора файлов на Wt+Boost.

Wt виджет-ориентированный фреймворк, который по API похож на Qt. То что Wt «похож» на Qt, но не Qt!

Модель данных


В основе абстрактная модель Wt::WAbstractItemModel(почти как QAbstractItemModel):

class FileSystemModel: public Wt::WAbstractItemModel
{
public:

    enum Roles
    {
        Name = Wt::UserRole + 1, Path = Wt::UserRole
    };

    FileSystemModel(Wt::WObject* parent = nullptr);

    bool isFile(const Wt::WModelIndex& index) const;
    bool isDir(const Wt::WModelIndex& index) const;

    virtual int columnCount(const Wt::WModelIndex& parent =
            Wt::WModelIndex()) const;

    virtual int rowCount(const Wt::WModelIndex& parent =
            Wt::WModelIndex()) const;

    virtual Wt::WModelIndex parent(const Wt::WModelIndex& index) const;

    virtual boost::any data(const Wt::WModelIndex& index, int role =
            Wt::DisplayRole) const;

    virtual Wt::WModelIndex index(int row, int column,
            const Wt::WModelIndex& parent = Wt::WModelIndex()) const;

    virtual boost::any headerData(int section, Orientation orientation =
            Horizontal, int role = DisplayRole) const;
private:

    std::unique_ptr<FileSystemModelPrivate> m_impl;
};

При реализации древовидной модели удобно иметь само дерево. Элемент дерева представлен структурой TreeNode:

struct TreeNode
{
    enum Type
    {
        File, Directory
    };

    std::string filename;
    std::string path;
    TreeNode* parent;
    std::vector<TreeNode*> children;
    Type type;
    bool childrenLoaded;

    TreeNode(TreeNode* prnt = nullptr) :
            parent(prnt),
            type(Directory),
            childrenLoaded(false)
    {
        if (parent)
        {
            parent->children.push_back(this);
        }
    }

    ~TreeNode()
    {
        parent = nullptr;
        for (TreeNode* child : children)
        {
            delete child;
        }
        children.clear();
    }

    size_t loadChildren()
    {
        if (childrenLoaded)
        {
            return children.size();
        }

        boost::filesystem::path p(path);

        childrenLoaded = true;
        size_t count = 0;

        try
        {
            for (directory_iterator iter(p), end; iter != end; ++iter)
            {
                auto itm = new TreeNode(this);
                itm->filename = iter->path().filename().string();
                itm->path = iter->path().string();
                itm->type =
                        is_directory(iter->path()) ?
                                TreeNode::Directory : TreeNode::File;
                ++count;
            }

            std::sort(children.begin(), children.end(),
                    [](const TreeNode* a, const TreeNode* b)
                    {
                        return a->filename<b->filename;
                    });

            return count;
        } catch (const filesystem_error&)
        {
            return 0;
        }
    }
};

Метод loadChildren осуществляет чтение файловой системы и подгрузку узлов. Узлы будут подгружаться по требованию, а именно тогда, когда у модели запросят rowCount. Корень дерева создается в конструкторе FileSystemModelPrivate:

struct FileSystemModelPrivate
{
    FileSystemModelPrivate() :
            root(new TreeNode)
    {
        root->filename = "/";
        root->path = "/";
    }

    std::unique_ptr<TreeNode> root;
};

В Wt так же как и в Qt есть метод createIndex создающий модельный индекс(WModelIndex) и позволяющий передавать указатель на TreeNode.

Остальной код очень прост:

FileSystemModel::FileSystemModel(WObject* parent) :
        WAbstractItemModel(parent),
        m_impl(new FileSystemModelPrivate)
{
}

int FileSystemModel::columnCount(const WModelIndex& parent) const
{
    return 1;
}

int FileSystemModel::rowCount(const WModelIndex& parent) const
{
    if (parent.isValid())
    {

        TreeNode* node = static_cast<TreeNode*>(parent.internalPointer());
        if (node == nullptr || node->type == TreeNode::File)
        {
            return 0;
        }

        return node->childrenLoaded ?
                node->children.size() : node->loadChildren();

    }
    else
    {
        //Unix root '/'
        return 1;
    }

    return 0;
}

WModelIndex FileSystemModel::parent(const WModelIndex& index) const
{
    if (!index.isValid())
    {
        return WModelIndex();
    }

    auto node = static_cast<TreeNode*>(index.internalPointer());
    if (node->parent == nullptr)
    {
        return WModelIndex();
    }

    if (node->parent->parent == nullptr)
    {
        return createIndex(0, 0, m_impl->root.get());
    }

    const auto grand = node->parent->parent;
    const auto parent = node->parent;

    const auto res = std::lower_bound(grand->children.cbegin(),
            grand->children.cend(), parent);

    const size_t row = std::distance(grand->children.cbegin(), res);

    return createIndex(row, 0, parent);
}

boost::any FileSystemModel::data(const WModelIndex &index, int role) const
{
    if (!index.isValid())
    {
        return boost::any();
    }

    auto node = static_cast<TreeNode*>(index.internalPointer());
    if (node == nullptr)
    {
        return boost::any();
    }

    switch (role)
    {
    case DisplayRole:
    {
        return node->filename;
    }
    case Path:
    {
        return node->path;
    }
    case DecorationRole:
    {
        try
        {
            return FILE_SYSTEM_ICONS.at(node->type);
        } catch (...)
        {

            return boost::any();
        }
    }
        break;
    default:
        return boost::any();
    }
}

WModelIndex FileSystemModel::index(int row, int column,
        const Wt::WModelIndex& parent) const
{
    if (!parent.isValid())
    {
        return createIndex(0, 0, m_impl->root.get());
    }
    TreeNode* pNode = static_cast<TreeNode*>(parent.internalPointer());
    if (pNode == nullptr)
    {
        return WModelIndex();
    }

    return createIndex(row, column, pNode->children[row]);
}
boost::any FileSystemModel::headerData(int section, Orientation orientation,
        int role) const
{
    if (role == DisplayRole && orientation == Horizontal)
    {
        return "File name";
    }

    return boost::any();
}


Диалог выбора файлов



Диалог выбора файлов наследуется от Wt::WDialog и имеет интерфейс:

class FileDialog: public Wt::WDialog
{
public:

    FileDialog(WObject* parent = nullptr);
    virtual void accept();

    Wt::WStringList selected() const;

private:

    Wt::WTreeView* m_view;
    FileSystemModel* m_fs;
};

Класс FileDialog содержит в себе нашу модель и древовидное представление Wt::WTreeView.

Рассмотрим конструктор:

FileDialog::FileDialog(WObject* parent) :
        WDialog(parent),
        m_view(new WTreeView()),
        m_fs(new FileSystemModel(this))

{
    setWindowTitle("Selecting files and directories");

    auto cancel = new WPushButton("Cancel", footer());
    cancel->clicked().connect(this, &WDialog::reject);

    m_view->setModel(m_fs);
    m_view->setSelectionBehavior(SelectItems);
    m_view->setSelectionMode(ExtendedSelection);

    auto select = new WPushButton("Select", footer());
    select->clicked().connect(this, &FileDialog::accept);
    m_view->setSortingEnabled(false);
    m_view->setHeaderHeight(0);
    m_view->expandToDepth(1);

    auto layout = new WVBoxLayout;
    layout->addWidget(m_view);
    contents()->setLayout(layout);

    resize(600, 500);
}

Внутри тела конструктора создаются две кнопки, которые размещаются в нижнем колонтитуле, вертикальный layout, который размещается на месте содержимого. Вертикальный layout нужен для выставления размеров WTreeView, иначе представление может выйти за пределы диалогового окна.

Конструкция
cancel->clicked().connect(this, &WDialog::reject);

это механизм сигнал/слотов на базе Boost.Signals2(или Boost.Signals, зависит от версии Boost). Два оставшихся методов тривиальны
void FileDialog::accept()
{
    const auto indxs = m_view->selectedIndexes();
    if (indxs.size() > 0)
    {
        WDialog::accept();
    }
}

Wt::WStringList FileDialog::selected() const
{
    WStringList list;
    const auto indxs = m_view->selectedIndexes();
    for (auto indx : indxs)
    {
        const WString pt = boost::any_cast<std::string>(
                indx.data(FileSystemModel::Path));
        list << pt;
    }

    return list;
}

Комментарии (7)


  1. scrutari
    26.01.2016 14:32

    Выглядит красиво. Скажите, это диалог позволяет выбрать файл только на сервере или и на клиенте тоже? То есть, зайдя на удаленный веб-сервер и открыв такой диалог, я увижу локальные файлы или файлы на удаленном сервере? Спасибо.


    1. RPG18
      26.01.2016 14:59

      На стороне сервера. В моей задаче пользователь выбирает какие файлы и каталоги он хочет бекапить на своем NAS'е.


      1. scrutari
        27.01.2016 00:17

        Ясно, спасибо. Была бы такая же красивая штука для клиентской стороны…


        1. Wedmer
          27.01.2016 10:04

          Если я не ошибаюсь, диалог выбора файла у клиента всегда предоставляется браузером, ибо js песочнице доступ к файловой системе закрыт.


  1. sashkin
    26.01.2016 16:13

    Как долго работаете с Wt? Можете поделиться впечатлениями о фреймворке? Ну там плюсы/минусы. Честно говоря веб разработка на плюсах даже с фреймворками кажется тяжким делом.


    1. RPG18
      26.01.2016 16:30

      Пятый месяц. В принципе работает. Несколько раз приходилось обращаться к своим frontend разработчикам, что бы те сделали верстку на html/css, которую впоследствии шаблонизировал.

      Для меня плюс это схожесть с Qt, что позволило сделать Wizard


    1. RPG18
      26.01.2016 16:41

      Из минусов: немного кривая поддержка Bootstrap 3, и нужно быть готовым что, что то может свалится в реализации сервера(многопоточный на базе Boost.Asio).

      Если бы хорошо знал бы фронтенд разработку, то взял бы Tufao т. к. изначально использовался Qt.

      Повторюсь, что задача не создать Web сайт, а приложение с Web интерфейсом, которое крутится на NAS(Synology, QNAP)