//=========================================== // Lumina-DE source code // Copyright (c) 2017-2018, Ken Moore // Available under the 3-clause BSD license // See the LICENSE file for full details //=========================================== #include "NativeWindowObject.h" #include #include #include // == QML Type Registration == void NativeWindowObject::RegisterType(){ static bool done = false; if(done){ return; } done=true; qmlRegisterType("Lumina.Backend.NativeWindowObject",2,0, "NativeWindowObject"); } // === PUBLIC === NativeWindowObject::NativeWindowObject(WId id) : QObject(){ winid = id; frameid = 0; dmgID = dmg = 0; geomTimer = new QTimer(this); geomTimer->setSingleShot(true); geomTimer->setInterval(50); //1/20 second connect(geomTimer, SIGNAL(timeout()), this, SLOT(sendNewGeom()) ); qRegisterMetaType("WId"); } NativeWindowObject::~NativeWindowObject(){ hash.clear(); } void NativeWindowObject::addFrameWinID(WId fid){ frameid = fid; } void NativeWindowObject::addDamageID(unsigned int dmg){ dmgID = dmg; } bool NativeWindowObject::isRelatedTo(WId tmp){ return (relatedTo.contains(tmp) || winid == tmp || frameid == tmp); } WId NativeWindowObject::id(){ return winid; } WId NativeWindowObject::frameId(){ return frameid; } unsigned int NativeWindowObject::damageId(){ return dmgID; } QVariant NativeWindowObject::property(NativeWindowObject::Property prop){ if(hash.contains(prop)){ return hash.value(prop); } else if(prop == NativeWindowObject::RelatedWindows){ return QVariant::fromValue(relatedTo); } return QVariant(); //null variant } void NativeWindowObject::setProperty(NativeWindowObject::Property prop, QVariant val, bool force){ if(prop == NativeWindowObject::RelatedWindows){ relatedTo = val.value< QList >(); } else if(prop == NativeWindowObject::None || (!force && hash.value(prop)==val)){ return; } else if(prop == NativeWindowObject::WinImage){ //special case - This should never be actually set in the property hash // it is loaded dynamically by the QMLImageProvider instead (prevent flickering/caching image) } else{ hash.insert(prop, val); } emitSinglePropChanged(prop); emit PropertiesChanged(QList() << prop, QList() << val); } void NativeWindowObject::setProperties(QList props, QList vals, bool force){ for(int i=0; i=vals.length()){ props.removeAt(i); i--; continue; } //no corresponding value for this property if(props[i] == NativeWindowObject::None || (!force && (hash.value(props[i]) == vals[i])) ){ props.removeAt(i); vals.removeAt(i); i--; continue; //Invalid property or identical value }else if(props[i] == NativeWindowObject::WinImage){ //special case - This should never be actually set in the property hash // it is loaded dynamically by the QMLImageProvider instead (prevent flickering/caching image) }else{ hash.insert(props[i], vals[i]); } emitSinglePropChanged(props[i]); } emit PropertiesChanged(props, vals); } void NativeWindowObject::requestProperty(NativeWindowObject::Property prop, QVariant val, bool force){ if(prop == NativeWindowObject::None || prop == NativeWindowObject::RelatedWindows || (!force && hash.value(prop)==val) ){ return; } emit RequestPropertiesChange(winid, QList() << prop, QList() << val); } void NativeWindowObject::requestProperties(QList props, QList vals, bool force){ //Verify/adjust inputs as needed for(int i=0; i=vals.length()){ props.removeAt(i); i--; continue; } //no corresponding value for this property if(props[i] == NativeWindowObject::None || props[i] == NativeWindowObject::RelatedWindows || (!force && hash.value(props[i])==vals[i]) ){ props.removeAt(i); vals.removeAt(i); i--; continue; } //Invalid property or identical value /*if( (props[i] == NativeWindowObject::Visible || props[i] == NativeWindowObject::Active) && frameid !=0){ //These particular properties needs to change the frame - not the window itself emit RequestPropertiesChange(frameid, QList() << props[i], QList() << vals[i]); props.removeAt(i); vals.removeAt(i); i--; }*/ } emit RequestPropertiesChange(winid, props, vals); } QRect NativeWindowObject::geometry(){ //Calculate the "full" geometry of the window + frame (if any) //Check that the size is between the min/max limitations QSize size = hash.value(NativeWindowObject::Size).toSize(); QSize min = hash.value(NativeWindowObject::MinSize).toSize(); QSize max = hash.value(NativeWindowObject::MaxSize).toSize(); if(min.isValid() && min.width() > size.width() ){ size.setWidth(min.width()); } if(min.isValid() && min.height() > size.height()){ size.setHeight(min.height()); } if(max.isValid() && max.width() < size.width() && max.width()>min.width()){ size.setWidth(max.width()); } if(max.isValid() && max.height() < size.height() && max.height()>min.height()){ size.setHeight(max.height()); } //Assemble the full geometry QRect geom( hash.value(NativeWindowObject::GlobalPos).toPoint(), size ); //Now adjust the window geom by the frame margins QList frame = hash.value(NativeWindowObject::FrameExtents).value< QList >(); //Left,Right,Top,Bottom //qDebug() << "Calculate Geometry:" << geom << frame; if(frame.length()==4){ geom = geom.adjusted( -frame[0], -frame[2], frame[1], frame[3] ); } //qDebug() << " - Total:" << geom; return geom; } void NativeWindowObject::setGeometryNow(QRect geom){ updateGeometry(geom.x(), geom.y(), geom.width(), geom.height(), true); } // QML ACCESS FUNCTIONS (shortcuts for particular properties in a format QML can use) QString NativeWindowObject::winImage(){ //Need to alternate something on the end to ensure that QML knows to fetch the new image (non-cached only) if(dmg==0){ dmg = 1; } else{ dmg = 0; } return "image://native_window/image:"+QString::number(winid)+":"+QString::number(dmg); } QString NativeWindowObject::name(){ return this->property(NativeWindowObject::Name).toString(); } QString NativeWindowObject::title(){ return this->property(NativeWindowObject::Title).toString(); } QString NativeWindowObject::shortTitle(){ QString tmp = this->property(NativeWindowObject::ShortTitle).toString(); if(tmp.isEmpty()){ tmp = title(); } if(tmp.isEmpty()){ tmp = name(); } return tmp; } QString NativeWindowObject::icon(){ if(icodmg==0){ icodmg=1; } else{ icodmg = 0; } qDebug() << "Window Icon:" << icodmg << this->property(NativeWindowObject::Icon).value().availableSizes(); return "image://native_window/icon:"+QString::number(winid)+":"+QString::number(icodmg); } //QML Button states bool NativeWindowObject::showCloseButton(){ QList types = this->property(NativeWindowObject::WinTypes).value >(); QList badtypes; badtypes << NativeWindowObject::T_DESKTOP << NativeWindowObject::T_TOOLBAR << NativeWindowObject::T_MENU \ << NativeWindowObject::T_SPLASH << NativeWindowObject::T_DROPDOWN_MENU << NativeWindowObject::T_POPUP_MENU \ << NativeWindowObject::T_NOTIFICATION << NativeWindowObject::T_COMBO << NativeWindowObject::T_DND; for(int i=0; i types = this->property(NativeWindowObject::WinTypes).value >(); QList badtypes; badtypes << NativeWindowObject::T_DESKTOP << NativeWindowObject::T_TOOLBAR << NativeWindowObject::T_MENU \ << NativeWindowObject::T_SPLASH << NativeWindowObject::T_DROPDOWN_MENU << NativeWindowObject::T_POPUP_MENU \ << NativeWindowObject::T_NOTIFICATION << NativeWindowObject::T_COMBO << NativeWindowObject::T_DND; for(int i=0; i types = this->property(NativeWindowObject::WinTypes).value >(); QList badtypes; badtypes << NativeWindowObject::T_DESKTOP << NativeWindowObject::T_TOOLBAR << NativeWindowObject::T_MENU \ << NativeWindowObject::T_SPLASH << NativeWindowObject::T_DROPDOWN_MENU << NativeWindowObject::T_POPUP_MENU \ << NativeWindowObject::T_NOTIFICATION << NativeWindowObject::T_COMBO << NativeWindowObject::T_DND << NativeWindowObject::T_DIALOG; for(int i=0; i types = this->property(NativeWindowObject::WinTypes).value >(); QList badtypes; badtypes << NativeWindowObject::T_DESKTOP << NativeWindowObject::T_TOOLBAR << NativeWindowObject::T_MENU \ << NativeWindowObject::T_SPLASH << NativeWindowObject::T_DROPDOWN_MENU << NativeWindowObject::T_POPUP_MENU \ << NativeWindowObject::T_NOTIFICATION << NativeWindowObject::T_COMBO << NativeWindowObject::T_DND; for(int i=0; i types = this->property(NativeWindowObject::WinTypes).value >(); QList badtypes; badtypes << NativeWindowObject::T_DESKTOP << NativeWindowObject::T_TOOLBAR << NativeWindowObject::T_MENU \ << NativeWindowObject::T_SPLASH << NativeWindowObject::T_DROPDOWN_MENU << NativeWindowObject::T_POPUP_MENU \ << NativeWindowObject::T_NOTIFICATION << NativeWindowObject::T_COMBO << NativeWindowObject::T_DND; for(int i=0; i types = this->property(NativeWindowObject::WinTypes).value >(); QList badtypes; badtypes << NativeWindowObject::T_DESKTOP << NativeWindowObject::T_TOOLBAR << NativeWindowObject::T_MENU \ << NativeWindowObject::T_SPLASH << NativeWindowObject::T_DROPDOWN_MENU << NativeWindowObject::T_POPUP_MENU \ << NativeWindowObject::T_NOTIFICATION << NativeWindowObject::T_COMBO << NativeWindowObject::T_DND; for(int i=0; iproperty(NativeWindowObject::Workspace).toInt()<0 || this->property(NativeWindowObject::States).value >().contains(NativeWindowObject::S_STICKY) ); } bool NativeWindowObject::isVisible(){ return (this->property(NativeWindowObject::Visible).toBool() ); } int NativeWindowObject::workspace(){ return this->property(NativeWindowObject::Workspace).toInt(); } //QML Geometry reporting QRect NativeWindowObject::frameGeometry(){ return geometry(); } QRect NativeWindowObject::imageGeometry(){ QRect geom( this->property(NativeWindowObject::GlobalPos).toPoint(), this->property(NativeWindowObject::Size).toSize() ); return geom; } void NativeWindowObject::updateGeometry(int x, int y, int width, int height, bool now){ // Full frame+window geometry - go ahead and pull it apart and only update the interior window geom QList fgeom = this->property(NativeWindowObject::FrameExtents).value >(); if(fgeom.isEmpty()){ fgeom << 0<<0<<0<<0; } //just in case (left/right/top/bottom) QPoint pos(x+fgeom[0], y+fgeom[2]); QSize sz(width-fgeom[0]-fgeom[1], height-fgeom[2]-fgeom[3]); newgeom = QRect(pos, sz); lastgeom = QRect(x,y,width,height); //save this for later if(!now){ //qDebug() << "Update Geometry:" << fgeom << QRect(x,y,width,height) << pos << sz; //requestProperties(QList() << NativeWindowObject::GlobalPos << NativeWindowObject::Size, QList() << pos << sz); if(!geomTimer->isActive()){ QTimer::singleShot(0,geomTimer, SLOT(start())); } }else{ sendNewGeom(); } } // ==== PUBLIC SLOTS === void NativeWindowObject::toggleVisibility(){ setProperty(NativeWindowObject::Visible, !property(NativeWindowObject::Visible).toBool() ); } void NativeWindowObject::toggleMaximize(){ //Find the screen containing this window (center of window) QRect curgeom = frameGeometry(); QPoint ctr = curgeom.center(); QList scrns = QApplication::screens(); QRect max; for(int i=0; igeometry().contains(ctr)){ max = scrns[i]->availableGeometry(); break; } } //Now compare the current geometry to the screen geometry qDebug() << "Maximize Toggle:" << curgeom << max; if(curgeom!=max){ qDebug() << " - maximize"; setGeometryNow(max); //will set newgeom to max lastgeom = curgeom; //save this for later }else{ qDebug() << " - restore" << lastgeom; //Already maximized, look at the old geometry and figure out how to restore it if(lastgeom.isNull() || lastgeom == max){ qDebug() << " -- Reset lastgeom to half-screen size"; //no old info available - center the window at half maximum size lastgeom = QRect(max.x()-max.width()/2, max.y()-max.height()/2, max.width()/2, max.height()/2); } setGeometryNow(lastgeom); } qDebug() << "After toggle:" << lastgeom; //emit geomChanged(); } void NativeWindowObject::requestClose(){ emit RequestClose(winid); } void NativeWindowObject::requestKill(){ emit RequestKill(winid); } void NativeWindowObject::requestPing(){ emit RequestPing(winid); } void NativeWindowObject::requestActivate(){ requestProperty(NativeWindowObject::Active, true); } void NativeWindowObject::announceClosed(){ this->emit WindowClosed(winid); } // ==== PRIVATE ==== void NativeWindowObject::emitSinglePropChanged(NativeWindowObject::Property prop){ //Simple switch to emit the QML-usable signals as properties are changed switch(prop){ case NativeWindowObject::Name: emit nameChanged(); break; case NativeWindowObject::Title: emit titleChanged(); if(this->property(NativeWindowObject::ShortTitle).toString().isEmpty()){ emit shortTitleChanged(); } break; case NativeWindowObject::ShortTitle: emit shortTitleChanged(); break; case NativeWindowObject::Icon: emit iconChanged(); break; case NativeWindowObject::Workspace: case NativeWindowObject::States: emit stickyChanged(); break; case NativeWindowObject::WinImage: emit winImageChanged(); break; case NativeWindowObject::WinTypes: emit winTypeChanged(); break; case NativeWindowObject::Visible: emit visibilityChanged(); break; default: break; //do nothing otherwise } } void NativeWindowObject::sendNewGeom(){ QList props; props << NativeWindowObject::GlobalPos << NativeWindowObject::Size; QList vals; vals << newgeom.topLeft() << newgeom.size(); requestProperties(props, vals); setProperties(props,vals); emit VerifyNewGeometry(winid); }