|
|
|
#pragma region License
|
|
|
|
/*
|
|
|
|
License (OLC-3)
|
|
|
|
~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Copyright 2024 Joshua Sigona <sigonasr2@gmail.com>
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
are permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
1. Redistributions or derivations of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
|
|
|
|
2. Redistributions or derivative works in binary form must reproduce the above
|
|
|
|
copyright notice. This list of conditions and the following disclaimer must be
|
|
|
|
reproduced in the documentation and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
3. Neither the name of the copyright holder nor the names of its contributors may
|
|
|
|
be used to endorse or promote products derived from this software without specific
|
|
|
|
prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
|
|
|
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
|
|
|
|
SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
|
|
|
|
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
|
|
|
BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
SUCH DAMAGE.
|
|
|
|
|
|
|
|
Portions of this software are copyright © 2024 The FreeType
|
|
|
|
Project (www.freetype.org). Please see LICENSE_FT.txt for more information.
|
|
|
|
All rights reserved.
|
|
|
|
*/
|
|
|
|
#pragma endregion
|
|
|
|
#include "util.h"
|
|
|
|
#include "olcPixelGameEngine.h"
|
|
|
|
#include "olcPGEX_TTF.h"
|
|
|
|
|
|
|
|
std::random_device rd;
|
|
|
|
std::mt19937 rng(rd());
|
|
|
|
|
|
|
|
float util::random(float range){
|
|
|
|
static std::uniform_real_distribution<float>distrib(0,1);
|
|
|
|
return distrib(rng)*range;
|
|
|
|
}
|
|
|
|
|
|
|
|
int util::random(){
|
|
|
|
static std::uniform_int_distribution<int>distrib(0,32767);
|
|
|
|
return distrib(rng);
|
|
|
|
}
|
|
|
|
|
|
|
|
vf2d util::pointTo(vf2d posFrom,vf2d posTo){
|
|
|
|
return geom2d::line(posFrom,posTo).vector().norm();
|
|
|
|
}
|
|
|
|
|
|
|
|
float util::angleTo(vf2d posFrom,vf2d posTo){
|
|
|
|
return geom2d::line<float>(posFrom,posTo).vector().polar().y;
|
|
|
|
}
|
|
|
|
|
|
|
|
float util::degToRad(float deg){
|
|
|
|
return deg*(PI/180);
|
|
|
|
}
|
|
|
|
|
|
|
|
float util::radToDeg(float rad){
|
|
|
|
return rad*57.2957795130823208767f;
|
|
|
|
}
|
|
|
|
|
|
|
|
float util::lerp(float n1,float n2,double t){
|
|
|
|
return float(n1*(1-t)+n2*t);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string util::PixelToHTMLColorCode(const Pixel&col){
|
|
|
|
std::string hexCode=std::format("#{:2x}{:2x}{:2x}",col.r,col.g,col.b);
|
|
|
|
for(char&c:hexCode){
|
|
|
|
c=toupper(c);
|
|
|
|
}
|
|
|
|
return hexCode;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string util::timerStr(float time){
|
|
|
|
int seconds=int(time);
|
|
|
|
int hours=seconds/3600;
|
|
|
|
int minutes=seconds/60;
|
|
|
|
|
|
|
|
std::string timeStr="";
|
|
|
|
if(hours>0){
|
|
|
|
if(hours<10)timeStr+="0";
|
|
|
|
timeStr+=std::to_string(hours)+":";
|
|
|
|
}
|
|
|
|
|
|
|
|
if(minutes%60<10)timeStr+="0";
|
|
|
|
timeStr+=std::to_string(minutes%60)+":";
|
|
|
|
|
|
|
|
if(seconds%60<10)timeStr+="0";
|
|
|
|
timeStr+=std::to_string(seconds%60);
|
|
|
|
|
|
|
|
return timeStr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string util::WrapText(PixelGameEngine*pge,std::string str,int width,bool proportional,vd2d scale){
|
|
|
|
std::string newStr="";
|
|
|
|
while(true){
|
|
|
|
std::string word="";
|
|
|
|
if(str.find(" ")==std::string::npos){
|
|
|
|
word=str;
|
|
|
|
}else{
|
|
|
|
word=str.substr(0,str.find(" "));
|
|
|
|
}
|
|
|
|
vi2d newSize = vd2d(proportional?pge->GetTextSizeProp(newStr+(newStr.size()>0?" ":"")+word):pge->GetTextSize(newStr+(newStr.size()>0?" ":"")+word))*scale;
|
|
|
|
if(newSize.x>width){
|
|
|
|
newStr+="\n"+word;
|
|
|
|
}else{
|
|
|
|
newStr+=(newStr.size()>0?" ":"")+word;
|
|
|
|
}
|
|
|
|
if(str.find(" ")==std::string::npos){
|
|
|
|
break;
|
|
|
|
}else{
|
|
|
|
str.erase(0,str.find(" ")+1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return newStr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::u32string util::WrapText(PixelGameEngine*pge,std::u32string str,int width,Font&font,vd2d scale){
|
|
|
|
std::u32string newStr=U"";
|
|
|
|
while(true){
|
|
|
|
std::u32string word=U"";
|
|
|
|
if(str.find(U" ")==std::u32string::npos){
|
|
|
|
word=str;
|
|
|
|
}else{
|
|
|
|
word=str.substr(0,str.find(U" "));
|
|
|
|
}
|
|
|
|
FontRect newSize = font.GetStringBounds(newStr+(newStr.size()>0?U" ":U"")+word);
|
|
|
|
if(newSize.size.x>width){
|
|
|
|
newStr+=U"\n"+word;
|
|
|
|
}else{
|
|
|
|
newStr+=(newStr.size()>0?U" ":U"")+word;
|
|
|
|
}
|
|
|
|
if(str.find(U" ")==std::u32string::npos){
|
|
|
|
break;
|
|
|
|
}else{
|
|
|
|
str.erase(0,str.find(U" ")+1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return newStr;
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma region std::string util::GetHash(std::string fileName) //DO NOT MODIFY!
|
|
|
|
std::string util::GetHash(std::string fileName){
|
|
|
|
//WARNING! This function is used to save/load files! This means if we ever modify this function all previous save files
|
|
|
|
//will no longer work! IN OTHER WORDS: DO NOT MODIFY THIS FUNCTION!
|
|
|
|
std::ifstream file(fileName);
|
|
|
|
std::string hash="";
|
|
|
|
uint8_t hashIndex=0;
|
|
|
|
while(file.good()){
|
|
|
|
uint8_t hashChar=0;
|
|
|
|
if(hash.size()==std::numeric_limits<uint8_t>::max()){
|
|
|
|
hashChar=hash[hashIndex];
|
|
|
|
}
|
|
|
|
hashChar+=file.get()*21-7;
|
|
|
|
hashChar^=hashIndex;
|
|
|
|
if(hashIndex>0)hashChar+=hash[hashIndex-1];
|
|
|
|
hashChar%=94;
|
|
|
|
hashChar+=32;
|
|
|
|
if(hashChar=='"'){hashChar+=60;}
|
|
|
|
if(hash.size()<std::numeric_limits<uint8_t>::max()){
|
|
|
|
hash+=hashChar;
|
|
|
|
}else{
|
|
|
|
hash[hashIndex]=hashChar;
|
|
|
|
}
|
|
|
|
hashIndex++;
|
|
|
|
}
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
#pragma endregion
|
|
|
|
|
|
|
|
long double operator""_Pixels(long double unitDist){
|
|
|
|
return unitDist/100*24.;
|
|
|
|
}
|
|
|
|
|
|
|
|
float util::angle_difference(float angle_1, float angle_2)
|
|
|
|
{
|
|
|
|
angle_1 = fmod(angle_1, 2 * PI);
|
|
|
|
angle_2 = fmod(angle_2, 2 * PI);
|
|
|
|
float angle_diff = angle_1 - angle_2;
|
|
|
|
|
|
|
|
if (angle_diff > PI)
|
|
|
|
angle_diff -= 2 * PI;
|
|
|
|
else if (angle_diff < -PI)
|
|
|
|
angle_diff += 2 * PI;
|
|
|
|
|
|
|
|
return -angle_diff;
|
|
|
|
}
|