123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228 |
- /*
- * MapView.cpp, part of VCMI engine
- *
- * Authors: listed in file AUTHORS in main folder
- *
- * License: GNU General Public License v2.0 or later
- * Full text of license available in license.txt file, in main folder
- *
- */
- #include "StdInc.h"
- #include "MapView.h"
- #include "MapViewActions.h"
- #include "MapViewCache.h"
- #include "MapViewController.h"
- #include "MapViewModel.h"
- #include "mapHandler.h"
- #include "../CPlayerInterface.h"
- #include "../adventureMap/AdventureMapInterface.h"
- #include "../GameEngine.h"
- #include "../GameInstance.h"
- #include "../render/CAnimation.h"
- #include "../render/Canvas.h"
- #include "../render/IImage.h"
- #include "../eventsSDL/InputHandler.h"
- #include "../../lib/callback/CCallback.h"
- #include "../../lib/CConfigHandler.h"
- #include "../../lib/mapObjects/CGHeroInstance.h"
- #include "MapOverlayLogVisualizer.h"
- BasicMapView::~BasicMapView() = default;
- std::shared_ptr<MapViewModel> BasicMapView::createModel(const Point & dimensions) const
- {
- auto result = std::make_shared<MapViewModel>();
- result->setLevel(0);
- result->setTileSize(Point(32, 32));
- result->setViewCenter(Point(0, 0));
- result->setViewDimensions(dimensions);
- return result;
- }
- BasicMapView::BasicMapView(const Point & offset, const Point & dimensions)
- : model(createModel(dimensions))
- , tilesCache(new MapViewCache(model))
- , controller(new MapViewController(model, tilesCache))
- , needFullUpdate(false)
- {
- OBJECT_CONSTRUCTION;
- pos += offset;
- pos.w = dimensions.x;
- pos.h = dimensions.y;
- }
- void BasicMapView::render(Canvas & target, bool fullUpdate)
- {
- Canvas targetClipped(target, pos);
- tilesCache->update(controller->getContext());
- tilesCache->render(controller->getContext(), targetClipped, fullUpdate);
- MapOverlayLogVisualizer r(targetClipped, model);
- logVisual->visualize(r);
- }
- void BasicMapView::tick(uint32_t msPassed)
- {
- controller->tick(msPassed);
- }
- void BasicMapView::show(Canvas & to)
- {
- CanvasClipRectGuard guard(to, pos);
- render(to, needFullUpdate);
- controller->afterRender();
- }
- void BasicMapView::showAll(Canvas & to)
- {
- CanvasClipRectGuard guard(to, pos);
- render(to, true);
- }
- void MapView::tick(uint32_t msPassed)
- {
- if(settings["adventure"]["smoothDragging"].Bool())
- postSwipe(msPassed);
- BasicMapView::tick(msPassed);
- }
- void MapView::show(Canvas & to)
- {
- actions->setContext(controller->getContext());
- BasicMapView::show(to);
- }
- MapView::MapView(const Point & offset, const Point & dimensions)
- : BasicMapView(offset, dimensions)
- {
- OBJECT_CONSTRUCTION;
- actions = std::make_shared<MapViewActions>(*this, model);
- actions->setContext(controller->getContext());
- // catch min 6 frames
- postSwipeCatchIntervalMs = std::max(100, static_cast<int>(6.0 * 1000.0 * (1.0 / settings["video"]["targetfps"].Float())));
- }
- void MapView::onMapLevelSwitched()
- {
- if(GAME->interface()->cb->getMapSize().z > 1)
- {
- int newLevel = model->getLevel() + 1;
- if(newLevel < GAME->interface()->cb->getMapSize().z)
- controller->setViewCenter(model->getMapViewCenter(), newLevel);
- else
- controller->setViewCenter(model->getMapViewCenter(), 0);
- }
- }
- void MapView::onMapScrolled(const Point & distance)
- {
- if(!isGesturing())
- {
- postSwipeSpeed = 0.0;
- controller->setViewCenter(model->getMapViewCenter() + distance, model->getLevel());
- }
- }
- void MapView::onMapSwiped(const Point & viewPosition)
- {
- if(settings["adventure"]["smoothDragging"].Bool())
- swipeHistory.push_back(std::pair<uint32_t, Point>(ENGINE->input().getTicks(), viewPosition));
- controller->setViewCenter(model->getMapViewCenter() + viewPosition, model->getLevel());
- }
- void MapView::postSwipe(uint32_t msPassed)
- {
- if(!actions->dragActive)
- {
- if(swipeHistory.size() > 1)
- {
- Point diff = Point(0, 0);
- std::pair<uint32_t, Point> firstAccepted;
- uint32_t now = ENGINE->input().getTicks();
- for (auto & x : swipeHistory) {
- if(now - x.first < postSwipeCatchIntervalMs) { // only the last x ms are caught
- if(firstAccepted.first == 0)
- firstAccepted = x;
- diff += x.second;
- }
- }
- uint32_t timediff = swipeHistory.back().first - firstAccepted.first;
- if(diff.length() > 0 && timediff > 0)
- {
- postSwipeAngle = diff.angle();
- postSwipeSpeed = static_cast<double>(diff.length()) / static_cast<double>(timediff); // unit: pixel/millisecond
- }
- }
- swipeHistory.clear();
- } else
- postSwipeSpeed = 0.0;
- if(postSwipeSpeed > postSwipeMinimalSpeed) {
- double len = postSwipeSpeed * static_cast<double>(msPassed);
- Point delta = Point(len * cos(postSwipeAngle), len * sin(postSwipeAngle));
- controller->setViewCenter(model->getMapViewCenter() + delta, model->getLevel());
- postSwipeSpeed /= 1 + msPassed * postSwipeSlowdownSpeed;
- }
- }
- void MapView::onCenteredTile(const int3 & tile)
- {
- controller->setViewCenter(tile);
- }
- void MapView::onCenteredObject(const CGObjectInstance * target)
- {
- controller->setViewCenter(target->getSightCenter());
- }
- void MapView::onViewSpellActivated(uint32_t tileSize, const std::vector<ObjectPosInfo> & objectPositions, bool showTerrain)
- {
- controller->activateSpellViewContext();
- controller->setTileSize(Point(tileSize, tileSize));
- controller->setOverlayVisibility(objectPositions);
- controller->setTerrainVisibility(showTerrain);
- }
- void MapView::onViewWorldActivated(uint32_t tileSize)
- {
- controller->activateWorldViewContext();
- controller->setTileSize(Point(tileSize, tileSize));
- }
- void MapView::onMapZoomLevelChanged(int stepsChange, bool useDeadZone)
- {
- controller->modifyTileSize(stepsChange, useDeadZone);
- }
- void MapView::onViewMapActivated()
- {
- controller->activateAdventureContext();
- int zoom = settings["adventure"]["tileZoom"].Integer();
- if(zoom)
- controller->setTileSize(Point(zoom, zoom));
- else
- controller->setTileSize(Point(32, 32));
- }
- PuzzleMapView::PuzzleMapView(const Point & offset, const Point & dimensions, const int3 & tileToCenter)
- : BasicMapView(offset, dimensions)
- {
- controller->activatePuzzleMapContext(tileToCenter);
- controller->setViewCenter(tileToCenter);
- }
|