/**************************************************************** * This file is distributed under the following license: * * Copyright (c) 2010, RetroShare Team * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. ****************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include "Emoticons.h" #include "util/HandleRichText.h" static QHash, QHash > > Smileys; static QVector grpOrdered; void Emoticons::load() { QString sm_AllLines; bool internalFiles = true; // First try external emoticons QFile sm_File(QApplication::applicationDirPath() + "/emoticons/emotes.acs"); if(sm_File.open(QIODevice::ReadOnly)) internalFiles = false; else { // Then embedded emotions sm_File.setFileName(":/emojione/emotes.acs"); if(!sm_File.open(QIODevice::ReadOnly)) { std::cout << "error opening ressource file" << std::endl; return; } } sm_AllLines = sm_File.readAll(); sm_File.close(); sm_AllLines.remove("\n"); sm_AllLines.remove("\r"); int i = 0 ; QString smGroup; QString smCode; QString smFile; while(i < sm_AllLines.length() && sm_AllLines[i] != '{') ++i ;//Ignore text before { while (i < sm_AllLines.length()-2) { // Type of lines: // "Group"|":code:":"emojione/file.png"; smGroup = ""; smCode = ""; smFile = ""; //Reading Groupe (First into "") while(sm_AllLines[i] != '\"') ++i; //Ignore text outside "" ++i; //'"' while (sm_AllLines[i] != '\"') smGroup += sm_AllLines[i++]; //Get group char by char ++i; //'"' if (sm_AllLines[i] == '|') { //File in new format with group ++i; //'|' //Reading Code (Second into "") while(sm_AllLines[i] != '\"') ++i; //Ignore text outside "" ++i; //'"' while (sm_AllLines[i] != '\"') smCode += sm_AllLines[i++]; //Get code char by char ++i; //'"' } else { //Old file without group ++i; //':' smCode = smGroup; smGroup = "NULL"; } //Reading File (Third into "") while(sm_AllLines[i] != '\"') ++i; //Ignore text outside "" ++i; //'"' while(sm_AllLines[i] != '\"' && sm_AllLines[i+1] != ';') smFile += sm_AllLines[i++]; //Get file char by char ++i; //'"' ++i; //';' if(!smGroup.isEmpty() && !smCode.isEmpty() && !smFile.isEmpty()) { while (smCode.right(1) == "|") smCode.remove(smCode.length() - 1, 1); if (internalFiles) { smFile = ":/" + smFile; if (smGroup.right(4).toLower() == ".png") smGroup = ":/" + smGroup; } QVector ordered; if (Smileys.contains(smGroup)) ordered = Smileys[smGroup].first; ordered.append(smCode); Smileys[smGroup].first = ordered; Smileys[smGroup].second.insert(smCode, smFile); if (!grpOrdered.contains(smGroup)) grpOrdered.append(smGroup); } } // init embedder RsHtml::initEmoticons(Smileys); } void Emoticons::showSmileyWidget(QWidget *parent, QWidget *button, const char *slotAddMethod, bool above) { QWidget *smWidget = new QWidget(parent, Qt::Popup) ; smWidget->setAttribute(Qt::WA_DeleteOnClose) ; smWidget->setWindowTitle("Emoticons") ; //QTabWidget::setTabBarAutoHide(true) is from QT5.4, no way to hide TabBar before. bool bOnlyOneGroup = (Smileys.count() == 1); QTabWidget *smTab = NULL; if (! bOnlyOneGroup) { smTab = new QTabWidget(smWidget); QGridLayout *smGLayout = new QGridLayout(smWidget); smGLayout->setContentsMargins(0,0,0,0); smGLayout->addWidget(smTab); } const int buttonWidth = QFontMetricsF(smWidget->font()).height()*1.2; const int buttonHeight = QFontMetricsF(smWidget->font()).height()*1.2; int maxRowCount = 0; int maxCountPerLine = 0; QVectorIterator grp(grpOrdered); while(grp.hasNext()) { QString groupName = grp.next(); QHash group = Smileys.value(groupName).second; QWidget *tabGrpWidget = NULL; if (! bOnlyOneGroup) { tabGrpWidget = new QWidget(smTab); // (Cyril) Never use an absolute size. It needs to be scaled to the actual font size on the screen. // QFontMetricsF fm(parent->font()) ; smTab->setIconSize(QSize(32*fm.height()/14.0,32*fm.height()/14.0)); if (groupName.right(4).toLower() == ".png") smTab->addTab( tabGrpWidget, QIcon(groupName), ""); else smTab->addTab( tabGrpWidget, groupName); } else { tabGrpWidget = smWidget; } QGridLayout *tabGLayout = new QGridLayout(tabGrpWidget); tabGLayout->setContentsMargins(0,0,0,0); tabGLayout->setSpacing(0); int rowCount = (int)sqrt((double)group.size()); int countPerLine = (group.size()/rowCount) + ((group.size() % rowCount) ? 1 : 0); maxRowCount = qMax(maxRowCount, rowCount); maxCountPerLine = qMax(maxCountPerLine, countPerLine); int lin = 0; int col = 0; QVector ordered = Smileys.value(groupName).first; QVectorIterator it(ordered); while(it.hasNext()) { QString key = it.next(); QPushButton *button = new QPushButton("", tabGrpWidget); button->setIconSize(QSize(buttonWidth, buttonHeight)); button->setFixedSize(QSize(buttonWidth, buttonHeight)); button->setIcon(QPixmap(group.value(key))); button->setToolTip(key); button->setStyleSheet("QPushButton:hover {border: 3px solid white; border-radius: 2px;}"); button->setFlat(true); tabGLayout->addWidget(button,col,lin); ++lin; if(lin >= countPerLine) { lin = 0; ++col; } QObject::connect(button, SIGNAL(clicked()), parent, slotAddMethod); QObject::connect(button, SIGNAL(clicked()), smWidget, SLOT(close())); } } //Get left up pos of button QPoint butTopLeft = button->mapToGlobal(QPoint(0,0)); //Get widget's size QSize sizeWidget = smWidget->sizeHint(); //Get screen's size QSize sizeScreen = QApplication::desktop()->size(); //Calculate left distance to screen start int distToScreenLeft = butTopLeft.x(); //Calculate right distance to screen end int distToRightScreen = sizeScreen.width() - (butTopLeft.x() + button->width()); //Calculate left position int x; if (distToScreenLeft >= distToRightScreen) //More distance in left than right in screen x = butTopLeft.x() - sizeWidget.width(); //Place widget on left of button else x = butTopLeft.x() + button->width(); //Place widget on right of button //Calculate top position int y; if (above) //Widget must be above the button y = butTopLeft.y() + button->height() - sizeWidget.height(); else y = butTopLeft.y() + button->height()/2 - sizeWidget.height()/2; //Centered on button height if (y + sizeWidget.height() > sizeScreen.height()) //Widget will be too low y = sizeScreen.height() - sizeWidget.height(); //Place widget bottom at screen bottom if (y < 0) //Widget will be too high y = 0; //Place widget top at screen top smWidget->move(x, y) ; smWidget->show() ; }