/***************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2008 Cédric Borgese You can Freely distribute this program under the GNU General Public License. See the file "COPYING" for the exact licensing terms. ******************************************************************/ #include "wobblywindows.h" #include "wobblywindows_constants.h" #include #include #include #define USE_ASSERT #ifdef USE_ASSERT #define ASSERT1 assert #else #define ASSERT1 #endif //#define COMPUTE_STATS // if you enable it and run kwin in a terminal from the session it manages, // be sure to redirect the output of kwin in a file or // you'll propably get deadlocks. //#define VERBOSE_MODE #if defined COMPUTE_STATS && !defined VERBOSE_MODE # ifdef __GNUC__ # warning "You enable COMPUTE_STATS without VERBOSE_MODE, computed stats will not be printed." # endif #endif namespace KWin { struct ParameterSet { qreal stiffness; qreal drag; qreal move_factor; qreal xTesselation; qreal yTesselation; WobblyWindowsEffect::GridFilter velocityFilter; WobblyWindowsEffect::GridFilter accelerationFilter; qreal minVelocity; qreal maxVelocity; qreal stopVelocity; qreal minAcceleration; qreal maxAcceleration; qreal stopAcceleration; bool moveEffectEnabled; bool openEffectEnabled; bool closeEffectEnabled; }; ParameterSet set_0 = { 0.1, 0.8, 0.1, 20.0, 20.0, WobblyWindowsEffect::FourRingLinearMean, WobblyWindowsEffect::HeightRingLinearMean, 0.0, 1000.0, 1.0, 0.0, 1000.0, 2.0, true, false, false }; ParameterSet set_1 = { 0.15, 0.85, 0.1, 20.0, 20.0, WobblyWindowsEffect::HeightRingLinearMean, WobblyWindowsEffect::MeanWithMean, 0.0, 1000.0, 1.0, 0.0, 1000.0, 2.0, true, false, false }; ParameterSet set_2 = { 0.06, 0.9, 0.1, 20.0, 20.0, WobblyWindowsEffect::HeightRingLinearMean, WobblyWindowsEffect::NoFilter, 0.0, 1000.0, 1.0, 0.0, 1000.0, 2.0, true, false, false }; ParameterSet set_3 = { 0.03, 0.92, 0.1, 20.0, 20.0, WobblyWindowsEffect::HeightRingLinearMean, WobblyWindowsEffect::HeightRingLinearMean, 0.0, 1000.0, 1.0, 0.0, 1000.0, 2.0, true, false, false }; ParameterSet set_4 = { 0.03, 0.92, 0.1, 20.0, 20.0, WobblyWindowsEffect::HeightRingLinearMean, WobblyWindowsEffect::HeightRingLinearMean, 0.0, 1000.0, 1.0, 0.0, 1000.0, 2.0, true, false, false }; ParameterSet pset[5] = { set_0, set_1, set_2, set_3, set_4 }; KWIN_EFFECT(wobblywindows, WobblyWindowsEffect) WobblyWindowsEffect::WobblyWindowsEffect() { KConfigGroup conf = effects->effectConfig("Wobbly"); QString settingsMode = conf.readEntry("Settings", "Auto"); if (settingsMode != "Custom") { unsigned int wobblynessLevel = conf.readEntry("WobblynessLevel", 2); if (wobblynessLevel > 4) { kDebug() << "Wrong value for \"WobblynessLevel\" : " << wobblynessLevel; wobblynessLevel = 4; } setParameterSet(pset[wobblynessLevel]); } else // Custom method, read all values from config file. { m_stiffness = conf.readEntry("Stiffness", STIFFNESS); m_drag = conf.readEntry("Drag", DRAG); m_move_factor = conf.readEntry("MoveFactor", MOVEFACTOR); m_xTesselation = conf.readEntry("XTesselation", XTESSELATION); m_yTesselation = conf.readEntry("YTesselation", YTESSELATION); m_minVelocity = conf.readEntry("MinVelocity", MINVELOCITY); m_maxVelocity = conf.readEntry("MaxVelocity", MAXVELOCITY); m_stopVelocity = conf.readEntry("StopVelocity", STOPVELOCITY); m_minAcceleration = conf.readEntry("MinAcceleration", MINACCELERATION); m_maxAcceleration = conf.readEntry("MaxAcceleration", MAXACCELERATION); m_stopAcceleration = conf.readEntry("StopAcceleration", STOPACCELERATION); QString velFilter = conf.readEntry("VelocityFilter", VELOCITYFILTER); if (velFilter == "NoFilter") { m_velocityFilter = NoFilter; } else if (velFilter == "FourRingLinearMean") { m_velocityFilter = FourRingLinearMean; } else if (velFilter == "HeightRingLinearMean") { m_velocityFilter = HeightRingLinearMean; } else if (velFilter == "MeanWithMean") { m_velocityFilter = MeanWithMean; } else if (velFilter == "MeanWithMedian") { m_velocityFilter = MeanWithMedian; } else { m_velocityFilter = FourRingLinearMean; kDebug() << "Unknown config value for VelocityFilter : " << velFilter; } QString accFilter = conf.readEntry("AccelerationFilter", ACCELERATIONFILTER); if (accFilter == "NoFilter") { m_accelerationFilter = NoFilter; } else if (accFilter == "FourRingLinearMean") { m_accelerationFilter = FourRingLinearMean; } else if (accFilter == "HeightRingLinearMean") { m_accelerationFilter = HeightRingLinearMean; } else if (accFilter == "MeanWithMean") { m_accelerationFilter = MeanWithMean; } else if (accFilter == "MeanWithMedian") { m_accelerationFilter = MeanWithMedian; } else { m_accelerationFilter = NoFilter; kDebug() << "Unknown config value for accelerationFilter : " << accFilter; } m_moveEffectEnabled = conf.readEntry("MoveEffect", true); m_openEffectEnabled = conf.readEntry("OpenEffect", false); // disable close effect by default for now as it doesn't do what I want. m_closeEffectEnabled = conf.readEntry("CloseEffect", false); } #if defined VERBOSE_MODE kDebug() << "Parameters :\n" << "move : " << m_moveEffectEnabled << ", open : " << m_openEffectEnabled << ", close : " << m_closeEffectEnabled << "\n" "grid(" << m_stiffness << ", " << m_drag << ", " << m_move_factor << ")\n" << "velocity(" << m_minVelocity << ", " << m_maxVelocity << ", " << m_stopVelocity << ")\n" << "acceleration(" << m_minAcceleration << ", " << m_maxAcceleration << ", " << m_stopAcceleration << ")\n" << "tesselation(" << m_xTesselation << ", " << m_yTesselation << ")"; #endif } WobblyWindowsEffect::~WobblyWindowsEffect() { if (!windows.empty()) { // we should be empty at this point... // emit a warning and clean the list. kDebug() << "Windows list not empty. Left items : " << windows.count(); QHash< const EffectWindow*, WindowWobblyInfos >::iterator i; for (i = windows.begin(); i != windows.end(); ++i) { freeWobblyInfo(i.value()); } } } void WobblyWindowsEffect::setParameterSet(ParameterSet& pset) { m_stiffness = pset.stiffness; m_drag = pset.drag; m_move_factor = pset.move_factor; m_xTesselation = pset.xTesselation; m_yTesselation = pset.yTesselation; m_velocityFilter = pset.velocityFilter ; m_accelerationFilter = pset.accelerationFilter; m_minVelocity = pset.minVelocity; m_maxVelocity = pset.maxVelocity; m_stopVelocity = pset.stopVelocity; m_minAcceleration = pset.minAcceleration; m_maxAcceleration = pset.maxAcceleration; m_stopAcceleration = pset.stopAcceleration; m_moveEffectEnabled = pset.moveEffectEnabled; m_openEffectEnabled = pset.openEffectEnabled; m_closeEffectEnabled = pset.closeEffectEnabled; } void WobblyWindowsEffect::setVelocityThreshold(qreal m_minVelocity) { this->m_minVelocity = m_minVelocity; } void WobblyWindowsEffect::setMoveFactor(qreal factor) { m_move_factor = factor; } void WobblyWindowsEffect::setStiffness(qreal stiffness) { m_stiffness = stiffness; } void WobblyWindowsEffect::setVelocityFilter(GridFilter filter) { m_velocityFilter = filter; } void WobblyWindowsEffect::setAccelerationFilter(GridFilter filter) { m_accelerationFilter = filter; } WobblyWindowsEffect::GridFilter WobblyWindowsEffect::velocityFilter() const { return m_velocityFilter; } WobblyWindowsEffect::GridFilter WobblyWindowsEffect::accelerationFilter() const { return m_accelerationFilter; } void WobblyWindowsEffect::setDrag(qreal drag) { m_drag = drag; } void WobblyWindowsEffect::prePaintScreen(ScreenPrePaintData& data, int time) { // We need to mark the screen windows as transformed. Otherwise the whole // screen won't be repainted, resulting in artefacts. // Could we just set a subset of the screen to be repainted ? if (windows.count() != 0) { data.mask |= PAINT_SCREEN_WITH_TRANSFORMED_WINDOWS; } // this set the QRect invalid. m_updateRegion.setWidth(0); effects->prePaintScreen(data, time); } const qreal maxTime = 10.0; void WobblyWindowsEffect::prePaintWindow(EffectWindow* w, WindowPrePaintData& data, int time) { if (windows.contains(w)) { data.setTransformed(); data.quads = data.quads.makeRegularGrid(m_xTesselation, m_yTesselation); bool stop = false; qreal updateTime = time; while (!stop && (updateTime > maxTime)) { #if defined VERBOSE_MODE kDebug() << "loop time " << updateTime << " / " << time; #endif stop = !updateWindowWobblyDatas(w, maxTime); updateTime -= maxTime; } if (!stop && updateTime > 0) { updateWindowWobblyDatas(w, updateTime); } } effects->prePaintWindow(w, data, time); } void WobblyWindowsEffect::paintWindow(EffectWindow* w, int mask, QRegion region, WindowPaintData& data) { if(windows.contains(w)) { WindowWobblyInfos& wwi = windows[w]; int tx = w->geometry().x(); int ty = w->geometry().y(); for (int i = 0; i < data.quads.count(); ++i) { for(int j = 0; j < 4; ++j) { WindowVertex& v = data.quads[i][j]; Pair oldPos = {tx + v.x(), ty + v.y()}; Pair newPos = computeBezierPoint(wwi, oldPos); v.move(newPos.x - tx, newPos.y - ty); } } } // Call the next effect. effects->paintWindow(w, mask, region, data); } void WobblyWindowsEffect::postPaintScreen() { if (!windows.isEmpty()) { effects->addRepaint(m_updateRegion); } // Call the next effect. effects->postPaintScreen(); } void WobblyWindowsEffect::windowUserMovedResized(EffectWindow* w, bool first, bool last) { if (m_moveEffectEnabled && first && !w->isSpecialWindow()) { if (!windows.contains(w)) { WindowWobblyInfos new_wwi; initWobblyInfo(new_wwi, w->geometry()); windows[w] = new_wwi; } WindowWobblyInfos& wwi = windows[w]; wwi.status = Moving; const QRectF& rect = w->geometry(); qreal x_increment = rect.width() / (wwi.width-1.0); qreal y_increment = rect.height() / (wwi.height-1.0); Pair picked = {cursorPos().x(), cursorPos().y()}; int indx = (picked.x - rect.x()) / x_increment; int indy = (picked.y - rect.y()) / y_increment; int pickedPointIndex = indy*wwi.width + indx; if (pickedPointIndex < 0) { kDebug() << "Picked index == " << pickedPointIndex << " with (" << cursorPos().x() << "," << cursorPos().y() << ")"; pickedPointIndex = 0; } else if (static_cast(pickedPointIndex) > wwi.count - 1) { kDebug() << "Picked index == " << pickedPointIndex << " with (" << cursorPos().x() << "," << cursorPos().y() << ")"; pickedPointIndex = wwi.count - 1; } #if defined VERBOSE_MODE kDebug() << "Original Picked point -- x : " << picked.x << " - y : " << picked.y; #endif wwi.constraint[pickedPointIndex] = true; } else if (m_moveEffectEnabled && last) { if (windows.contains(w)) { WindowWobblyInfos& wwi = windows[w]; wwi.status = Free; } } } void WobblyWindowsEffect::windowAdded(EffectWindow* w) { if (m_openEffectEnabled) { if(windows.contains(w)) { // could this happen ?? WindowWobblyInfos& wwi = windows[w]; wobblyOpenInit(wwi); } else { WindowWobblyInfos new_wwi; initWobblyInfo(new_wwi, w->geometry()); wobblyOpenInit(new_wwi); windows[w] = new_wwi; } } } void WobblyWindowsEffect::windowClosed(EffectWindow* w) { if(windows.contains(w)) { WindowWobblyInfos& wwi = windows[w]; if (m_closeEffectEnabled) { wobblyCloseInit(wwi, w); w->refWindow(); } else { freeWobblyInfo(wwi); windows.remove(w); } } else if (m_closeEffectEnabled) { WindowWobblyInfos new_wwi; initWobblyInfo(new_wwi, w->geometry()); wobblyCloseInit(new_wwi, w); windows[w] = new_wwi; w->refWindow(); } } void WobblyWindowsEffect::wobblyOpenInit(WindowWobblyInfos& wwi) const { Pair middle = { (wwi.origin[0].x + wwi.origin[15].x)/2, (wwi.origin[0].y + wwi.origin[15].y)/2 }; for (unsigned int j=0; j<4; ++j) { for (unsigned int i=0; i<4; ++i) { unsigned int idx = j*4 + i; wwi.constraint[idx] = false; wwi.position[idx].x = (wwi.position[idx].x + 3*middle.x)/4; wwi.position[idx].y = (wwi.position[idx].y + 3*middle.y)/4; } } wwi.status = Openning; } void WobblyWindowsEffect::wobblyCloseInit(WindowWobblyInfos& wwi, EffectWindow* w) const { const QRectF& rect = w->geometry(); QPointF center = rect.center(); int x1 = (rect.x() + 3*center.x())/4; int x2 = (rect.x() + rect.width() + 3*center.x())/4; int y1 = (rect.y() + 3*center.y())/4; int y2 = (rect.y() + rect.height() + 3*center.y())/4; wwi.closeRect.setCoords(x1, y1, x2, y2); // for closing, not yet used... for (unsigned int j=0; j<4; ++j) { for (unsigned int i=0; i<4; ++i) { unsigned int idx = j*4 + i; wwi.constraint[idx] = false; } } wwi.status = Closing; } void WobblyWindowsEffect::initWobblyInfo(WindowWobblyInfos& wwi, QRect geometry) const { wwi.count = 4*4; wwi.width = 4; wwi.height = 4; wwi.bezierWidth = m_xTesselation; wwi.bezierHeight = m_yTesselation; wwi.bezierCount = m_xTesselation * m_yTesselation; wwi.origin = new Pair[wwi.count]; wwi.position = new Pair[wwi.count]; wwi.velocity = new Pair[wwi.count]; wwi.acceleration = new Pair[wwi.count]; wwi.buffer = new Pair[wwi.count]; wwi.constraint = new bool[wwi.count]; wwi.bezierSurface = new Pair[wwi.bezierCount]; wwi.status = Moving; qreal x = geometry.x(), y = geometry.y(); qreal width = geometry.width(), height = geometry.height(); Pair initValue = {x, y}; static const Pair nullPair = {0.0, 0.0}; qreal x_increment = width / (wwi.width-1.0); qreal y_increment = height / (wwi.height-1.0); for (unsigned int j=0; j<4; ++j) { for (unsigned int i=0; i<4; ++i) { unsigned int idx = j*4 + i; wwi.origin[idx] = initValue; wwi.position[idx] = initValue; wwi.velocity[idx] = nullPair; wwi.constraint[idx] = false; if (i != 4-2) // x grid count - 2, i.e. not the last point { initValue.x += x_increment; } else { initValue.x = width + x; } initValue.x = initValue.x; } initValue.x = x; initValue.x = initValue.x; if (j != 4-2) // y grid count - 2, i.e. not the last point { initValue.y += y_increment; } else { initValue.y = height + y; } initValue.y = initValue.y; } } void WobblyWindowsEffect::freeWobblyInfo(WindowWobblyInfos& wwi) const { delete wwi.origin; delete wwi.position; delete wwi.velocity; delete wwi.acceleration; delete wwi.buffer; delete wwi.constraint; delete wwi.bezierSurface; } WobblyWindowsEffect::Pair WobblyWindowsEffect::computeBezierPoint(const WindowWobblyInfos& wwi, Pair point) const { // compute the input value Pair topleft = wwi.origin[0]; Pair bottomright = wwi.origin[wwi.count-1]; // ASSERT1(point.x >= topleft.x); // ASSERT1(point.y >= topleft.y); // ASSERT1(point.x <= bottomright.x); // ASSERT1(point.y <= bottomright.y); qreal tx = (point.x - topleft.x) / (bottomright.x - topleft.x); qreal ty = (point.y - topleft.y) / (bottomright.y - topleft.y); // ASSERT1(tx >= 0); // ASSERT1(tx <= 1); // ASSERT1(ty >= 0); // ASSERT1(ty <= 1); // compute polinomial coeff qreal px[4]; px[0] = (1-tx)*(1-tx)*(1-tx); px[1] = 3*(1-tx)*(1-tx)*tx; px[2] = 3*(1-tx)*tx*tx; px[3] = tx*tx*tx; qreal py[4]; py[0] = (1-ty)*(1-ty)*(1-ty); py[1] = 3*(1-ty)*(1-ty)*ty; py[2] = 3*(1-ty)*ty*ty; py[3] = ty*ty*ty; Pair res = {0.0, 0.0}; for (unsigned int j = 0; j < 4; ++j) { for (unsigned int i = 0; i < 4; ++i) { // this assume the grid is 4*4 res.x += px[i] * py[j] * wwi.position[i + j * wwi.width].x; res.y += px[i] * py[j] * wwi.position[i + j * wwi.width].y; } } return res; } namespace { static inline void fixVectorBounds(WobblyWindowsEffect::Pair& vec, qreal min, qreal max) { if (fabs(vec.x) < min) { vec.x = 0.0; } else if (fabs(vec.x) > max) { if (vec.x > 0.0) { vec.x = max; } else { vec.x = -max; } } if (fabs(vec.y) < min) { vec.y = 0.0; } else if (fabs(vec.y) > max) { if (vec.y > 0.0) { vec.y = max; } else { vec.y = -max; } } } static inline void computeVectorBounds(WobblyWindowsEffect::Pair& vec, WobblyWindowsEffect::Pair& bound) { if (fabs(vec.x) < bound.x) { bound.x = fabs(vec.x); } else if (fabs(vec.x) > bound.y) { bound.y = fabs(vec.x); } if (fabs(vec.y) < bound.x) { bound.x = fabs(vec.y); } else if (fabs(vec.y) > bound.y) { bound.y = fabs(vec.y); } } } // close the anonymous namespace bool WobblyWindowsEffect::updateWindowWobblyDatas(EffectWindow* w, qreal time) { QRectF rect = w->geometry(); WindowWobblyInfos& wwi = windows[w]; if (wwi.status == Closing) { rect = wwi.closeRect; } qreal x_length = rect.width() / (wwi.width-1.0); qreal y_length = rect.height() / (wwi.height-1.0); #if defined VERBOSE_MODE kDebug() << "time " << time; kDebug() << "increment x " << x_length << " // y" << y_length; #endif Pair origine = {rect.x(), rect.y()}; for (unsigned int j=0; j bottomRightCorner.x) { bottomRightCorner.x = pos.x; } if (pos.y < topLeftCorner.y) { topLeftCorner.y = pos.y; } if (pos.y > bottomRightCorner.y) { bottomRightCorner.y = pos.y; } vel_sum += fabs(vel.x) + fabs(vel.y); #if defined VERBOSE_MODE if (wwi.constraint[i]) { kDebug() << "Constraint point ** vel : " << vel.x << "," << vel.y << " ** move : " << vel.x*time << "," << vel.y*time; } #endif } #if defined VERBOSE_MODE # if defined COMPUTE_STATS kDebug() << "Acceleration bounds (" << accBound.x << ", " << accBound.y << ")"; kDebug() << "Velocity bounds (" << velBound.x << ", " << velBound.y << ")"; # endif kDebug() << "sum_acc : " << acc_sum << " *** sum_vel :" << vel_sum; #endif if (wwi.status != Moving && acc_sum < m_stopAcceleration && vel_sum < m_stopVelocity) { freeWobblyInfo(wwi); windows.remove(w); if (wwi.status == Closing) { w->unrefWindow(); } return false; } QRect windowRect(topLeftCorner.x, topLeftCorner.y, bottomRightCorner.x - topLeftCorner.x, bottomRightCorner.y - topLeftCorner.y); if (m_updateRegion.isValid()) { m_updateRegion = m_updateRegion.united(windowRect); } else { m_updateRegion = windowRect; } return true; } void WobblyWindowsEffect::fourRingLinearMean(Pair** datas_pointer, WindowWobblyInfos& wwi) { Pair* datas = *datas_pointer; Pair neibourgs[4]; // for corners // top-left { Pair& res = wwi.buffer[0]; Pair vit = datas[0]; neibourgs[0] = datas[1]; neibourgs[1] = datas[wwi.width]; res.x = (neibourgs[0].x + neibourgs[1].x + 2.0*vit.x) / 4.0; res.y = (neibourgs[0].y + neibourgs[1].y + 2.0*vit.y) / 4.0; } // top-right { Pair& res = wwi.buffer[wwi.width-1]; Pair vit = datas[wwi.width-1]; neibourgs[0] = datas[wwi.width-2]; neibourgs[1] = datas[2*wwi.width-1]; res.x = (neibourgs[0].x + neibourgs[1].x + 2.0*vit.x) / 4.0; res.y = (neibourgs[0].y + neibourgs[1].y + 2.0*vit.y) / 4.0; } // bottom-left { Pair& res = wwi.buffer[wwi.width*(wwi.height-1)]; Pair vit = datas[wwi.width*(wwi.height-1)]; neibourgs[0] = datas[wwi.width*(wwi.height-1)+1]; neibourgs[1] = datas[wwi.width*(wwi.height-2)]; res.x = (neibourgs[0].x + neibourgs[1].x + 2.0*vit.x) / 4.0; res.y = (neibourgs[0].y + neibourgs[1].y + 2.0*vit.y) / 4.0; } // bottom-right { Pair& res = wwi.buffer[wwi.count-1]; Pair vit = datas[wwi.count-1]; neibourgs[0] = datas[wwi.count-2]; neibourgs[1] = datas[wwi.width*(wwi.height-1)-1]; res.x = (neibourgs[0].x + neibourgs[1].x + 2.0*vit.x) / 4.0; res.y = (neibourgs[0].y + neibourgs[1].y + 2.0*vit.y) / 4.0; } // for borders // top border for (unsigned int i=1; i xmax) { xmax = datas[i].x; } if (datas[i].y < ymin) { ymin = datas[i].y; } if (datas[i].y > ymax) { ymax = datas[i].y; } } Pair median = {(xmin + xmax)/2.0, (ymin + ymax)/2.0}; for (unsigned int i = 0; i < wwi.count; ++i) { wwi.buffer[i].x = (datas[i].x + median.x) / 2.0; wwi.buffer[i].y = (datas[i].y + median.y) / 2.0; } Pair* tmp = datas; *datas_pointer = wwi.buffer; wwi.buffer = tmp; } void WobblyWindowsEffect::heightRingLinearMean(Pair** datas_pointer, WindowWobblyInfos& wwi) { Pair* datas = *datas_pointer; Pair neibourgs[8]; // for corners // top-left { Pair& res = wwi.buffer[0]; Pair vit = datas[0]; neibourgs[0] = datas[1]; neibourgs[1] = datas[wwi.width]; neibourgs[2] = datas[wwi.width+1]; res.x = (neibourgs[0].x + neibourgs[1].x + neibourgs[2].x + 3.0*vit.x) / 6.0; res.y = (neibourgs[0].y + neibourgs[1].y + neibourgs[2].y + 3.0*vit.y) / 6.0; } // top-right { Pair& res = wwi.buffer[wwi.width-1]; Pair vit = datas[wwi.width-1]; neibourgs[0] = datas[wwi.width-2]; neibourgs[1] = datas[2*wwi.width-1]; neibourgs[2] = datas[2*wwi.width-2]; res.x = (neibourgs[0].x + neibourgs[1].x + neibourgs[2].x + 3.0*vit.x) / 6.0; res.y = (neibourgs[0].y + neibourgs[1].y + neibourgs[2].y + 3.0*vit.y) / 6.0; } // bottom-left { Pair& res = wwi.buffer[wwi.width*(wwi.height-1)]; Pair vit = datas[wwi.width*(wwi.height-1)]; neibourgs[0] = datas[wwi.width*(wwi.height-1)+1]; neibourgs[1] = datas[wwi.width*(wwi.height-2)]; neibourgs[2] = datas[wwi.width*(wwi.height-2)+1]; res.x = (neibourgs[0].x + neibourgs[1].x + neibourgs[2].x + 3.0*vit.x) / 6.0; res.y = (neibourgs[0].y + neibourgs[1].y + neibourgs[2].y + 3.0*vit.y) / 6.0; } // bottom-right { Pair& res = wwi.buffer[wwi.count-1]; Pair vit = datas[wwi.count-1]; neibourgs[0] = datas[wwi.count-2]; neibourgs[1] = datas[wwi.width*(wwi.height-1)-1]; neibourgs[2] = datas[wwi.width*(wwi.height-1)-2]; res.x = (neibourgs[0].x + neibourgs[1].x + neibourgs[2].x + 3.0*vit.x) / 6.0; res.y = (neibourgs[0].y + neibourgs[1].y + neibourgs[2].y + 3.0*vit.y) / 6.0; } // for borders // top border for (unsigned int i=1; i