/* * Copyright (c) 2018, 2019, 2020 shedaniel * Licensed under the MIT License (the "License"). */ package me.shedaniel.rei.impl; import com.google.common.collect.Lists; import com.google.common.collect.Maps; import me.shedaniel.math.api.Rectangle; import me.shedaniel.rei.api.BaseBoundsHandler; import me.shedaniel.rei.api.DisplayHelper; import me.shedaniel.rei.api.OverlayDecider; import me.shedaniel.rei.utils.CollectionUtils; import org.jetbrains.annotations.ApiStatus; import java.util.Collections; import java.util.Comparator; import java.util.List; import java.util.Map; import java.util.stream.Collectors; @ApiStatus.Internal public class DisplayHelperImpl implements DisplayHelper { private static final Comparator BOUNDS_HANDLER_COMPARATOR; private static final DisplayBoundsHandler EMPTY = new DisplayBoundsHandler() { @Override public Class getBaseSupportedClass() { return null; } @Override public Rectangle getLeftBounds(Object screen) { return new Rectangle(); } @Override public Rectangle getRightBounds(Object screen) { return new Rectangle(); } @Override public float getPriority() { return -10f; } }; static { Comparator comparator = Comparator.comparingDouble(OverlayDecider::getPriority); BOUNDS_HANDLER_COMPARATOR = comparator.reversed(); } private List screenDisplayBoundsHandlers = Lists.newArrayList(); private Map, DisplayBoundsHandler> handlerCache = Maps.newHashMap(); private Map, List>> handlerSortedCache = Maps.newHashMap(); private BaseBoundsHandler baseBoundsHandler; private Class tempScreen; @SuppressWarnings("rawtypes") @Override public List> getSortedBoundsHandlers(Class screenClass) { List> possibleCached = handlerSortedCache.get(screenClass); if (possibleCached != null) return possibleCached; tempScreen = screenClass; handlerSortedCache.put(screenClass, (List) CollectionUtils.castAndMap(CollectionUtils.filter(screenDisplayBoundsHandlers, this::filterResponsible), DisplayBoundsHandler.class)); return handlerSortedCache.get(screenClass); } @Override public List getAllOverlayDeciders() { return Collections.unmodifiableList(screenDisplayBoundsHandlers); } @Override public DisplayBoundsHandler getResponsibleBoundsHandler(Class screenClass) { DisplayBoundsHandler possibleCached = handlerCache.get(screenClass); if (possibleCached != null) return possibleCached; List> handlers = getSortedBoundsHandlers(screenClass); handlerCache.put(screenClass, handlers.isEmpty() ? EMPTY : handlers.get(0)); return handlerCache.get(screenClass); } private boolean filterResponsible(OverlayDecider handler) { return handler.isHandingScreen(tempScreen); } @Override public void registerHandler(OverlayDecider decider) { screenDisplayBoundsHandlers.add(decider); screenDisplayBoundsHandlers.sort(BOUNDS_HANDLER_COMPARATOR); } @Override public BaseBoundsHandler getBaseBoundsHandler() { return baseBoundsHandler; } @ApiStatus.Internal public void setBaseBoundsHandler(BaseBoundsHandler baseBoundsHandler) { this.baseBoundsHandler = baseBoundsHandler; } @ApiStatus.Internal public void resetData() { screenDisplayBoundsHandlers.clear(); } @ApiStatus.Internal public void resetCache() { handlerCache.clear(); handlerSortedCache.clear(); } }