crossplatform.ru

Здравствуйте, гость ( Вход | Регистрация )

2 страниц V   1 2 >  
Ответить в данную темуНачать новую тему
> Программа для создания электронных книг, Обсуждение программы и кода
xwicked
  опции профиля:
сообщение 8.8.2011, 20:36
Сообщение #1


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Здравствуйте! Вот вторая моя программа, написанная на C++(Qt). Выношу на обсуждение её и исходник - 0.0.7.

Ссылка на SourceForge.net. Код под GNU GPL v2 - OpenSource.

Она предназначена для создания электронных книг, с централизованным хранением в базе данных SQLite3.

Этот код публикуется впервые. Его не было нигде, включая репозиторий программ Sisyphus от дистрибутива AltLinux. Там был только Клёст-кроссворд(0.1.9, 0.2.0). Комментируем Windows-версия.

Описание:
Страница книги представляет собой одну переменную QString Page.
Текст книги отображается в компоненте QTextEdit TextBook.
Текст в компоненте TextBook имеет форматирование документа HTML. Поэтому редактирование текста страницы осуществляется редактированием html-кода.

Стандартное сохранение идёт в базу данных. Но есть и возможность экспорта в файл.

Изображения хранятся, как в обычной html-странице - отдельно от текста в папке "temp". В тексте указывается путь.
При загрузке книги - во всех страницах происходит перезапись тегов изображения "<img src=":
      QString str1;
      int iFirst,
          iLast,
          iLength,
          iSearch;
          for(int i = 0; i < ListPages.count(); i++)
              {
                  str1 = ListPages.at(i);
                      if(str1.indexOf("<img src=\"") != -1)
                          {
                               int z = 1;
                               iSearch = 0;
                                   while(str1.indexOf("<img src=\"", iSearch) != -1)
                                       {
                                                if(z == 1) iSearch = -1;
                                                if(iSearch == -1)
                                                    {
                                                         iFirst = str1.indexOf("<img src=\"", 0);
                                                         iLast = str1.indexOf("/temp/Image", 0);
                                                         z++;
                                                    }
                                                else
                                                    {
                                                         iFirst = str1.indexOf("<img src=\"", iSearch);
                                                         iLast = str1.indexOf("/temp/Image", iFirst);
                                                    }
                                            iFirst += 10;
                                            iLength = iLast - iFirst;
                                            str1.remove(iFirst, iLength);
                                            str1.insert(iFirst, "/tmp/xwel");
                                            ListPages.replace(i, str1);
                                            iSearch = iLast;
                                       }
                          }
              }
Это актуально для переносимой версии. В этом случае необходимо будет поменять строку:
str1.insert(iFirst, "/tmp/xwel");
на
str1.insert(iFirst, QApplication::applicationDirPath());//Текущая папка, из которой запущено приложение
Пароль книги хранится в открытом виде в переменной QString Password.
Список страниц хранится в списке строк QStringList ListPages.
Изображения хранятся в массиве QByteArray ListImages[255]. Здесь видно, что изображений на всю книгу 255.

Сохранение в базу происходит обычными SQL-запросами:
      strBook = "DROP TABLE Table" + sNumBook + ";";
      QueryBook.exec(strBook);
      strBook = "CREATE TABLE Table" + sNumBook + " (Pages VARCHAR, Images BLOB);";
      QueryBook.exec(strBook);
      ...
      //Сохранение страниц
      if(ListPages.count() != 0)
          for(int i = 0; i < ListPages.count(); i++)
              {
                 QueryBook.prepare("INSERT INTO Table" + sNumBook + " (Pages) VALUES (?);");
                 QueryBook.bindValue(0, ListPages.at(i));
                 QueryBook.exec(/*strBook*/);
              }
      //Сохранение изображений
          for(int i = 0; i < 75; i++)//Здесь видно - что сохраняются только 75 изображений(!) из 255 :(
              {
                    if(ListImages[i].count() != 0)
                       {
                          QueryBook.prepare("INSERT INTO Table" + sNumBook + " (Images) VALUES (?);");
                          QueryBook.bindValue(0, ListImages[i]);
                          QueryBook.exec(/*strBook*/);
                       }
                   else break;
       }
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
Алексей1153
  опции профиля:
сообщение 8.8.2011, 21:32
Сообщение #2


фрилансер
******

Группа: Участник
Сообщений: 2939
Регистрация: 19.6.2010
Из: Обливион
Пользователь №: 1822

Спасибо сказали: 215 раз(а)




Репутация:   34  


лично я просел на фразе "Этот код публикуется впервые" :)

Цитата
Изображения хранятся в массиве QByteArray ListImages[255].

лучше примени контейнер
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 9.8.2011, 22:27
Сообщение #3


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Цитата(Алексей1153 @ 8.8.2011, 21:32) *
...
Цитата
Изображения хранятся в массиве QByteArray ListImages[255].

лучше примени контейнер
Согласен применю. Думаю уже в версии 0.2.1.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 10.8.2011, 9:00
Сообщение #4


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Вот спекфайл, по которому можно собрать rpm-пакет для дистрибутива AltLinux или своего, основанного на rpm:
Цитата
Summary: Qt program for creating and reading electronic books.

Summary(ru): Qt программа для создания и чтения электронных книг.

%define version 0.0.7

Group: Education

Name: klen-library

BuildRequires: gcc-c++, libqt4-sql-sqlite >= 4.4.3, libqtgui4 >= 4.4.3, libqt4-core >= 4.4.3

BuildArch: %ix86

Provides: klen-library

Release: alt1

Source: klen-library-%{version}.tar.gz

Url: http://labfreetech.org/index_e.html

Version: %{version}

License: GPLv2

Packager: LabFreeTech.org admin@labfreetech.org

%description
Qt program for creating and reading electronic books.
Supported features: - Creation of a textbook; - Editing of the textbook; - Reading the textbook; - Adding / removing books from the database; - Editing the font of the textbook; - Export and import of books in the file.

%description -l ru
Qt программа для создания и чтения электронных книг.
Поддерживаемые функции: - Создание учебника; - Редактирование учебника; - Чтение учебника; - Добавление/удаление учебников из базы данных; - Редактирование шрифта учебника; - Экспорт и импорт учебников в файл.

%prep

%setup -q

%build
%make_build
#cp -f klen-library debian/klen/opt/klen-library/

%install

cp -Rv klen/ %buildroot/

#rm -rf $RPM_BUILD_ROOT

%clean

rm -rf $RPM_BUILD_ROOT

%files

/usr/bin/*
/usr/share/*

%post

test -e /usr/bin/xwel-0.0.7 || ln -s /usr/bin/xwel-0.0.7 /usr/bin/klen-library-0.0.7
test -e $HOME/Desktop/xwel.desktop || cp -r /usr/share/applications/xwel.desktop $HOME/Desktop/klen-library.desktop

%postun

test -L /usr/bin/klen-library-0.0.7 && rm -r /usr/bin/klen-library-0.0.7
test -e $HOME/Desktop/klen-library.desktop && rm -r $HOME/Desktop/klen-library.desktop
И вот архив, подготовленный для сборки. Он включает несколько изменений в оформлении кода: ссылка. Этот спекфайл находится внутри архива. Для сборки необходим пакет rpm-build, Команда для сборки двоичного пакета:
Цитата
$rpmbuild -bb /home/username/RPM/SPECS/xwel-0.0.7-alt.spec
Оставляем комментарии, ищем ошибки.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 12.8.2011, 21:11
Сообщение #5


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Здравствуйте!

Следующая версия программы 0.0.8. Исходник + сборки. Её ключевой особенностью является импорт html-файла. Вот функция:
void ElectronicLibrary::OpenHTMLFile()
{
     QString FileNameBook, str, sHTML;
     FileNameBook = QFileDialog::getOpenFileName(0, tr("Book import"), "", "*.html");

         if(FileNameBook == "") return;
         else
             {
                 QFile ifile(FileNameBook);
                 QTextStream istream(&ifile);
                 float i = 0;//Величина %
                 float ipt,//Численное значние "pt"
               iptc;//Отвечает за таблицу
                 iptc = 0;
                 ipt = 0;
                 ifile.open(QIODevice::ReadOnly);

                     while(!ifile.atEnd())
                         {
                             str.append(istream.readLine());
                             str.append(" ");
                             sHTML.append(str);

                                 if(str.indexOf("<TABLE") != -1) iptc = 1;//Начало таблицы
                                 if(str.indexOf("/TABLE>") != -1) iptc = 2;//Конец таблицы

                             int j, iptc2;
                             QString s1,s2;
                             s1.clear();
                             s2.clear();
                             iptc2 = 0;

                //Поиск элемента обозначения шрифта
                             j = str.indexOf("font-size:");

//Если элемент найден и найдено вхождение "pt", то
                                 if(j != -1 && str.indexOf("pt", j) != -1)
                                     {
                      //Добавление одной цифры, стоящей за "pt"
                                          s1.append(str.at(str.indexOf("pt", j) - 1));

//Если второй символ стоящий за первым числом не пробел, то
                                              if(str.at(str.indexOf("pt", j) - 2) != ' ')
                      //Добавить его как вторую цифру числа шрифта(ex. "14pt")
                          s2.append(str.at(str.indexOf("pt", j) - 2));

              //Соединить с первой цифрой в s1. Образовать число.
                                              if(s2 != "") s1.append(s2);

                                          bool bOk;
                                          ipt = s1.toFloat(&bOk);
//6pt = 113 = 0.885% от страницы
//16pt = 42 = 2.380% от страницы
//0.1375 - условный средний коэффициент % занимаемого значения 6pt на одной странице, формата A4
//Усреднённая формула нахождения % текущей строки от страницы, формата A4, относительно 6pt
                                          ipt = (ipt / 6) * 0.1375;
                                     }

//Поиск элемента обозначения шрифта другой формы
                                 if(j == -1)
                                     {
                                         if(str.indexOf("FONT SIZE=1") != -1) j = 1;
                                         if(str.indexOf("FONT SIZE=2") != -1) j = 2;
                                         if(str.indexOf("FONT SIZE=3") != -1) j = 3;
                                         if(str.indexOf("FONT SIZE=4") != -1) j = 4;
                                         if(str.indexOf("FONT SIZE=5") != -1) j = 5;
                                         if(str.indexOf("FONT SIZE=6") != -1) j = 6;
                                         if(str.indexOf("FONT SIZE=7") != -1) j = 7;

                                             switch(j)//Вычисление процента
                                                 {
                                                      case 1:
                                                          ipt = (8 / 6) * 0.1375;
                                                      case 2:
                                                          ipt = (10 / 6) * 0.1375;
                                                      case 3:
                                                          ipt = (12 / 6) * 0.1375;
                                                      case 4:
                                                          ipt = (14 / 6) * 0.1375;
                                                      case 5:
                                                          ipt = (18 / 6) * 0.1375;
                                                      case 6:
                                                          ipt = (24 / 6) * 0.1375;
                                                      case 7:
                                                          ipt = (36 / 6) * 0.1375;
                                                 }
                                     }

                             i += ipt;//Суммирование процента

                                 if(i >= 100)//Если набралось 100%
                                     {
                                         if(iptc == 0)//Если нет таблицы в конце страницы
                                             {
                                                  sHTML.append("</body></html>");
                                                  ListPages << sHTML;
                                                  sHTML.clear();
                                                  sHTML.append("<html><body>");
                                                  i = 0;
                                             }
                                         else
                                             {
        //Если таблица была закрыта на данной странице
                                                  if(iptc == 2)
                                                      {
                                                           sHTML.append("</body></html>");
                                                           ListPages << sHTML;
                                                           sHTML.clear();
                                                           sHTML.append("<html><body>");
                                                           i = 0;
                                                           iptc = 0;//Значение = таблиц нет
                                                      }
                                             }
                                     }

                             str.clear();
                         }

                     if(sHTML != "") ListPages << sHTML << "</body></html>";

                 ifile.close();

                     if(gRegimeViewing == true) return;
}
Она производит разбивку на страницы, подсчитывая условное значение количества строк располагаемых на странице, формата A4. Была ещё задумка подсчитать количество символов в строке но, пока решил остановиться на таком варианте.

Разбивая страницы, она переносит всю таблицу на одну страницу до конца, даже, если таблица физически должна быть на 2-х или более страницах.

Импортируются пока только html-страницы, созданные в OpenOffice. Это делалось для того, чтобы преподаватели, у которых имеются методички в формате MSWord(*.doc) могли наиболее быстро перегнать своё пособие в мою программу.

Как Вам эта функция и есть ли возможность всё это максимально упростить? Благодарю за ответ.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 15.8.2011, 6:31
Сообщение #6


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Теперь и счастливым обладателям дистрибутивов GNU / Linux Debian и его производным(Ubuntu), предоставилась возможность попробовать себя в качестве разработчиков ПО.

Новая версия электронной библиотеки 0.0.9. Исходник + сборки.

Серьёзным изменением в данной версии является ускорение загрузки книги из базы данных. Это связано с выгрузкой изображений из базы на диск в папку. Сначала я не знал, как сделать лучше, поэтому сделал выгрузку по одному байту, что занимало большой промежуток времени.

При использовании объекта QImage - этот процесс уcкорился на моей тестовой машине раз в 20(!). Код ниже:
              for(int i = 0; i < 75; i++)
                  if(ListImages[i].count() != 0)
                      {
                          QString sNumImage;
                          QByteArray sByteImage;
                          sByteImage.clear();
                          sNumImage.setNum(i);

                              if(i >= 0 && i <= 9) sNumImage.insert(0,"0");
                                  ifile.setFileName("/tmp/xwel/temp/Image" + sNumImage + ".png");

                          QDataStream sifile(&ifile);
                          uchar cdata;
                          ifile.open(QIODevice::WriteOnly);
                          sByteImage.append(ListImages[i]);

                              while(sByteImage.count() != 0)
                                  {
                                      cdata = uchar(sByteImage.at(0));
                                      sifile << cdata;
                                      sByteImage.remove(0, 1);
                                  }

                              if(i > SelectWordDialog->value()) SelectWordDialog->setValue(i);

                          QApplication::processEvents();

                              if(SelectWordDialog->wasCanceled()) break;

                      }

            for(int i = 0; i < CountImages; i++)
                if(ListImages[i].count() != 0)
                    {
                        QString sNumImage;
                        QByteArray sByteImage;
                        sByteImage.clear();
                        sNumImage.setNum(i);

                            if(i >= 0 && i <= 9) sNumImage.insert(0,"0");

                        sByteImage.append(ListImages[i]);
                        QImage ImageSave;
                        ImageSave.loadFromData(sByteImage);
                        ImageSave.save("/tmp/xwel/temp/Image" + sNumImage, "PNG");

                            if(i > SelectWordDialog->value()) SelectWordDialog->setValue(i);

                        QApplication::processEvents();

                            if(SelectWordDialog->wasCanceled()) break;

                    }

Для сборки deb-пакета необходимо наличие установленных пакетов: autotools-dev, dh-make, fakeroot и другие.
Команда сборки:
Цитата
user@pc:/klen-library-0.0.9-deb$ dpkg-buildpackage -rfakeroot
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 17.8.2011, 21:42
Сообщение #7


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Здравствуйте!

В этой версии я сделал одно важное изменение, на которое хотел бы обратить Ваше особое внимание. Это функция "int slotChangeFont()". Там я копирую в переменную QString sText всё выделение:
sText = TextBook->textCursor().selection().toHtml();
Потом заменяю все вхождения модификаторов шрифта: "font-family:", "font-size:", "font-style:", "font-weight:" на значения из диалога шрифта.

Так вот, там должен быть способ в одну единственную строчку. Я даже нашёл его - записал в свою амбарную тетрадь, а сейчас найти не могу sad.gif . Прошу помочь в нахождении этого способа, так как "int slotChangeFont()" занимает не много не мало, аж 195 строк(!). Буду благодарен.

Исходный код + сборки.

Что изменилось:
Цитата
- Добавлены значки "Oxygen", распространяемые по лицензии GNU GPL;
- При редактировании шрифта, он изменяется теперь, не для всего текста страницы, а по частям(словам, предложениям, абзацами и т. д.);
- Улучшен импорт HTML-документов, включая копирование изображений, которые преобразуются в формат PNG;
- Изменена стандартная панель - добавлены несколько новых действий;
- Добавлена панель редактирования, в которой продублированы действия из меню "Редактирование", для более удобной работы.
- Счётчик страниц переместился со стандартной панели в строку состояния.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 18.8.2011, 21:08
Сообщение #8


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Здравствуйте! Вот новая версия 0.1.1 - исходный код + сборки.

Внешний вид:


Одно из добавлений - это функция поиска текста по книге:
           while(iNumberPage < ListPages.count())
               {
                 lFind:
                    if(TextBook->find(sFindText) == false)
                        {
                            slotNextPage(); //iNumberPage++

                                if(iNumberPage == ListPages.count() - 1) break;

                            goto lFind;
                        }
                    else break;
               }

QString sFindText - искомый текст;
int iNumberPage - текущая страница
QStringList ListPages - список страниц с текстом.

Для поиска используется "TextBook->find(sFindText)" - будет ли выгода, если использовать функцию поиска в QString непосредственно или использовать контейнер для ускорения? Предполагаю, что сейчас тратится дополнительное время для прорисовки QTextEdit TextBook.

Все новинки:
Цитата
- Добавлена возможность импорта / экспорта простого кодированного текста в формате utf-8;
- Добавлена функция поиска слов по книге;
- Добавлена возможность расширенного редактирования вставленного простого текста;
- Исправлена ошибка, при ответе "Отмена", вопроса о сохранении, при выходе;
- Добавлена панель поиска, с дубликатами команд из меню "Поиск";
- Исправлено несколько ошибок с кнопками панелей;
- Исправлены несколько ошибок при импорте файлов.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 27.8.2011, 20:49
Сообщение #9


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Цитата(xwicked @ 17.8.2011, 22:42) *
...
Так вот, там должен быть способ в одну единственную строчку. Я даже нашёл его ...
Вот он:
         QFont TextFont;
         TextFont = QFontDialog::getFont(&Accept, TextFont);
         QTextCharFormat tcf;
         tcf.setFont(TextFont);
         TextBook->textCursor().setCharFormat(tcf);
--------------------------
В версии 0.1.2, была реализована возможность добавления в базу данных библиотеки произвольной информации, в виде ссылок на отдельные файлы. Что представляет собой достаточно хорошее улучшение, так как преподавателю можно будет совершать меньше движений, при создании учебника.

Эта возможность позволит не только сократить время создания учебника, но и послужит отличным расширением возможностей программы, при использовании материалов, содержащихся в отдельных ссылках на файлы, как дополнение к создаваемому учебнику. Например, к книге о художнике, можно будет хранить галерею с изображениями(его работами) в разделе "Фотографии".

Все изменения:
Цитата
- Появилась возможность добавления дополнительной информации произвольного содержания, в виде ссылок на отдельные файлы;
- Добавлена возможность выбора пользователем варианта сохранения книги с паролем или без него;
- Добавлена защита режима редактирования программы отдельным паролем, с возможностью его будущего изменения;
- Добавлена возможность сохранения состояния показа инструментальных панелей после выхода из программы;
- Исправлена ошибка отсутствия изображений, при импорте книги из формата XWB;
- Исправлена ошибка неэкспортирования в форматы XWB, HTML, импортированной книги;
- Исправлена ошибка отсутствия расширения .txt, когда оно явно не указано, при экспорте в текстовый формат.
Исходник + сборки. Дабы быть уверенным в том, что файлы не побились, при закачке, я добавил текстовик с md5-контрольными суммами.
Git-репозиторий:
Цитата
git://klen-library.git.sourceforge.net/gitroot/klen-library/klen-library (read-only)
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
xwicked
  опции профиля:
сообщение 16.10.2011, 12:14
Сообщение #10


Студент
*

Группа: Участник
Сообщений: 66
Регистрация: 2.8.2011
Из: Старый Оскол
Пользователь №: 2781

Спасибо сказали: 0 раз(а)




Репутация:   0  


Здравствуйте! Версия 0.1.3 содержит добавление защиты документов дополнительной информации от случайного изменения в режиме просмотра:
              connect(ListAdditionallyWidget, SIGNAL(doubleClicked(QModelIndex)), SLOT(slotRunFile()));
      
          void ListAdditionally::slotRunFile()
          {
              if(*sItem == tr("Documents") && gRegime == true)
                  {
                      QString sTmp;
                      sTmp = ListAdditionallyWidget->item(ListAdditionallyWidget->currentRow())->text();
                      int ipos = sTmp.lastIndexOf("/") + 1;
                      sTmp.remove(0, ipos);
                      QFile ifile(ListAdditionallyWidget->item(ListAdditionallyWidget->currentRow())->text());
                      QDataStream idata(&ifile);
                      QByteArray bfile;
                      ifile.open(QIODevice::ReadOnly);
                      bfile = ifile.readAll();
                      QFile ofile("/tmp/xwel/tempfile/" + sTmp);
                      QDataStream odata(&ofile);
                      ofile.open(QIODevice::WriteOnly);
                      ofile.write(bfile);
                      ifile.close();
                      ofile.close();
                      QDesktopServices::openUrl(QUrl::fromLocalFile("/tmp/xwel/tempfile/" + sTmp));//Открытие копии документа
                  }
               else
      //Открытие оригинала документа
               QDesktopServices::openUrl(QUrl::fromLocalFile(ListAdditionallyWidget->item(ListAdditionallyWidget->currentRow())->text()));
          }
То бишь, происходит простое копирование во временный файл документа и его запуск. Но, есть одно небольшое ограничение, при использовании такого подхода(bfile = ifile.readAll();) - файл читается в переменную полностью и он не должен быть слишком большим. А все изменения выглядят вот так:
Цитата
- Исправлена ошибка возможной потери пароля, при экспорте;
- Добавлена защита документов дополнительной информации в режиме просмотра от случайного изменения;
- Появилась функция экспорта всех книг;
- Исправлена ошибка неполного показа длинных названий книг или издательств в списке книг;
- В окне "О программе" все ссылки, для удобного поиска, изменили цвет на синий;
- Исправлена ошибка возможности редактирования списка книг в режиме просмотра;
- В таблицу списка книг добавлены пункты: "Категория", "Подкатегория";
- Исправлена ошибка неудаления книги из базы, при удалении её из списка книг;
- В меню "Помощь" добавились пункты: "Пожертвовать" и "Получить исходный код".
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение

2 страниц V   1 2 >
Быстрый ответОтветить в данную темуНачать новую тему
Теги
Нет тегов для показа


1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0




RSS Текстовая версия Сейчас: 29.3.2024, 2:38