n
Oct. 17, 2022, 9:40 p.m.

Как сохранить данные древовидной модели на основе QStandardItemModel в файл

На самом деле вопросов у меня много, и не только вопросов, но и хотелок.
Хотелось бы чтобы кто-нибудь очень умный написал статью/материал/лекцию о работе с древовидными моделями в Qt, именно с древовидными, и досконально разжевал бы для "чайников" как это все устроено:

Что такое модель данных? Хотелось бы по-подробнее, с примерами, простыми словами. В одной статье при попытке объяснить что такое модель встретил такую фразу "Ваши данные - это и есть модель. Используйте их". Очень информативно :)))

Хотелось бы по-подробнее узнать сферу применения моделей, какие преимущества они дают, а где лучше их не использовать?

Что такое "источник данных"? Например если мы в программе считываем содержимое текстового файла в QStringList list что является источником файл или list?
Что может быть источником данных? База данных, файл, QStringList, QVector и т.д. Какие-еще могут быть варианты?

Какие функциии для чего нужны при реализации приложения модель/представление? Например во многих статьях просто пишут, "для того чтобы все работало, нам нужно реализовать вот эти функции". И перечисляется список функций. А для чего они необходимы и почему без них нельзя - ни слова.

Когда возникает тот или иной сигнал и каким из них лучше воспользоваться в той или иной ситуации.

И самое для меня важное: каковы алгоритмы заполнения ДРЕВОВИДНЫХ моделей данными из "источника данных" и как затем эти измененные данные записать обратно в "источник данных".

Может быть уже есть такие материалы, а я просто не знаю, тогда подскажите где почитать.
Я пытался искать на https://doc.qt.io, и в книжке Макса Шлее, а в книжке Марка Саммерфилда например написано: "Методы load() и save() служат для загрузки и сохранения данных при-
ложения; мы их не приводим".
Видимо их реализация элементарна :)))

Интернет-поиск дает десятки, сотни ссылок на статьи о том как работать с моделями, но практически все они описавают одни и те же примеры как создать модель списка или таблицы, но не дерева вложенности N и нигде не рассматриваются вопросы заполнения модели данными и сохранения данных из модели после ее редактирования. Максимум рассматриваются примеры работы с базой данных, но это таблица, а про ДЕРЕВО вложенности N ничего нет.

Ну это были мои хотелки, а теперь собственно и вопрос.

Я пишу приложение с использованием QStandardItemModel и QTreeView.
В QTreeView мне нужно отображать дерево элементов со степенью вложенности N. Ну хотя-бы N=5. Всего элементов будет до 500. Данные для отображения хранятся в текстовом файле.
Вот как описывается каждый элемент дерева в моем файле:

  1. {
  2. node:нода 1-2
  3. parent:node1
  4. Text:
  5. текст ноды 1-2
  6. }

node: название элемента
parent: название элемента-родителя
Text: здесь может быть, а может и не быть какой-то нужный мне текст

Я создал модель на основе QStandardItemModel, худо-бедно написал функцию заполняющую ее данными, в представлении QTreeView я могу добавлять элементы, переименовывать, перетаскивать. Но после этого мне нужно сохранить данные обратно в файл. Вот с этим возникла проблема.

Я рассуждаю примерно так:
Нужно перебрать в цикле все элементы модели (items) и для каждого из них записать в файл название элемента и измененные данные о его родителе, если родителя например переименовали, или элемент поменял родителя при перетаскивании.
Но как перебрать все элементы модели? Не могу ничего придумать.
Может нужен какой-то другой алгоритм?

2
6
МЛ
  • Oct. 17, 2022, 10:26 p.m.

структура файла регламентирована? или вы ее сами придумали и она не обязательна должна быть такой. в вашем случае нужно либо хранить в программе изменения дибо каждый раз очищать весь файл и писать заново. вы как то хоаните в моделе индексы на ноды из файла?

    n
    • Oct. 17, 2022, 11:10 p.m.

    Структуру я придумал сам, она может быть другой. Но тогда нужно будет переписывать функцию, которая заполняет модель. Я потратил на написание этой функции кучу времени.
    Зачем хранить в программе изменения, ведь они есть в модели?
    В программе я считываю файл в QStringList и модель заполняется уже из него. Меня устроил бы вариант когда я полностью перезаписываю этот QStringListl данными из модели, а потом записываю его в файл поверх старого содержимого.
    Проблема в том как достать из модели данные по всем элементам.

      МЛ
      • Oct. 18, 2022, 12:07 a.m.

      у QStandartModelItem есть фунции index, data и hasChildren. в цикле бегаете по веткам рекурсивно. вообще пооцесс обратный заполнению. как вы в модель когда писали, искали нужного родителя? можно код функции в студию.

        n
        • Oct. 18, 2022, 11:54 a.m.
        • (edited)

        Привожу код функции построения дерева элементов.
        full_list это текст считанный из файла. Он содержит всю информацию о дереве и другие данные.
        ui->txtDbg это окошко которое я добавил для отладки, я пишу туда отладочные сообщения.
        oneNoda - это как бы объект элемента.
        Функция просто перебирает в QStringList все элементы по порядку, анализирует имя элемента, и кто у него родитель и просто добавляет итем к родителю.
        Кстати обратите внимание на строки 24 и 25. В одном случае мы добавляем итемы к корневому элементу root и получается что в модели один элемент root с дочерними элементами. А в другом случае каждый итем добавляется в модель. Результат в обоих случаях выглядит одинаково, но как надо делать правильно?
        Не могу понять как запрограммировать в функции процесс обратный процессу построения дерева. Если можно накидайте хотя-бы каркас.

        1. void tetr::build_tree()
        2. {
        3. QStringList node_name_list=full_list->filter("node:");
        4. QStringList roditel_name_list=full_list->filter("parent:");
        5. struct Noda {
        6. QString noda_name;
        7. QString roditel_name;
        8. };
        9.  
        10. if(node_name_list.size()!=roditel_name_list.size()){
        11. ui->txtDbg->append("WARNING!!! node_name_list.size != roditel_name_list.size");
        12. return;
        13. }
        14.  
        15. Noda oneNoda;
        16. for(int i=0;i<node_name_list.size();i++){
        17. oneNoda.noda_name=node_name_list.at(i).section(':',1,1);
        18. oneNoda.roditel_name=roditel_name_list.at(i).section(':',1,1);
        19. QStandardItem *item=new QStandardItem(oneNoda.noda_name);
        20. item->setFlags(Qt::ItemIsEnabled|Qt::ItemIsSelectable|Qt::ItemIsEditable|Qt::ItemIsDragEnabled|Qt::ItemIsDropEnabled);
        21. QList <QStandardItem*>list=model->findItems(oneNoda.roditel_name,Qt::MatchExactly|Qt::MatchRecursive);
        22.  
        23. if(oneNoda.roditel_name=="root"){ //если элемент корневой, добавляем его в модель.
        24. //model->appendRow(item);
        25. root->appendRow(item);
        26. }
        27. //проверка количества найденных элементов-родителей
        28. if(list.size()>1){
        29. ui->txtDbg->append("В модели найдено более одного элемента с данным именем.");
        30. for(int i=0;i<list.size();i++){
        31. ui->txtDbg->append(list[i]->data().toString());
        32. }
        33. }
        34.  
        35. if(list.size()==1){
        36. //ui->txtDbg->append("Для элемента \""+oneNoda.noda_name+"\" в модели найден единственный элемент-родитель \""+list[0]->data().toString()+"\"");
        37. list[0]->appendRow(item);
        38. }
        39. if(list.size()==0 && oneNoda.roditel_name!="root"){//если элемент не найден И это не элемент "root"
        40. //(элемента root действительно нет. Он указан как родитель верхнего уровня
        41. ui-> txtDbg->append("Для элемента \""+oneNoda.noda_name+"\" в модели отсутствует элемент-родитель \""+oneNoda.roditel_name+"\"");
        42. }
        43. }
        44.  
        45. node_name_list.clear();
        46. roditel_name_list.clear();
        47. }
        48.  
          МЛ
          • Oct. 18, 2022, 2:35 p.m.

          Зачем вы тогда храните в файле номера нодов и родителей, если вы ищите родителя в своей модели по имени? При таком раскладе конечно можно найти несколько родителей, а кто настоящий то?
          Я так понимаю

          1. node:node 1-2
          2. parent:node1
          3. Text:
          4. текст ноды 1-2

          в наименовании node 1-2 вы уже храните и индекс родителя и индекс нода, тоесть node 1-2 это 2ой потомок node1.

          Если вы создаете QStandartItem а потом добавляете его в QTreeWidget, то он все равно это все добавляет в модель. Пишите сразу в модель через setData.

            n
            • Oct. 18, 2022, 8:14 p.m.

            Вы меня неправильно поняли. Допустим я собираю кулинарные рецепты и один из них - рецепт супа Харчо. Структура файла данных такова:

            1. {
            2. node:Рецепт супа Харчо
            3. parent:Вкусные блюда, которые мне нравятся
            4. Text:
            5. Нужно взять все необходимые продукты и приготовить суп Харчо. Суп Харчо очень вкусный
            6. Я очень его люблю!!!
            7. }

            Если вы создаете QStandartItem а потом добавляете его в QTreeWidget, то он все равно это все добавляет в модель.

            Не могли бы Вы выразиться более понятно. Кто "он" все равно добавляет все это в модель? QStandardItem добавляет или QTreeWidget добавляет? Каким образом "он" добавляет это в модель? Где эта функция?
            Я использую QTreeView а не QTreeWidget.

            Давайте забудем пока о функции build_tree. Я сделал ее как мог и она записывает данные в модель и они правильно отображаются.
            Вернемся к началу;
            У меня есть модель MyModel model. Она содержит всю необходимую информацию. Мне нужно перебрать все итемы в модели и каждый записать в QStringList full_list.
            Для этого мне нужно создать функцию которая это будет делать, возможно с использованием функции setData()
            Должно получиться что-то вот такое

            1. void tetr::write_data_from_model_to_full_list()
            2. {
            3. for("тип" i="что-то";i<="еще_что-то";i++) //перебираем все итемы модели
            4. {
            5. QModelIndex ind=<что-то>; //вычисляем индекс для i-того итема
            6. QStandardItem *item=model->itemFromIndex(ind);// получаем итем по индексу
            7. QString str=item->data(Qt::DisplayRole).toString(); //получаем имя итема
            8. full_list.append(str); //записываем текст в QStringList full_list который затем будем записывать в файл
            9.  
            10. }
            11. }

            Что использовать вместо "что-то" и "еще_что-то"? Вот в этом моя проблема.

            Алгоритм предложенный мной для функции write_data_from_model_to_full_list() прост и понятен. Но как перебрать все итемы модели. Может быть нужно использовать какой-то другой алгоритм переноса данных из модели в файл?

              Comments

              Only authorized users can post comments.
              Please, Log in or Sign up
              • Last comments
              • Evgenii Legotckoi
                March 9, 2025, 9:02 p.m.
                К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
              • VP
                March 9, 2025, 4:14 p.m.
                Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…
              • ИМ
                Nov. 22, 2024, 9:51 p.m.
                Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
              • Evgenii Legotckoi
                Oct. 31, 2024, 11:37 p.m.
                Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
              • A
                Oct. 19, 2024, 5:19 p.m.
                Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html