Skip to content

Commit

Permalink
Add a converter for sky cultures from old format to new
Browse files Browse the repository at this point in the history
It's still quite raw, especially regarding the conversion of
descriptions and their translations, but at least it converts
the other data.
  • Loading branch information
10110111 committed Jan 14, 2025
1 parent b91304c commit 1f950f0
Show file tree
Hide file tree
Showing 10 changed files with 2,471 additions and 0 deletions.
296 changes: 296 additions & 0 deletions util/skyculture-converter/AsterismOldLoader.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,296 @@
#include <cmath>
#include <limits>
#include <QDir>
#include <QFile>
#include <QDebug>
#include <QRegularExpression>

#include "AsterismOldLoader.hpp"

class Asterism
{
public:
struct Star
{
int HIP = -1;
double RA = NAN, DE = NAN;
bool operator==(const Star& rhs) const
{
if (HIP > 0) return HIP == rhs.HIP;
return RA == rhs.RA && DE == rhs.DE;
}
};
private:
//! International name (translated using gettext)
QString nameI18;
//! Name in english (second column in asterism_names.eng.fab)
QString englishName;
//! Abbreviation
//! A skyculture designer must invent it. (usually 2-5 letters)
//! This MUST be filled and be unique within a sky culture.
QString abbreviation;
//! Context for name
QString context;
//! Number of segments in the lines
unsigned int numberOfSegments;
//! Type of asterism
int typeOfAsterism;
bool flagAsterism;

std::vector<Star> asterism;

friend class AsterismOldLoader;
public:
bool read(const QString& record);
};

std::ostream& operator<<(std::ostream& s, const Asterism::Star& star)
{
if (star.HIP > 0)
s << star.HIP;
else
s << "[" << star.RA << ", " << star.DE << "]";
return s;
}

bool Asterism::read(const QString& record)
{
abbreviation.clear();
numberOfSegments = 0;
typeOfAsterism = 1;
flagAsterism = true;

QString buf(record);
QTextStream istr(&buf, QIODevice::ReadOnly);
// We allow mixed-case abbreviations now that they can be displayed on screen. We then need toUpper() in comparisons.
istr >> abbreviation >> typeOfAsterism >> numberOfSegments;
if (istr.status()!=QTextStream::Ok)
return false;

asterism.resize(numberOfSegments*2);
for (unsigned int i=0;i<numberOfSegments*2;++i)
{
switch (typeOfAsterism)
{
case 0: // Ray helpers
case 1: // A big asterism with lines by HIP stars
{
unsigned int HP = 0;
istr >> HP;
if(HP == 0)
{
return false;
}
asterism[i] = Star{int(HP), NAN, NAN};
break;
}
case 2: // A small asterism with lines by J2000.0 coordinates
{
double RA, DE;
istr >> RA >> DE;
asterism[i] = Star{-1, RA, DE};
break;
}
}
}

return true;
}

Asterism* AsterismOldLoader::findFromAbbreviation(const QString& abbrev) const
{
for (const auto asterism : asterisms)
if (asterism->abbreviation == abbrev)
return asterism;
return nullptr;
}

void AsterismOldLoader::load(const QString& skyCultureDir, const QString& cultureId)
{
this->cultureId = cultureId;
QString fic = skyCultureDir+"/asterism_lines.fab";
if (fic.isEmpty())
{
hasAsterism = false;
qWarning() << "No asterisms in " << skyCultureDir;
}
else
{
hasAsterism = true;
loadLines(fic);
}

// load asterism names
fic = skyCultureDir + "/asterism_names.eng.fab";
if (!fic.isEmpty())
loadNames(fic);
}

void AsterismOldLoader::loadLines(const QString &fileName)
{
QFile in(fileName);
if (!in.open(QIODevice::ReadOnly | QIODevice::Text))
{
qWarning() << "Can't open asterism data file" << QDir::toNativeSeparators(fileName);
return;
}

int totalRecords=0;
QString record;
static const QRegularExpression commentRx("^(\\s*#.*|\\s*)$");
while (!in.atEnd())
{
record = QString::fromUtf8(in.readLine());
if (!commentRx.match(record).hasMatch())
totalRecords++;
}
in.seek(0);

// delete existing data, if any
for (auto* asterism : asterisms)
delete asterism;

asterisms.clear();
Asterism *aster = Q_NULLPTR;

// read the file of line patterns, adding a record per non-comment line
int currentLineNumber = 0; // line in file
int readOk = 0; // count of records processed OK
while (!in.atEnd())
{
record = QString::fromUtf8(in.readLine());
currentLineNumber++;
if (commentRx.match(record).hasMatch())
continue;

aster = new Asterism;
if(aster->read(record))
{
asterisms.push_back(aster);
++readOk;
}
else
{
qWarning() << "ERROR reading asterism lines record at line " << currentLineNumber;
delete aster;
}
}
in.close();
qDebug() << "Loaded" << readOk << "/" << totalRecords << "asterism records successfully";
}

void AsterismOldLoader::loadNames(const QString& namesFile)
{
// Asterism not loaded yet
if (asterisms.empty()) return;

// clear previous names
for (auto* asterism : asterisms)
{
asterism->englishName.clear();
}

// Open file
QFile commonNameFile(namesFile);
if (!commonNameFile.open(QIODevice::ReadOnly | QIODevice::Text))
{
qDebug() << "Cannot open file" << QDir::toNativeSeparators(namesFile);
return;
}

// Now parse the file
// lines to ignore which start with a # or are empty
static const QRegularExpression commentRx("^(\\s*#.*|\\s*)$");
static const QRegularExpression recRx("^\\s*(\\S+)\\s+_[(]\"(.*)\"[)]\\s*([\\,\\d\\s]*)\\n");
static const QRegularExpression ctxRx("(.*)\",\\s*\"(.*)");

// keep track of how many records we processed.
int totalRecords=0;
int readOk=0;
int lineNumber=0;
while (!commonNameFile.atEnd())
{
QString record = QString::fromUtf8(commonNameFile.readLine());
lineNumber++;

// Skip comments
if (commentRx.match(record).hasMatch())
continue;

totalRecords++;

QRegularExpressionMatch recMatch=recRx.match(record);
if (!recMatch.hasMatch())
{
qWarning() << "ERROR - cannot parse record at line" << lineNumber << "in asterism names file" << QDir::toNativeSeparators(namesFile) << ":" << record;
}
else
{
QString shortName = recMatch.captured(1);
Asterism *aster = findFromAbbreviation(shortName);
// If the asterism exists, set the English name
if (aster != Q_NULLPTR)
{
QString ctxt = recMatch.captured(2);
QRegularExpressionMatch ctxMatch=ctxRx.match(ctxt);
if (ctxMatch.hasMatch())
{
aster->englishName = ctxMatch.captured(1);
aster->context = ctxMatch.captured(2);
}
else
{
aster->englishName = ctxt;
aster->context = "";
}
readOk++;
}
else
{
qWarning() << "WARNING - asterism abbreviation" << shortName << "not found when loading asterism names";
}
}
}
commonNameFile.close();
qDebug() << "Loaded" << readOk << "/" << totalRecords << "asterism names";
}

bool AsterismOldLoader::dumpJSON(std::ostream& s) const
{
if (!hasAsterism) return false;

s << " \"asterisms\": [\n";
for (const Asterism*const ast : asterisms)
{
s << " {\n";
s << " \"id\": \"AST " << cultureId.toStdString() << " " << ast->abbreviation.toStdString() << "\",\n";
if (!ast->englishName.isEmpty())
s << " \"common_name\": {\"english\": \"" << ast->englishName.toStdString() << "\"},\n";
const bool isRayHelper = ast->typeOfAsterism == 0;
if(isRayHelper)
s << " \"is_ray_helper\": true,\n";

s.precision(std::numeric_limits<double>::digits10);
s << " \"lines\": [";
auto& points = ast->asterism;
for (unsigned n = 1; n < points.size(); n += 2)
{
s << (n > 1 ? ", [" : "[") << points[n - 1] << ", " << points[n];
// Merge connected segments into polylines
while (n + 2 < points.size() && points[n + 1] == points[n])
{
s << ", " << points[n + 2];
n += 2;
}
s << "]";
}
s << "]\n";
if (ast == asterisms.back())
s << " }\n";
else
s << " },\n";
}
s << " ],\n";

return true;
}
19 changes: 19 additions & 0 deletions util/skyculture-converter/AsterismOldLoader.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
#pragma once

#include <vector>
#include <QString>

class Asterism;
class AsterismOldLoader
{
QString cultureId;
bool hasAsterism = false;
std::vector<Asterism*> asterisms;

Asterism* findFromAbbreviation(const QString& abbrev) const;
void loadLines(const QString& fileName);
void loadNames(const QString& namesFile);
public:
void load(const QString& skyCultureDir, const QString& cultureId);
bool dumpJSON(std::ostream& s) const;
};
21 changes: 21 additions & 0 deletions util/skyculture-converter/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
cmake_minimum_required(VERSION 3.10.0)
project(skyculture-converter VERSION 0.0.1 LANGUAGES CXX)

set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
if(${CMAKE_CXX_COMPILER_ID} MATCHES "GNU|Clang")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Werror=return-type -Wall -Wextra")
endif()

set(CMAKE_AUTOMOC ON)
set(CMAKE_AUTOUIC ON)
find_package(Qt5 5.15 REQUIRED Core OpenGL)

add_executable(skyculture-converter
NamesOldLoader.cpp
AsterismOldLoader.cpp
DescriptionOldLoader.cpp
ConstellationOldLoader.cpp
main.cpp
)
target_link_libraries(skyculture-converter Qt5::Core Qt5::Gui gettextpo)
Loading

0 comments on commit 1f950f0

Please sign in to comment.