linked events

This commit is contained in:
Robert Janetzko 2022-04-09 16:37:44 +00:00
parent 0cba9a87f7
commit bd4efed85e
27 changed files with 909 additions and 673 deletions

View File

@ -75,7 +75,8 @@
"browserTarget": "legendsbrowser:build:production" "browserTarget": "legendsbrowser:build:production"
}, },
"development": { "development": {
"browserTarget": "legendsbrowser:build:development" "browserTarget": "legendsbrowser:build:development",
"proxyConfig": "src/proxy.conf.json"
} }
}, },
"defaultConfiguration": "development" "defaultConfiguration": "development"
@ -108,4 +109,4 @@
} }
}, },
"defaultProject": "legendsbrowser" "defaultProject": "legendsbrowser"
} }

View File

@ -1,7 +1,13 @@
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router'; import { RouterModule, Routes } from '@angular/router';
import { EntitiesResolver, EntityResolver } from './entity.service';
import { CivilizationsComponent } from './pages/civilizations/civilizations.component';
import { EntityComponent } from './pages/entity/entity.component';
const routes: Routes = []; const routes: Routes = [
{ path: '', component: CivilizationsComponent, resolve: { civilizations: EntitiesResolver } },
{ path: 'entity/:id', component: EntityComponent, resolve: { entity: EntityResolver } },
];
@NgModule({ @NgModule({
imports: [RouterModule.forRoot(routes)], imports: [RouterModule.forRoot(routes)],

View File

@ -1,484 +1,2 @@
<!-- * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * --> <h1>LegendsBrowser 2</h1>
<!-- * * * * * * * * * * * The content below * * * * * * * * * * * --> <router-outlet></router-outlet>
<!-- * * * * * * * * * * is only a placeholder * * * * * * * * * * -->
<!-- * * * * * * * * * * and can be replaced. * * * * * * * * * * * -->
<!-- * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * -->
<!-- * * * * * * * * * Delete the template below * * * * * * * * * * -->
<!-- * * * * * * * to get started with your project! * * * * * * * * -->
<!-- * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * -->
<style>
:host {
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
font-size: 14px;
color: #333;
box-sizing: border-box;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
h1,
h2,
h3,
h4,
h5,
h6 {
margin: 8px 0;
}
p {
margin: 0;
}
.spacer {
flex: 1;
}
.toolbar {
position: absolute;
top: 0;
left: 0;
right: 0;
height: 60px;
display: flex;
align-items: center;
background-color: #1976d2;
color: white;
font-weight: 600;
}
.toolbar img {
margin: 0 16px;
}
.toolbar #twitter-logo {
height: 40px;
margin: 0 8px;
}
.toolbar #youtube-logo {
height: 40px;
margin: 0 16px;
}
.toolbar #twitter-logo:hover,
.toolbar #youtube-logo:hover {
opacity: 0.8;
}
.content {
display: flex;
margin: 82px auto 32px;
padding: 0 16px;
max-width: 960px;
flex-direction: column;
align-items: center;
}
svg.material-icons {
height: 24px;
width: auto;
}
svg.material-icons:not(:last-child) {
margin-right: 8px;
}
.card svg.material-icons path {
fill: #888;
}
.card-container {
display: flex;
flex-wrap: wrap;
justify-content: center;
margin-top: 16px;
}
.card {
all: unset;
border-radius: 4px;
border: 1px solid #eee;
background-color: #fafafa;
height: 40px;
width: 200px;
margin: 0 8px 16px;
padding: 16px;
display: flex;
flex-direction: row;
justify-content: center;
align-items: center;
transition: all 0.2s ease-in-out;
line-height: 24px;
}
.card-container .card:not(:last-child) {
margin-right: 0;
}
.card.card-small {
height: 16px;
width: 168px;
}
.card-container .card:not(.highlight-card) {
cursor: pointer;
}
.card-container .card:not(.highlight-card):hover {
transform: translateY(-3px);
box-shadow: 0 4px 17px rgba(0, 0, 0, 0.35);
}
.card-container .card:not(.highlight-card):hover .material-icons path {
fill: rgb(105, 103, 103);
}
.card.highlight-card {
background-color: #1976d2;
color: white;
font-weight: 600;
border: none;
width: auto;
min-width: 30%;
position: relative;
}
.card.card.highlight-card span {
margin-left: 60px;
}
svg#rocket {
width: 80px;
position: absolute;
left: -10px;
top: -24px;
}
svg#rocket-smoke {
height: calc(100vh - 95px);
position: absolute;
top: 10px;
right: 180px;
z-index: -10;
}
a,
a:visited,
a:hover {
color: #1976d2;
text-decoration: none;
}
a:hover {
color: #125699;
}
.terminal {
position: relative;
width: 80%;
max-width: 600px;
border-radius: 6px;
padding-top: 45px;
margin-top: 8px;
overflow: hidden;
background-color: rgb(15, 15, 16);
}
.terminal::before {
content: "\2022 \2022 \2022";
position: absolute;
top: 0;
left: 0;
height: 4px;
background: rgb(58, 58, 58);
color: #c2c3c4;
width: 100%;
font-size: 2rem;
line-height: 0;
padding: 14px 0;
text-indent: 4px;
}
.terminal pre {
font-family: SFMono-Regular,Consolas,Liberation Mono,Menlo,monospace;
color: white;
padding: 0 1rem 1rem;
margin: 0;
}
.circle-link {
height: 40px;
width: 40px;
border-radius: 40px;
margin: 8px;
background-color: white;
border: 1px solid #eeeeee;
display: flex;
justify-content: center;
align-items: center;
cursor: pointer;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.12), 0 1px 2px rgba(0, 0, 0, 0.24);
transition: 1s ease-out;
}
.circle-link:hover {
transform: translateY(-0.25rem);
box-shadow: 0px 3px 15px rgba(0, 0, 0, 0.2);
}
footer {
margin-top: 8px;
display: flex;
align-items: center;
line-height: 20px;
}
footer a {
display: flex;
align-items: center;
}
.github-star-badge {
color: #24292e;
display: flex;
align-items: center;
font-size: 12px;
padding: 3px 10px;
border: 1px solid rgba(27,31,35,.2);
border-radius: 3px;
background-image: linear-gradient(-180deg,#fafbfc,#eff3f6 90%);
margin-left: 4px;
font-weight: 600;
}
.github-star-badge:hover {
background-image: linear-gradient(-180deg,#f0f3f6,#e6ebf1 90%);
border-color: rgba(27,31,35,.35);
background-position: -.5em;
}
.github-star-badge .material-icons {
height: 16px;
width: 16px;
margin-right: 4px;
}
svg#clouds {
position: fixed;
bottom: -160px;
left: -230px;
z-index: -10;
width: 1920px;
}
/* Responsive Styles */
@media screen and (max-width: 767px) {
.card-container > *:not(.circle-link) ,
.terminal {
width: 100%;
}
.card:not(.highlight-card) {
height: 16px;
margin: 8px 0;
}
.card.highlight-card span {
margin-left: 72px;
}
svg#rocket-smoke {
right: 120px;
transform: rotate(-5deg);
}
}
@media screen and (max-width: 575px) {
svg#rocket-smoke {
display: none;
visibility: hidden;
}
}
</style>
<!-- Toolbar -->
<div class="toolbar" role="banner">
<img
width="40"
alt="Angular Logo"
src=""
/>
<span>Welcome</span>
<div class="spacer"></div>
<a aria-label="Angular on twitter" target="_blank" rel="noopener" href="https://twitter.com/angular" title="Twitter">
<svg id="twitter-logo" height="24" data-name="Logo" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 400 400">
<rect width="400" height="400" fill="none"/>
<path d="M153.62,301.59c94.34,0,145.94-78.16,145.94-145.94,0-2.22,0-4.43-.15-6.63A104.36,104.36,0,0,0,325,122.47a102.38,102.38,0,0,1-29.46,8.07,51.47,51.47,0,0,0,22.55-28.37,102.79,102.79,0,0,1-32.57,12.45,51.34,51.34,0,0,0-87.41,46.78A145.62,145.62,0,0,1,92.4,107.81a51.33,51.33,0,0,0,15.88,68.47A50.91,50.91,0,0,1,85,169.86c0,.21,0,.43,0,.65a51.31,51.31,0,0,0,41.15,50.28,51.21,51.21,0,0,1-23.16.88,51.35,51.35,0,0,0,47.92,35.62,102.92,102.92,0,0,1-63.7,22A104.41,104.41,0,0,1,75,278.55a145.21,145.21,0,0,0,78.62,23" fill="#fff"/>
</svg>
</a>
<a aria-label="Angular on YouTube" target="_blank" rel="noopener" href="https://youtube.com/angular" title="YouTube">
<svg id="youtube-logo" height="24" width="24" data-name="Logo" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="#fff">
<path d="M0 0h24v24H0V0z" fill="none"/>
<path d="M21.58 7.19c-.23-.86-.91-1.54-1.77-1.77C18.25 5 12 5 12 5s-6.25 0-7.81.42c-.86.23-1.54.91-1.77 1.77C2 8.75 2 12 2 12s0 3.25.42 4.81c.23.86.91 1.54 1.77 1.77C5.75 19 12 19 12 19s6.25 0 7.81-.42c.86-.23 1.54-.91 1.77-1.77C22 15.25 22 12 22 12s0-3.25-.42-4.81zM10 15V9l5.2 3-5.2 3z"/>
</svg>
</a>
</div>
<div class="content" role="main">
<!-- Highlight Card -->
<div class="card highlight-card card-small">
<svg id="rocket" xmlns="http://www.w3.org/2000/svg" width="101.678" height="101.678" viewBox="0 0 101.678 101.678">
<title>Rocket Ship</title>
<g id="Group_83" data-name="Group 83" transform="translate(-141 -696)">
<circle id="Ellipse_8" data-name="Ellipse 8" cx="50.839" cy="50.839" r="50.839" transform="translate(141 696)" fill="#dd0031"/>
<g id="Group_47" data-name="Group 47" transform="translate(165.185 720.185)">
<path id="Path_33" data-name="Path 33" d="M3.4,42.615a3.084,3.084,0,0,0,3.553,3.553,21.419,21.419,0,0,0,12.215-6.107L9.511,30.4A21.419,21.419,0,0,0,3.4,42.615Z" transform="translate(0.371 3.363)" fill="#fff"/>
<path id="Path_34" data-name="Path 34" d="M53.3,3.221A3.09,3.09,0,0,0,50.081,0,48.227,48.227,0,0,0,18.322,13.437c-6-1.666-14.991-1.221-18.322,7.218A33.892,33.892,0,0,1,9.439,25.1l-.333.666a3.013,3.013,0,0,0,.555,3.553L23.985,43.641a2.9,2.9,0,0,0,3.553.555l.666-.333A33.892,33.892,0,0,1,32.647,53.3c8.55-3.664,8.884-12.326,7.218-18.322A48.227,48.227,0,0,0,53.3,3.221ZM34.424,9.772a6.439,6.439,0,1,1,9.106,9.106,6.368,6.368,0,0,1-9.106,0A6.467,6.467,0,0,1,34.424,9.772Z" transform="translate(0 0.005)" fill="#fff"/>
</g>
</g>
</svg>
<span>{{ title }} app is running!</span>
<svg id="rocket-smoke" xmlns="http://www.w3.org/2000/svg" width="516.119" height="1083.632" viewBox="0 0 516.119 1083.632">
<title>Rocket Ship Smoke</title>
<path id="Path_40" data-name="Path 40" d="M644.6,141S143.02,215.537,147.049,870.207s342.774,201.755,342.774,201.755S404.659,847.213,388.815,762.2c-27.116-145.51-11.551-384.124,271.9-609.1C671.15,139.365,644.6,141,644.6,141Z" transform="translate(-147.025 -140.939)" fill="#f5f5f5"/>
</svg>
</div>
<!-- Resources -->
<h2>Resources</h2>
<p>Here are some links to help you get started:</p>
<div class="card-container">
<a class="card" target="_blank" rel="noopener" href="https://angular.io/tutorial">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M5 13.18v4L12 21l7-3.82v-4L12 17l-7-3.82zM12 3L1 9l11 6 9-4.91V17h2V9L12 3z"/></svg>
<span>Learn Angular</span>
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"/></svg> </a>
<a class="card" target="_blank" rel="noopener" href="https://angular.io/cli">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M9.4 16.6L4.8 12l4.6-4.6L8 6l-6 6 6 6 1.4-1.4zm5.2 0l4.6-4.6-4.6-4.6L16 6l6 6-6 6-1.4-1.4z"/></svg>
<span>CLI Documentation</span>
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"/></svg>
</a>
<a class="card" target="_blank" rel="noopener" href="https://material.angular.io">
<svg xmlns="http://www.w3.org/2000/svg" style="margin-right: 8px" width="21.813" height="23.453" viewBox="0 0 21.813 23.453"><path d="M4099.584,972.736h0l-10.882,3.9,1.637,14.4,9.245,5.153,9.245-5.153,1.686-14.4Z" transform="translate(-4088.702 -972.736)" fill="#808080"/><path d="M4181.516,972.736v23.453l9.245-5.153,1.686-14.4Z" transform="translate(-4170.633 -972.736)" fill="#808080"/><path d="M4137.529,1076.127l-7.7-3.723,4.417-2.721,7.753,3.723Z" transform="translate(-4125.003 -1058.315)" fill="#ffe0b2"/><path d="M4137.529,1051.705l-7.7-3.723,4.417-2.721,7.753,3.723Z" transform="translate(-4125.003 -1036.757)" fill="#fff3e0"/><path d="M4137.529,1027.283l-7.7-3.723,4.417-2.721,7.753,3.723Z" transform="translate(-4125.003 -1015.199)" fill="#fff"/></svg>
<span>Angular Material</span>
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"/></svg>
</a>
<a class="card" target="_blank" rel="noopener" href="https://blog.angular.io/">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M13.5.67s.74 2.65.74 4.8c0 2.06-1.35 3.73-3.41 3.73-2.07 0-3.63-1.67-3.63-3.73l.03-.36C5.21 7.51 4 10.62 4 14c0 4.42 3.58 8 8 8s8-3.58 8-8C20 8.61 17.41 3.8 13.5.67zM11.71 19c-1.78 0-3.22-1.4-3.22-3.14 0-1.62 1.05-2.76 2.81-3.12 1.77-.36 3.6-1.21 4.62-2.58.39 1.29.59 2.65.59 4.04 0 2.65-2.15 4.8-4.8 4.8z"/></svg>
<span>Angular Blog</span>
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"/></svg>
</a>
<a class="card" target="_blank" rel="noopener" href="https://angular.io/devtools/">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" enable-background="new 0 0 24 24" height="24px" viewBox="0 0 24 24" width="24px" fill="#000000"><g><rect fill="none" height="24" width="24"/></g><g><g><path d="M14.73,13.31C15.52,12.24,16,10.93,16,9.5C16,5.91,13.09,3,9.5,3S3,5.91,3,9.5C3,13.09,5.91,16,9.5,16 c1.43,0,2.74-0.48,3.81-1.27L19.59,21L21,19.59L14.73,13.31z M9.5,14C7.01,14,5,11.99,5,9.5S7.01,5,9.5,5S14,7.01,14,9.5 S11.99,14,9.5,14z"/><polygon points="10.29,8.44 9.5,6 8.71,8.44 6.25,8.44 8.26,10.03 7.49,12.5 9.5,10.97 11.51,12.5 10.74,10.03 12.75,8.44"/></g></g></svg>
<span>Angular DevTools</span>
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"/></svg>
</a>
</div>
<!-- Next Steps -->
<h2>Next Steps</h2>
<p>What do you want to do next with your app?</p>
<input type="hidden" #selection>
<div class="card-container">
<button class="card card-small" (click)="selection.value = 'component'" tabindex="0">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/></svg>
<span>New Component</span>
</button>
<button class="card card-small" (click)="selection.value = 'material'" tabindex="0">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/></svg>
<span>Angular Material</span>
</button>
<button class="card card-small" (click)="selection.value = 'pwa'" tabindex="0">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/></svg>
<span>Add PWA Support</span>
</button>
<button class="card card-small" (click)="selection.value = 'dependency'" tabindex="0">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/></svg>
<span>Add Dependency</span>
</button>
<button class="card card-small" (click)="selection.value = 'test'" tabindex="0">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/></svg>
<span>Run and Watch Tests</span>
</button>
<button class="card card-small" (click)="selection.value = 'build'" tabindex="0">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/></svg>
<span>Build for Production</span>
</button>
</div>
<!-- Terminal -->
<div class="terminal" [ngSwitch]="selection.value">
<pre *ngSwitchDefault>ng generate component xyz</pre>
<pre *ngSwitchCase="'material'">ng add @angular/material</pre>
<pre *ngSwitchCase="'pwa'">ng add @angular/pwa</pre>
<pre *ngSwitchCase="'dependency'">ng add _____</pre>
<pre *ngSwitchCase="'test'">ng test</pre>
<pre *ngSwitchCase="'build'">ng build</pre>
</div>
<!-- Links -->
<div class="card-container">
<a class="circle-link" title="Find a Local Meetup" href="https://www.meetup.com/find/?keywords=angular" target="_blank" rel="noopener">
<svg xmlns="http://www.w3.org/2000/svg" width="24.607" height="23.447" viewBox="0 0 24.607 23.447">
<title>Meetup Logo</title>
<path id="logo--mSwarm" d="M21.221,14.95A4.393,4.393,0,0,1,17.6,19.281a4.452,4.452,0,0,1-.8.069c-.09,0-.125.035-.154.117a2.939,2.939,0,0,1-2.506,2.091,2.868,2.868,0,0,1-2.248-.624.168.168,0,0,0-.245-.005,3.926,3.926,0,0,1-2.589.741,4.015,4.015,0,0,1-3.7-3.347,2.7,2.7,0,0,1-.043-.38c0-.106-.042-.146-.143-.166a3.524,3.524,0,0,1-1.516-.69A3.623,3.623,0,0,1,2.23,14.557a3.66,3.66,0,0,1,1.077-3.085.138.138,0,0,0,.026-.2,3.348,3.348,0,0,1-.451-1.821,3.46,3.46,0,0,1,2.749-3.28.44.44,0,0,0,.355-.281,5.072,5.072,0,0,1,3.863-3,5.028,5.028,0,0,1,3.555.666.31.31,0,0,0,.271.03A4.5,4.5,0,0,1,18.3,4.7a4.4,4.4,0,0,1,1.334,2.751,3.658,3.658,0,0,1,.022.706.131.131,0,0,0,.1.157,2.432,2.432,0,0,1,1.574,1.645,2.464,2.464,0,0,1-.7,2.616c-.065.064-.051.1-.014.166A4.321,4.321,0,0,1,21.221,14.95ZM13.4,14.607a2.09,2.09,0,0,0,1.409,1.982,4.7,4.7,0,0,0,1.275.221,1.807,1.807,0,0,0,.9-.151.542.542,0,0,0,.321-.545.558.558,0,0,0-.359-.534,1.2,1.2,0,0,0-.254-.078c-.262-.047-.526-.086-.787-.138a.674.674,0,0,1-.617-.75,3.394,3.394,0,0,1,.218-1.109c.217-.658.509-1.286.79-1.918a15.609,15.609,0,0,0,.745-1.86,1.95,1.95,0,0,0,.06-1.073,1.286,1.286,0,0,0-1.051-1.033,1.977,1.977,0,0,0-1.521.2.339.339,0,0,1-.446-.042c-.1-.092-.2-.189-.307-.284a1.214,1.214,0,0,0-1.643-.061,7.563,7.563,0,0,1-.614.512A.588.588,0,0,1,10.883,8c-.215-.115-.437-.215-.659-.316a2.153,2.153,0,0,0-.695-.248A2.091,2.091,0,0,0,7.541,8.562a9.915,9.915,0,0,0-.405.986c-.559,1.545-1.015,3.123-1.487,4.7a1.528,1.528,0,0,0,.634,1.777,1.755,1.755,0,0,0,1.5.211,1.35,1.35,0,0,0,.824-.858c.543-1.281,1.032-2.584,1.55-3.875.142-.355.28-.712.432-1.064a.548.548,0,0,1,.851-.24.622.622,0,0,1,.185.539,2.161,2.161,0,0,1-.181.621c-.337.852-.68,1.7-1.018,2.552a2.564,2.564,0,0,0-.173.528.624.624,0,0,0,.333.71,1.073,1.073,0,0,0,.814.034,1.22,1.22,0,0,0,.657-.655q.758-1.488,1.511-2.978.35-.687.709-1.37a1.073,1.073,0,0,1,.357-.434.43.43,0,0,1,.463-.016.373.373,0,0,1,.153.387.7.7,0,0,1-.057.236c-.065.157-.127.316-.2.469-.42.883-.846,1.763-1.262,2.648A2.463,2.463,0,0,0,13.4,14.607Zm5.888,6.508a1.09,1.09,0,0,0-2.179.006,1.09,1.09,0,0,0,2.179-.006ZM1.028,12.139a1.038,1.038,0,1,0,.01-2.075,1.038,1.038,0,0,0-.01,2.075ZM13.782.528a1.027,1.027,0,1,0-.011,2.055A1.027,1.027,0,0,0,13.782.528ZM22.21,6.95a.882.882,0,0,0-1.763.011A.882.882,0,0,0,22.21,6.95ZM4.153,4.439a.785.785,0,1,0,.787-.78A.766.766,0,0,0,4.153,4.439Zm8.221,18.22a.676.676,0,1,0-.677.666A.671.671,0,0,0,12.374,22.658ZM22.872,12.2a.674.674,0,0,0-.665.665.656.656,0,0,0,.655.643.634.634,0,0,0,.655-.644A.654.654,0,0,0,22.872,12.2ZM7.171-.123A.546.546,0,0,0,6.613.43a.553.553,0,1,0,1.106,0A.539.539,0,0,0,7.171-.123ZM24.119,9.234a.507.507,0,0,0-.493.488.494.494,0,0,0,.494.494.48.48,0,0,0,.487-.483A.491.491,0,0,0,24.119,9.234Zm-19.454,9.7a.5.5,0,0,0-.488-.488.491.491,0,0,0-.487.5.483.483,0,0,0,.491.479A.49.49,0,0,0,4.665,18.936Z" transform="translate(0 0.123)" fill="#f64060"/>
</svg>
</a>
<a class="circle-link" title="Join the Conversation on Discord" href="https://discord.gg/angular" target="_blank" rel="noopener">
<svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 245 240">
<title>Discord Logo</title>
<path d="M104.4 103.9c-5.7 0-10.2 5-10.2 11.1s4.6 11.1 10.2 11.1c5.7 0 10.2-5 10.2-11.1.1-6.1-4.5-11.1-10.2-11.1zM140.9 103.9c-5.7 0-10.2 5-10.2 11.1s4.6 11.1 10.2 11.1c5.7 0 10.2-5 10.2-11.1s-4.5-11.1-10.2-11.1z"/>
<path d="M189.5 20h-134C44.2 20 35 29.2 35 40.6v135.2c0 11.4 9.2 20.6 20.5 20.6h113.4l-5.3-18.5 12.8 11.9 12.1 11.2 21.5 19V40.6c0-11.4-9.2-20.6-20.5-20.6zm-38.6 130.6s-3.6-4.3-6.6-8.1c13.1-3.7 18.1-11.9 18.1-11.9-4.1 2.7-8 4.6-11.5 5.9-5 2.1-9.8 3.5-14.5 4.3-9.6 1.8-18.4 1.3-25.9-.1-5.7-1.1-10.6-2.7-14.7-4.3-2.3-.9-4.8-2-7.3-3.4-.3-.2-.6-.3-.9-.5-.2-.1-.3-.2-.4-.3-1.8-1-2.8-1.7-2.8-1.7s4.8 8 17.5 11.8c-3 3.8-6.7 8.3-6.7 8.3-22.1-.7-30.5-15.2-30.5-15.2 0-32.2 14.4-58.3 14.4-58.3 14.4-10.8 28.1-10.5 28.1-10.5l1 1.2c-18 5.2-26.3 13.1-26.3 13.1s2.2-1.2 5.9-2.9c10.7-4.7 19.2-6 22.7-6.3.6-.1 1.1-.2 1.7-.2 6.1-.8 13-1 20.2-.2 9.5 1.1 19.7 3.9 30.1 9.6 0 0-7.9-7.5-24.9-12.7l1.4-1.6s13.7-.3 28.1 10.5c0 0 14.4 26.1 14.4 58.3 0 0-8.5 14.5-30.6 15.2z"/>
</svg>
</a>
</div>
<!-- Footer -->
<footer>
Love Angular?&nbsp;
<a href="https://github.com/angular/angular" target="_blank" rel="noopener"> Give our repo a star.
<div class="github-star-badge">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M0 0h24v24H0z" fill="none"/><path d="M12 17.27L18.18 21l-1.64-7.03L22 9.24l-7.19-.61L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21z"/></svg>
Star
</div>
</a>
<a href="https://github.com/angular/angular" target="_blank" rel="noopener">
<svg class="material-icons" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z" fill="#1976d2"/><path d="M0 0h24v24H0z" fill="none"/></svg>
</a>
</footer>
<svg id="clouds" xmlns="http://www.w3.org/2000/svg" width="2611.084" height="485.677" viewBox="0 0 2611.084 485.677">
<title>Gray Clouds Background</title>
<path id="Path_39" data-name="Path 39" d="M2379.709,863.793c10-93-77-171-168-149-52-114-225-105-264,15-75,3-140,59-152,133-30,2.83-66.725,9.829-93.5,26.25-26.771-16.421-63.5-23.42-93.5-26.25-12-74-77-130-152-133-39-120-212-129-264-15-54.084-13.075-106.753,9.173-138.488,48.9-31.734-39.726-84.4-61.974-138.487-48.9-52-114-225-105-264,15a162.027,162.027,0,0,0-103.147,43.044c-30.633-45.365-87.1-72.091-145.206-58.044-52-114-225-105-264,15-75,3-140,59-152,133-53,5-127,23-130,83-2,42,35,72,70,86,49,20,106,18,157,5a165.625,165.625,0,0,0,120,0c47,94,178,113,251,33,61.112,8.015,113.854-5.72,150.492-29.764a165.62,165.62,0,0,0,110.861-3.236c47,94,178,113,251,33,31.385,4.116,60.563,2.495,86.487-3.311,25.924,5.806,55.1,7.427,86.488,3.311,73,80,204,61,251-33a165.625,165.625,0,0,0,120,0c51,13,108,15,157-5a147.188,147.188,0,0,0,33.5-18.694,147.217,147.217,0,0,0,33.5,18.694c49,20,106,18,157,5a165.625,165.625,0,0,0,120,0c47,94,178,113,251,33C2446.709,1093.793,2554.709,922.793,2379.709,863.793Z" transform="translate(142.69 -634.312)" fill="#eee"/>
</svg>
</div>
<!-- * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * -->
<!-- * * * * * * * * * * * The content above * * * * * * * * * * * -->
<!-- * * * * * * * * * * is only a placeholder * * * * * * * * * * -->
<!-- * * * * * * * * * * and can be replaced. * * * * * * * * * * * -->
<!-- * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * -->
<!-- * * * * * * * * * * End of Placeholder * * * * * * * * * * * -->
<!-- * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * -->
<router-outlet></router-outlet>

View File

@ -1,15 +1,20 @@
import { HttpClientModule } from '@angular/common/http';
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser'; import { BrowserModule } from '@angular/platform-browser';
import { AppRoutingModule } from './app-routing.module'; import { AppRoutingModule } from './app-routing.module';
import { AppComponent } from './app.component'; import { AppComponent } from './app.component';
import { CivilizationsComponent } from './pages/civilizations/civilizations.component';
import { EntityComponent } from './pages/entity/entity.component';
@NgModule({ @NgModule({
declarations: [ declarations: [
AppComponent AppComponent,
CivilizationsComponent,
EntityComponent
], ],
imports: [ imports: [
BrowserModule, BrowserModule,
HttpClientModule,
AppRoutingModule AppRoutingModule
], ],
providers: [], providers: [],

View File

@ -0,0 +1,46 @@
import { HttpClient } from '@angular/common/http';
import { Injectable } from '@angular/core';
import { ActivatedRouteSnapshot, Resolve, RouterStateSnapshot } from '@angular/router';
import { firstValueFrom, Observable } from 'rxjs';
import { Entity } from './types';
@Injectable({
providedIn: 'root'
})
export class EntityService {
constructor(private http: HttpClient) { }
getAll(): Promise<Entity[]> {
return firstValueFrom(this.http.get<Entity[]>("./api/entity"));
}
getOne(id: string | number): Promise<Entity> {
return firstValueFrom(this.http.get<Entity>("./api/entity/" + id));
}
}
@Injectable({ providedIn: 'root' })
export class EntitiesResolver implements Resolve<Entity[]> {
constructor(private service: EntityService) { }
resolve(
route: ActivatedRouteSnapshot,
state: RouterStateSnapshot
): Observable<Entity[]> | Promise<Entity[]> | Entity[] {
return this.service.getAll();
}
}
@Injectable({ providedIn: 'root' })
export class EntityResolver implements Resolve<Entity> {
constructor(private service: EntityService) { }
resolve(
route: ActivatedRouteSnapshot,
state: RouterStateSnapshot
): Observable<Entity> | Promise<Entity> | Entity {
return this.service.getOne(route.paramMap.get('id')!);
}
}

View File

@ -0,0 +1,4 @@
<p>civilizations works!</p>
<ul>
<li *ngFor="let civ of civilizations"><a [routerLink]="['entity', civ.id]">{{civ.name}}</a></li>
</ul>

View File

@ -0,0 +1,25 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { CivilizationsComponent } from './civilizations.component';
describe('CivilizationsComponent', () => {
let component: CivilizationsComponent;
let fixture: ComponentFixture<CivilizationsComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ CivilizationsComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(CivilizationsComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@ -0,0 +1,21 @@
import { Component, OnInit } from '@angular/core';
import { ActivatedRoute } from '@angular/router';
import { Entity } from 'src/app/types';
@Component({
selector: 'app-civilizations',
templateUrl: './civilizations.component.html',
styleUrls: ['./civilizations.component.scss']
})
export class CivilizationsComponent implements OnInit {
civilizations: Entity[] = [];
constructor(private route: ActivatedRoute) { }
ngOnInit(): void {
this.civilizations = this.route.snapshot.data['civilizations'];
this.civilizations = this.civilizations.filter(c => c.name.length > 0)
}
}

View File

@ -0,0 +1 @@
<h1>{{entity.name}}</h1>

View File

@ -0,0 +1,25 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { EntityComponent } from './entity.component';
describe('EntityComponent', () => {
let component: EntityComponent;
let fixture: ComponentFixture<EntityComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ EntityComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(EntityComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@ -0,0 +1,20 @@
import { Component, OnInit } from '@angular/core';
import { ActivatedRoute } from '@angular/router';
import { Entity } from 'src/app/types';
@Component({
selector: 'app-entity',
templateUrl: './entity.component.html',
styleUrls: ['./entity.component.scss']
})
export class EntityComponent implements OnInit {
entity!: Entity;
constructor(private route: ActivatedRoute) { }
ngOnInit(): void {
this.entity = this.route.snapshot.data['entity'];
}
}

View File

@ -0,0 +1,4 @@
export interface Entity {
id: number;
name: string;
}

View File

@ -0,0 +1,6 @@
{
"/api": {
"target": "http://localhost:8080",
"secure": false
}
}

1
go.mod
View File

@ -4,5 +4,6 @@ go 1.18
require ( require (
github.com/gorilla/mux v1.8.0 github.com/gorilla/mux v1.8.0
github.com/iancoleman/strcase v0.2.0
github.com/sa-/slicefunk v0.1.2 github.com/sa-/slicefunk v0.1.2
) )

2
go.sum
View File

@ -1,4 +1,6 @@
github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI= github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI=
github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So=
github.com/iancoleman/strcase v0.2.0 h1:05I4QRnGpI0m37iZQRuskXh+w77mr6Z41lwQzuHLwW0=
github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho=
github.com/sa-/slicefunk v0.1.2 h1:uTQiyKCTwc0RKNxXmG6NNlm3J1LF2hKmLgtf9kRAld4= github.com/sa-/slicefunk v0.1.2 h1:uTQiyKCTwc0RKNxXmG6NNlm3J1LF2hKmLgtf9kRAld4=
github.com/sa-/slicefunk v0.1.2/go.mod h1:k0abNpV9EW8LIPl2+Hc9RiKsojKmsUhNNGFyMpjMTCI= github.com/sa-/slicefunk v0.1.2/go.mod h1:k0abNpV9EW8LIPl2+Hc9RiKsojKmsUhNNGFyMpjMTCI=

BIN
legendsbrowser Executable file

Binary file not shown.

207
main.go
View File

@ -1,202 +1,41 @@
package main package main
import ( import (
"encoding/json"
"encoding/xml"
"fmt" "fmt"
"io/ioutil" "legendsbrowser/model"
"legendsbrowser/server"
"net/http" "net/http"
"os"
"strconv"
"github.com/gorilla/mux" "github.com/gorilla/mux"
sf "github.com/sa-/slicefunk"
) )
type World struct { var world model.World
XMLName xml.Name `xml:"df_world"`
Name string `xml:"name"`
AltName string `xml:"altname"`
Regions []Region `xml:"regions>region"`
UndergroundRegions []UndergroundRegion `xml:"underground_regions>underground_region"`
Landmasses []Landmass `xml:"landmasses>landmass"`
Sites []Site `xml:"sites>site"`
WorldConstructions []WorldConstruction `xml:"world_constructions>world_construction"`
Artifacts []Artifact `xml:"artifacts>artifact"`
HistoricalFigures []HistoricalFigure `xml:"historical_figures>historical_figure"`
HistoricalEvents []HistoricalEvent `xml:"historical_events>historical_event"`
HistoricalEventCollections []HistoricalEventCollection `xml:"historical_event_collections>historical_event_collection"`
}
type NamedObject struct {
Id int `xml:"id" json:"id"`
Name string `xml:"name" json:"name"`
}
func (r NamedObject) id() int { return r.Id }
func (r NamedObject) name() string { return r.Name }
type Named interface {
id() int
name() string
}
type Region struct {
XMLName xml.Name `xml:"region" json:"-"`
NamedObject
Type string `xml:"type" json:"type"`
}
type UndergroundRegion struct {
XMLName xml.Name `xml:"underground_region"`
NamedObject
Type string `xml:"type" json:"type"`
}
type Landmass struct {
XMLName xml.Name `xml:"landmass"`
NamedObject
}
type Site struct {
XMLName xml.Name `xml:"site" json:"-"`
NamedObject
Type string `xml:"type" json:"type"`
Coords string `xml:"coords" json:"coords"`
Rectangle string `xml:"rectangle" json:"rectangle"`
Structures []Structure `xml:"structures>structure" json:"structures"`
}
// func (obj Site) id() int { return obj.Id }
// func (obj Site) name() string { return obj.Name }
type Structure struct {
XMLName xml.Name `xml:"structure" json:"-"`
LocalId int `xml:"local_id" json:"localId"`
Name string `xml:"name" json:"name"`
Type string `xml:"type" json:"type"`
}
type WorldConstruction struct {
XMLName xml.Name `xml:"world_construction"`
NamedObject
}
type Artifact struct {
XMLName xml.Name `xml:"artifact"`
NamedObject
SiteId int `xml:"site_id" json:"siteId"`
}
type Element struct {
XMLName xml.Name
Value string `xml:",innerxml"`
}
type HistoricalFigure struct {
XMLName xml.Name `xml:"historical_figure"`
NamedObject
Race string `xml:"race" json:"race"`
Caste string `xml:"caste" json:"caste"`
Other []Element `xml:",any" json:"-"`
}
type HistoricalEvent struct {
XMLName xml.Name `xml:"historical_event"`
Id int `xml:"id"`
Year int `xml:"year"`
Seconds int `xml:"seconds72"`
Type string `xml:"type"`
}
type HistoricalEventCollection struct {
XMLName xml.Name `xml:"historical_event_collection"`
NamedObject
StartYear int `xml:"year"`
StartSeconds int `xml:"seconds72"`
EndYear int `xml:"end_year"`
EndSeconds int `xml:"end_seconds72"`
Type string `xml:"type" json:"type"`
EventIds []int `xml:"event" json:"eventIds"`
}
var world World
func main() { func main() {
fmt.Println("Hallo Welt!") fmt.Println("Hallo Welt!")
xmlFile, err := os.Open("region1-00152-01-01-legends.xml") world.Load("region1-00152-01-01-legends.xml")
if err != nil { world.Process()
fmt.Println(err)
}
fmt.Println("Successfully Opened users.xml") model.ListOtherElements(&world.HistoricalEvents)
defer xmlFile.Close() // listOtherElements(&world.HistoricalFigures)
byteValue, _ := ioutil.ReadAll(xmlFile)
fmt.Println(len(byteValue))
err = xml.Unmarshal(byteValue, &world)
if err != nil {
fmt.Println(err)
}
fmt.Printf("Sites: %d\n", len(world.Sites))
fmt.Printf("artifacts: %d\n", len(world.Artifacts))
fmt.Printf("events: %d\n", len(world.HistoricalEvents))
fmt.Printf("collections: %d\n", len(world.HistoricalEventCollections))
fmt.Printf(" events: %v\n", len(world.HistoricalEventCollections[0].EventIds))
fmt.Printf("figures: %d\n", len(world.HistoricalFigures))
fmt.Printf(" len: %d\n", len(world.HistoricalFigures[0].Other))
// fmt.Printf(" other: %v\n", world.HistoricalFigures[0].Other)
// for i := 0; i < len(world.Regions); i++ {
// fmt.Println("Regions Name: " + world.Regions[i].Name)
// }
// for i := 0; i < len(world.Sites); i++ {
// fmt.Println("Sites Name: " + world.Sites[i].Name)
// }
router := mux.NewRouter().StrictSlash(true) router := mux.NewRouter().StrictSlash(true)
registerResource(router, "region", world.Regions)
registerResource(router, "undergroundRegion", world.UndergroundRegions) server.RegisterResource(router, "region", world.RegionMap)
registerResource(router, "landmass", world.Landmasses) server.RegisterResource(router, "undergroundRegion", world.UndergroundRegionMap)
registerResource(router, "site", world.Sites) server.RegisterResource(router, "landmass", world.LandmassMap)
registerResource(router, "worldConstruction", world.WorldConstructions) server.RegisterResource(router, "site", world.SiteMap)
registerResource(router, "artifact", world.Artifacts) server.RegisterResource(router, "worldConstruction", world.WorldConstructionMap)
registerResource(router, "hf", world.HistoricalFigures) server.RegisterResource(router, "artifact", world.ArtifactMap)
registerResource(router, "collection", world.HistoricalEventCollections) server.RegisterResource(router, "hf", world.HistoricalFigureMap)
server.RegisterResource(router, "collection", world.HistoricalEventCollectionMap)
server.RegisterResource(router, "entity", world.EntityMap)
server.RegisterResource(router, "event", world.HistoricalEventMap)
spa := server.SpaHandler{StaticPath: "frontend/dist/legendsbrowser", IndexPath: "index.html"}
router.PathPrefix("/").Handler(spa)
fmt.Println("Serving at :8080")
http.ListenAndServe(":8080", router) http.ListenAndServe(":8080", router)
} }
type Info struct {
Id int `json:"id"`
Name string `json:"name"`
}
func registerResource[T Named](router *mux.Router, resourceName string, resources []T) {
list := func(w http.ResponseWriter, r *http.Request) {
values := sf.Map(resources, func(item T) *Info { return &Info{Id: item.id(), Name: item.name()} })
json.NewEncoder(w).Encode(values)
}
get := func(w http.ResponseWriter, r *http.Request) {
id, err := strconv.Atoi(mux.Vars(r)["id"])
if err != nil {
fmt.Println(err)
}
for _, item := range resources {
if item.id() == id {
json.NewEncoder(w).Encode(item)
}
}
}
router.HandleFunc(fmt.Sprintf("/%s", resourceName), list).Methods("GET")
router.HandleFunc(fmt.Sprintf("/%s/{id}", resourceName), get).Methods("GET")
}

84
model/analyze.go Normal file
View File

@ -0,0 +1,84 @@
package model
import (
"fmt"
"legendsbrowser/util"
"sort"
"strconv"
"strings"
"github.com/iancoleman/strcase"
)
func ListOtherElements[T TypedOthers](items *[]T) {
m := make(map[string]map[string]bool)
cantInt := make(map[string]bool)
isObj := make(map[string]bool)
isMultiple := make(map[string]bool)
for _, item := range *items {
found := make(map[string]bool)
for _, el := range item.Others() {
if !m[el.XMLName.Local][item.Type()] {
if m[el.XMLName.Local] == nil {
m[el.XMLName.Local] = map[string]bool{}
}
m[el.XMLName.Local][item.Type()] = true
}
_, err := strconv.Atoi(el.Value)
if err != nil {
cantInt[el.XMLName.Local] = true
}
if strings.Contains(el.Value, "<") {
isObj[el.XMLName.Local] = true
}
if found[el.XMLName.Local] {
isMultiple[el.XMLName.Local] = true
}
found[el.XMLName.Local] = true
}
}
ks := util.Keys(m)
sort.Strings(ks)
for _, k := range ks {
events := util.Keys(m[k])
sort.Strings(events)
// fmt.Println(strconv.FormatBool(cantInt[k]) + " - " + k + ": " + strings.Join(events, ", "))
var mult string
if isMultiple[k] {
mult = "[]"
} else {
mult = ""
}
if isObj[k] {
fmt.Printf("// %s object\n", k)
} else if cantInt[k] {
fmt.Printf("%s *%sstring `xml:\"%s\" json:\"%s,omitempty\"`\n", strcase.ToCamel(k), mult, k, strcase.ToLowerCamel(k))
} else {
var types []string
if util.ContainsAny(k, "entity_id", "enid", "civ_id", "entity_1", "entity_2") {
types = append(types, "entity")
}
if util.ContainsAny(k, "site_id") {
types = append(types, "site")
}
if util.ContainsAny(k, "structure_id") {
types = append(types, "structure")
}
if util.ContainsAny(k, "hfid", "hist_figure_id", "hist_fig_id") {
types = append(types, "hf")
}
if util.ContainsAny(k, "wcid", "wc_id") {
types = append(types, "wc")
}
if util.ContainsAny(k, "artifact_id") {
types = append(types, "artifact")
}
typestr := strings.Join(types, ",")
if typestr != "" {
typestr = fmt.Sprintf(" legend:\"%s\"", typestr)
}
fmt.Printf("%s *%sint `xml:\"%s\" json:\"%s,omitempty\"%s`\n", strcase.ToCamel(k), mult, k, strcase.ToLowerCamel(k), typestr)
}
}
}

245
model/events.go Normal file
View File

@ -0,0 +1,245 @@
package model
import "encoding/xml"
type HistoricalEvent struct {
XMLName xml.Name `xml:"historical_event" json:"-"`
Id_ int `xml:"id" json:"id"`
Year int `xml:"year" json:"year"`
Seconds int `xml:"seconds72" json:"seconds72"`
TypedObject
ASupportMercEnid *int `xml:"a_support_merc_enid" json:"aSupportMercEnid,omitempty" legend:"entity"`
AccountShift *int `xml:"account_shift" json:"accountShift,omitempty"`
AcquirerEnid *int `xml:"acquirer_enid" json:"acquirerEnid,omitempty" legend:"entity"`
AcquirerHfid *int `xml:"acquirer_hfid" json:"acquirerHfid,omitempty" legend:"hf"`
Action *string `xml:"action" json:"action,omitempty"`
ActorHfid *int `xml:"actor_hfid" json:"actorHfid,omitempty" legend:"hf"`
AgreementId *int `xml:"agreement_id" json:"agreementId,omitempty"`
Allotment *int `xml:"allotment" json:"allotment,omitempty"`
AllotmentIndex *int `xml:"allotment_index" json:"allotmentIndex,omitempty"`
AllyDefenseBonus *int `xml:"ally_defense_bonus" json:"allyDefenseBonus,omitempty"`
AppointerHfid *int `xml:"appointer_hfid" json:"appointerHfid,omitempty" legend:"hf"`
ArrestingEnid *int `xml:"arresting_enid" json:"arrestingEnid,omitempty" legend:"entity"`
ArtifactId *int `xml:"artifact_id" json:"artifactId,omitempty" legend:"artifact"`
AttackerCivId *int `xml:"attacker_civ_id" json:"attackerCivId,omitempty" legend:"entity"`
AttackerGeneralHfid *int `xml:"attacker_general_hfid" json:"attackerGeneralHfid,omitempty" legend:"hf"`
AttackerHfid *int `xml:"attacker_hfid" json:"attackerHfid,omitempty" legend:"hf"`
AttackerMercEnid *int `xml:"attacker_merc_enid" json:"attackerMercEnid,omitempty" legend:"entity"`
BodyState *string `xml:"body_state" json:"bodyState,omitempty"`
BuilderHfid *int `xml:"builder_hfid" json:"builderHfid,omitempty" legend:"hf"`
BuildingProfileId *int `xml:"building_profile_id" json:"buildingProfileId,omitempty"`
Cause *string `xml:"cause" json:"cause,omitempty"`
ChangeeHfid *int `xml:"changee_hfid" json:"changeeHfid,omitempty" legend:"hf"`
ChangerHfid *int `xml:"changer_hfid" json:"changerHfid,omitempty" legend:"hf"`
Circumstance *string `xml:"circumstance" json:"circumstance,omitempty"`
CircumstanceId *int `xml:"circumstance_id" json:"circumstanceId,omitempty"`
CivEntityId *int `xml:"civ_entity_id" json:"civEntityId,omitempty" legend:"entity"`
CivId *int `xml:"civ_id" json:"civId,omitempty" legend:"entity"`
Claim *string `xml:"claim" json:"claim,omitempty"`
CoconspiratorBonus *int `xml:"coconspirator_bonus" json:"coconspiratorBonus,omitempty"`
CompetitorHfid *[]int `xml:"competitor_hfid" json:"competitorHfid,omitempty" legend:"hf"`
ConfessedAfterApbArrestEnid *int `xml:"confessed_after_apb_arrest_enid" json:"confessedAfterApbArrestEnid,omitempty" legend:"entity"`
ConspiratorHfid *[]int `xml:"conspirator_hfid" json:"conspiratorHfid,omitempty" legend:"hf"`
ContactHfid *int `xml:"contact_hfid" json:"contactHfid,omitempty" legend:"hf"`
ConvictIsContact *string `xml:"convict_is_contact" json:"convictIsContact,omitempty"`
ConvictedHfid *int `xml:"convicted_hfid" json:"convictedHfid,omitempty" legend:"hf"`
ConvicterEnid *int `xml:"convicter_enid" json:"convicterEnid,omitempty" legend:"entity"`
Coords *string `xml:"coords" json:"coords,omitempty"`
CorruptConvicterHfid *int `xml:"corrupt_convicter_hfid" json:"corruptConvicterHfid,omitempty" legend:"hf"`
CorruptorHfid *int `xml:"corruptor_hfid" json:"corruptorHfid,omitempty" legend:"hf"`
CorruptorIdentity *int `xml:"corruptor_identity" json:"corruptorIdentity,omitempty"`
CorruptorSeenAs *string `xml:"corruptor_seen_as" json:"corruptorSeenAs,omitempty"`
CreatorHfid *int `xml:"creator_hfid" json:"creatorHfid,omitempty" legend:"hf"`
Crime *string `xml:"crime" json:"crime,omitempty"`
DSupportMercEnid *int `xml:"d_support_merc_enid" json:"dSupportMercEnid,omitempty" legend:"entity"`
DeathPenalty *string `xml:"death_penalty" json:"deathPenalty,omitempty"`
DefenderCivId *int `xml:"defender_civ_id" json:"defenderCivId,omitempty" legend:"entity"`
DefenderGeneralHfid *int `xml:"defender_general_hfid" json:"defenderGeneralHfid,omitempty" legend:"hf"`
DefenderMercEnid *int `xml:"defender_merc_enid" json:"defenderMercEnid,omitempty" legend:"entity"`
Delegated *string `xml:"delegated" json:"delegated,omitempty"`
DestEntityId *int `xml:"dest_entity_id" json:"destEntityId,omitempty" legend:"entity"`
DestSiteId *int `xml:"dest_site_id" json:"destSiteId,omitempty" legend:"site"`
DestStructureId *int `xml:"dest_structure_id" json:"destStructureId,omitempty" legend:"structure"`
DestroyedStructureId *int `xml:"destroyed_structure_id" json:"destroyedStructureId,omitempty" legend:"structure"`
DestroyerEnid *int `xml:"destroyer_enid" json:"destroyerEnid,omitempty" legend:"entity"`
Detected *string `xml:"detected" json:"detected,omitempty"`
DidNotRevealAllInInterrogation *string `xml:"did_not_reveal_all_in_interrogation" json:"didNotRevealAllInInterrogation,omitempty"`
Dispute *string `xml:"dispute" json:"dispute,omitempty"`
DoerHfid *int `xml:"doer_hfid" json:"doerHfid,omitempty" legend:"hf"`
Entity1 *int `xml:"entity_1" json:"entity1,omitempty" legend:"entity"`
Entity2 *int `xml:"entity_2" json:"entity2,omitempty" legend:"entity"`
EntityId *int `xml:"entity_id" json:"entityId,omitempty" legend:"entity"`
EntityId1 *int `xml:"entity_id_1" json:"entityId1,omitempty" legend:"entity"`
EntityId2 *int `xml:"entity_id_2" json:"entityId2,omitempty" legend:"entity"`
Exiled *string `xml:"exiled" json:"exiled,omitempty"`
ExpelledCreature *[]int `xml:"expelled_creature" json:"expelledCreature,omitempty"`
ExpelledHfid *[]int `xml:"expelled_hfid" json:"expelledHfid,omitempty" legend:"hf"`
ExpelledNumber *[]int `xml:"expelled_number" json:"expelledNumber,omitempty"`
ExpelledPopId *[]int `xml:"expelled_pop_id" json:"expelledPopId,omitempty"`
FailedJudgmentTest *string `xml:"failed_judgment_test" json:"failedJudgmentTest,omitempty"`
FeatureLayerId *int `xml:"feature_layer_id" json:"featureLayerId,omitempty"`
First *string `xml:"first" json:"first,omitempty"`
FooledHfid *int `xml:"fooled_hfid" json:"fooledHfid,omitempty" legend:"hf"`
FormId *int `xml:"form_id" json:"formId,omitempty"`
FramerHfid *int `xml:"framer_hfid" json:"framerHfid,omitempty" legend:"hf"`
FromOriginal *string `xml:"from_original" json:"fromOriginal,omitempty"`
GamblerHfid *int `xml:"gambler_hfid" json:"gamblerHfid,omitempty" legend:"hf"`
GiverEntityId *int `xml:"giver_entity_id" json:"giverEntityId,omitempty" legend:"entity"`
GiverHistFigureId *int `xml:"giver_hist_figure_id" json:"giverHistFigureId,omitempty" legend:"hf"`
Group1Hfid *int `xml:"group_1_hfid" json:"group1Hfid,omitempty" legend:"hf"`
Group2Hfid *[]int `xml:"group_2_hfid" json:"group2Hfid,omitempty" legend:"hf"`
GroupHfid *[]int `xml:"group_hfid" json:"groupHfid,omitempty" legend:"hf"`
HeldFirmInInterrogation *string `xml:"held_firm_in_interrogation" json:"heldFirmInInterrogation,omitempty"`
HfRep1Of2 *string `xml:"hf_rep_1_of_2" json:"hfRep1Of2,omitempty"`
HfRep2Of1 *string `xml:"hf_rep_2_of_1" json:"hfRep2Of1,omitempty"`
Hfid *[]int `xml:"hfid" json:"hfid,omitempty" legend:"hf"`
Hfid1 *int `xml:"hfid1" json:"hfid1,omitempty" legend:"hf"`
Hfid2 *int `xml:"hfid2" json:"hfid2,omitempty" legend:"hf"`
HfidTarget *int `xml:"hfid_target" json:"hfidTarget,omitempty" legend:"hf"`
HistFigId *int `xml:"hist_fig_id" json:"histFigId,omitempty" legend:"hf"`
HistFigureId *int `xml:"hist_figure_id" json:"histFigureId,omitempty" legend:"hf"`
HonorId *int `xml:"honor_id" json:"honorId,omitempty"`
IdentityId *int `xml:"identity_id" json:"identityId,omitempty" legend:"entity"`
IdentityId1 *int `xml:"identity_id1" json:"identityId1,omitempty" legend:"entity"`
IdentityId2 *int `xml:"identity_id2" json:"identityId2,omitempty" legend:"entity"`
ImplicatedHfid *[]int `xml:"implicated_hfid" json:"implicatedHfid,omitempty" legend:"hf"`
Inherited *string `xml:"inherited" json:"inherited,omitempty"`
InitiatingEnid *int `xml:"initiating_enid" json:"initiatingEnid,omitempty" legend:"entity"`
InstigatorHfid *int `xml:"instigator_hfid" json:"instigatorHfid,omitempty" legend:"hf"`
Interaction *string `xml:"interaction" json:"interaction,omitempty"`
InterrogatorHfid *int `xml:"interrogator_hfid" json:"interrogatorHfid,omitempty" legend:"hf"`
JoinEntityId *int `xml:"join_entity_id" json:"joinEntityId,omitempty" legend:"entity"`
JoinedEntityId *int `xml:"joined_entity_id" json:"joinedEntityId,omitempty" legend:"entity"`
JoinerEntityId *int `xml:"joiner_entity_id" json:"joinerEntityId,omitempty" legend:"entity"`
JoiningEnid *[]int `xml:"joining_enid" json:"joiningEnid,omitempty" legend:"entity"`
Knowledge *string `xml:"knowledge" json:"knowledge,omitempty"`
LastOwnerHfid *int `xml:"last_owner_hfid" json:"lastOwnerHfid,omitempty" legend:"hf"`
LeaderHfid *int `xml:"leader_hfid" json:"leaderHfid,omitempty" legend:"hf"`
Link *string `xml:"link" json:"link,omitempty"`
LureHfid *int `xml:"lure_hfid" json:"lureHfid,omitempty" legend:"hf"`
MasterWcid *int `xml:"master_wcid" json:"masterWcid,omitempty" legend:"wc"`
Method *string `xml:"method" json:"method,omitempty"`
Modification *string `xml:"modification" json:"modification,omitempty"`
ModifierHfid *int `xml:"modifier_hfid" json:"modifierHfid,omitempty" legend:"hf"`
Mood *string `xml:"mood" json:"mood,omitempty"`
NameOnly *string `xml:"name_only" json:"nameOnly,omitempty"`
NewAbId *int `xml:"new_ab_id" json:"newAbId,omitempty"`
NewAccount *int `xml:"new_account" json:"newAccount,omitempty"`
NewCaste *string `xml:"new_caste" json:"newCaste,omitempty"`
NewEquipmentLevel *int `xml:"new_equipment_level" json:"newEquipmentLevel,omitempty"`
NewLeaderHfid *int `xml:"new_leader_hfid" json:"newLeaderHfid,omitempty" legend:"hf"`
NewRace *string `xml:"new_race" json:"newRace,omitempty"`
NewSiteCivId *int `xml:"new_site_civ_id" json:"newSiteCivId,omitempty" legend:"entity"`
OccasionId *int `xml:"occasion_id" json:"occasionId,omitempty"`
OldAbId *int `xml:"old_ab_id" json:"oldAbId,omitempty"`
OldAccount *int `xml:"old_account" json:"oldAccount,omitempty"`
OldCaste *string `xml:"old_caste" json:"oldCaste,omitempty"`
OldRace *string `xml:"old_race" json:"oldRace,omitempty"`
OverthrownHfid *int `xml:"overthrown_hfid" json:"overthrownHfid,omitempty" legend:"hf"`
PartialIncorporation *string `xml:"partial_incorporation" json:"partialIncorporation,omitempty"`
PersecutorEnid *int `xml:"persecutor_enid" json:"persecutorEnid,omitempty" legend:"entity"`
PersecutorHfid *int `xml:"persecutor_hfid" json:"persecutorHfid,omitempty" legend:"hf"`
PlotterHfid *int `xml:"plotter_hfid" json:"plotterHfid,omitempty" legend:"hf"`
PopFlid *int `xml:"pop_flid" json:"popFlid,omitempty"`
PopNumberMoved *int `xml:"pop_number_moved" json:"popNumberMoved,omitempty"`
PopRace *int `xml:"pop_race" json:"popRace,omitempty"`
PopSrid *int `xml:"pop_srid" json:"popSrid,omitempty"`
PosTakerHfid *int `xml:"pos_taker_hfid" json:"posTakerHfid,omitempty" legend:"hf"`
PositionId *int `xml:"position_id" json:"positionId,omitempty"`
PositionProfileId *int `xml:"position_profile_id" json:"positionProfileId,omitempty"`
PrisonMonths *int `xml:"prison_months" json:"prisonMonths,omitempty"`
ProductionZoneId *int `xml:"production_zone_id" json:"productionZoneId,omitempty"`
PromiseToHfid *int `xml:"promise_to_hfid" json:"promiseToHfid,omitempty" legend:"hf"`
PropertyConfiscatedFromHfid *[]int `xml:"property_confiscated_from_hfid" json:"propertyConfiscatedFromHfid,omitempty" legend:"hf"`
PurchasedUnowned *string `xml:"purchased_unowned" json:"purchasedUnowned,omitempty"`
Quality *int `xml:"quality" json:"quality,omitempty"`
Reason *string `xml:"reason" json:"reason,omitempty"`
ReasonId *int `xml:"reason_id" json:"reasonId,omitempty"`
RebuiltRuined *string `xml:"rebuilt_ruined" json:"rebuiltRuined,omitempty"`
ReceiverEntityId *int `xml:"receiver_entity_id" json:"receiverEntityId,omitempty" legend:"entity"`
ReceiverHistFigureId *int `xml:"receiver_hist_figure_id" json:"receiverHistFigureId,omitempty" legend:"hf"`
Relationship *string `xml:"relationship" json:"relationship,omitempty"`
RelevantEntityId *int `xml:"relevant_entity_id" json:"relevantEntityId,omitempty" legend:"entity"`
RelevantIdForMethod *int `xml:"relevant_id_for_method" json:"relevantIdForMethod,omitempty"`
RelevantPositionProfileId *int `xml:"relevant_position_profile_id" json:"relevantPositionProfileId,omitempty"`
ReligionId *int `xml:"religion_id" json:"religionId,omitempty"`
ResidentCivId *int `xml:"resident_civ_id" json:"residentCivId,omitempty" legend:"entity"`
Return *string `xml:"return" json:"return,omitempty"`
ScheduleId *int `xml:"schedule_id" json:"scheduleId,omitempty"`
SecretGoal *string `xml:"secret_goal" json:"secretGoal,omitempty"`
SeekerHfid *int `xml:"seeker_hfid" json:"seekerHfid,omitempty" legend:"hf"`
ShrineAmountDestroyed *int `xml:"shrine_amount_destroyed" json:"shrineAmountDestroyed,omitempty"`
SiteCivId *int `xml:"site_civ_id" json:"siteCivId,omitempty" legend:"entity"`
SiteEntityId *int `xml:"site_entity_id" json:"siteEntityId,omitempty" legend:"entity"`
SiteHfid *int `xml:"site_hfid" json:"siteHfid,omitempty" legend:"hf"`
SiteId *int `xml:"site_id" json:"siteId,omitempty" legend:"site"`
SiteId1 *int `xml:"site_id1" json:"siteId1,omitempty" legend:"site"`
SiteId2 *int `xml:"site_id2" json:"siteId2,omitempty" legend:"site"`
SiteId_1 *int `xml:"site_id_1" json:"siteId_1,omitempty" legend:"site"`
SiteId_2 *int `xml:"site_id_2" json:"siteId_2,omitempty" legend:"site"`
SitePropertyId *int `xml:"site_property_id" json:"sitePropertyId,omitempty"`
Situation *string `xml:"situation" json:"situation,omitempty"`
SlayerCaste *string `xml:"slayer_caste" json:"slayerCaste,omitempty"`
SlayerHfid *int `xml:"slayer_hfid" json:"slayerHfid,omitempty" legend:"hf"`
SlayerItemId *int `xml:"slayer_item_id" json:"slayerItemId,omitempty"`
SlayerRace *string `xml:"slayer_race" json:"slayerRace,omitempty"`
SlayerShooterItemId *int `xml:"slayer_shooter_item_id" json:"slayerShooterItemId,omitempty"`
SnatcherHfid *int `xml:"snatcher_hfid" json:"snatcherHfid,omitempty" legend:"hf"`
SourceEntityId *int `xml:"source_entity_id" json:"sourceEntityId,omitempty" legend:"entity"`
SourceSiteId *int `xml:"source_site_id" json:"sourceSiteId,omitempty" legend:"site"`
SourceStructureId *int `xml:"source_structure_id" json:"sourceStructureId,omitempty" legend:"structure"`
SpeakerHfid *int `xml:"speaker_hfid" json:"speakerHfid,omitempty" legend:"hf"`
State *string `xml:"state" json:"state,omitempty"`
StructureId *int `xml:"structure_id" json:"structureId,omitempty" legend:"structure"`
StudentHfid *int `xml:"student_hfid" json:"studentHfid,omitempty" legend:"hf"`
SubregionId *int `xml:"subregion_id" json:"subregionId,omitempty"`
Subtype *string `xml:"subtype" json:"subtype,omitempty"`
Successful *string `xml:"successful" json:"successful,omitempty"`
SurveiledContact *string `xml:"surveiled_contact" json:"surveiledContact,omitempty"`
SurveiledConvicted *string `xml:"surveiled_convicted" json:"surveiledConvicted,omitempty"`
TargetEnid *int `xml:"target_enid" json:"targetEnid,omitempty" legend:"entity"`
TargetHfid *int `xml:"target_hfid" json:"targetHfid,omitempty" legend:"hf"`
TargetIdentity *int `xml:"target_identity" json:"targetIdentity,omitempty"`
TargetSeenAs *string `xml:"target_seen_as" json:"targetSeenAs,omitempty"`
TeacherHfid *int `xml:"teacher_hfid" json:"teacherHfid,omitempty" legend:"hf"`
TopFacet *string `xml:"top_facet" json:"topFacet,omitempty"`
TopFacetModifier *int `xml:"top_facet_modifier" json:"topFacetModifier,omitempty"`
TopFacetRating *int `xml:"top_facet_rating" json:"topFacetRating,omitempty"`
TopRelationshipFactor *string `xml:"top_relationship_factor" json:"topRelationshipFactor,omitempty"`
TopRelationshipModifier *int `xml:"top_relationship_modifier" json:"topRelationshipModifier,omitempty"`
TopRelationshipRating *int `xml:"top_relationship_rating" json:"topRelationshipRating,omitempty"`
TopValue *string `xml:"top_value" json:"topValue,omitempty"`
TopValueModifier *int `xml:"top_value_modifier" json:"topValueModifier,omitempty"`
TopValueRating *int `xml:"top_value_rating" json:"topValueRating,omitempty"`
Topic *string `xml:"topic" json:"topic,omitempty"`
TraderEntityId *int `xml:"trader_entity_id" json:"traderEntityId,omitempty" legend:"entity"`
TraderHfid *int `xml:"trader_hfid" json:"traderHfid,omitempty" legend:"hf"`
TricksterHfid *int `xml:"trickster_hfid" json:"tricksterHfid,omitempty" legend:"hf"`
UnitId *int `xml:"unit_id" json:"unitId,omitempty"`
UnitType *string `xml:"unit_type" json:"unitType,omitempty"`
WantedAndRecognized *string `xml:"wanted_and_recognized" json:"wantedAndRecognized,omitempty"`
WcId *int `xml:"wc_id" json:"wcId,omitempty" legend:"wc"`
Wcid *int `xml:"wcid" json:"wcid,omitempty" legend:"wc"`
WinnerHfid *int `xml:"winner_hfid" json:"winnerHfid,omitempty" legend:"hf"`
WoundeeHfid *int `xml:"woundee_hfid" json:"woundeeHfid,omitempty" legend:"hf"`
WounderHfid *int `xml:"wounder_hfid" json:"wounderHfid,omitempty" legend:"hf"`
WrongfulConviction *string `xml:"wrongful_conviction" json:"wrongfulConviction,omitempty"`
OtherElements
}
func (r *HistoricalEvent) Id() int { return r.Id_ }
func (r *HistoricalEvent) Name() string { return r.Type() }
type EventObject struct {
Events []*HistoricalEvent `json:"events"`
}
func (r *EventObject) GetEvents() []*HistoricalEvent { return r.Events }
func (r *EventObject) SetEvents(events []*HistoricalEvent) { r.Events = events }
type HasEvents interface {
GetEvents() []*HistoricalEvent
SetEvents([]*HistoricalEvent)
}

83
model/model.go Normal file
View File

@ -0,0 +1,83 @@
package model
import "encoding/xml"
type Region struct {
XMLName xml.Name `xml:"region" json:"-"`
NamedObject
Type string `xml:"type" json:"type"`
}
type UndergroundRegion struct {
XMLName xml.Name `xml:"underground_region" json:"-"`
NamedObject
Type string `xml:"type" json:"type"`
}
type Landmass struct {
XMLName xml.Name `xml:"landmass" json:"-"`
NamedObject
}
type Site struct {
XMLName xml.Name `xml:"site" json:"-"`
NamedObject
Type string `xml:"type" json:"type"`
Coords string `xml:"coords" json:"coords"`
Rectangle string `xml:"rectangle" json:"rectangle"`
Structures []Structure `xml:"structures>structure" json:"structures"`
EventObject
}
// func (obj Site) id() int { return obj.Id }
// func (obj Site) name() string { return obj.Name }
type Structure struct {
XMLName xml.Name `xml:"structure" json:"-"`
LocalId int `xml:"local_id" json:"localId"`
Name string `xml:"name" json:"name"`
Type string `xml:"type" json:"type"`
}
type WorldConstruction struct {
XMLName xml.Name `xml:"world_construction" json:"-"`
NamedObject
}
type Artifact struct {
XMLName xml.Name `xml:"artifact" json:"-"`
NamedObject
SiteId int `xml:"site_id" json:"siteId"`
EventObject
}
type HistoricalFigure struct {
XMLName xml.Name `xml:"historical_figure" json:"-"`
NamedObject
Race string `xml:"race" json:"race"`
Caste string `xml:"caste" json:"caste"`
OtherElements
EventObject
}
func (r *HistoricalFigure) Type() string { return "hf" }
type HistoricalEventCollection struct {
XMLName xml.Name `xml:"historical_event_collection" json:"-"`
NamedObject
StartYear int `xml:"year"`
StartSeconds int `xml:"seconds72"`
EndYear int `xml:"end_year"`
EndSeconds int `xml:"end_seconds72"`
Type string `xml:"type" json:"type"`
EventIds []int `xml:"event" json:"eventIds"`
}
type Entity struct {
XMLName xml.Name `xml:"entity" json:"-"`
NamedObject
EventObject
}

50
model/util.go Normal file
View File

@ -0,0 +1,50 @@
package model
import "encoding/xml"
type NamedObject struct {
Id_ int `xml:"id" json:"id"`
Name_ string `xml:"name" json:"name"`
}
func (r *NamedObject) Id() int { return r.Id_ }
func (r *NamedObject) Name() string { return r.Name_ }
type Named interface {
Id() int
Name() string
}
type Identifiable interface {
Id() int
}
type TypedObject struct {
Type_ string `xml:"type" json:"type"`
}
func (r *TypedObject) Type() string { return r.Type_ }
type Typed interface {
Type() string
}
type OtherElements struct {
Others_ []Element `xml:",any" json:"-"`
}
func (r *OtherElements) Others() []Element { return r.Others_ }
type Others interface {
Others() []Element
}
type TypedOthers interface {
Type() string
Others() []Element
}
type Element struct {
XMLName xml.Name
Value string `xml:",innerxml"`
}

146
model/world.go Normal file
View File

@ -0,0 +1,146 @@
package model
import (
"encoding/xml"
"fmt"
"io/ioutil"
"os"
"reflect"
)
type World struct {
XMLName xml.Name `xml:"df_world"`
Name string `xml:"name"`
AltName string `xml:"altname"`
Regions []*Region `xml:"regions>region"`
UndergroundRegions []*UndergroundRegion `xml:"underground_regions>underground_region"`
Landmasses []*Landmass `xml:"landmasses>landmass"`
Sites []*Site `xml:"sites>site"`
WorldConstructions []*WorldConstruction `xml:"world_constructions>world_construction"`
Artifacts []*Artifact `xml:"artifacts>artifact"`
HistoricalFigures []*HistoricalFigure `xml:"historical_figures>historical_figure"`
HistoricalEvents []*HistoricalEvent `xml:"historical_events>historical_event"`
HistoricalEventCollections []*HistoricalEventCollection `xml:"historical_event_collections>historical_event_collection"`
Entities []*Entity `xml:"entities>entity"`
RegionMap map[int]*Region
UndergroundRegionMap map[int]*UndergroundRegion
LandmassMap map[int]*Landmass
SiteMap map[int]*Site
WorldConstructionMap map[int]*WorldConstruction
ArtifactMap map[int]*Artifact
HistoricalFigureMap map[int]*HistoricalFigure
HistoricalEventMap map[int]*HistoricalEvent
HistoricalEventCollectionMap map[int]*HistoricalEventCollection
EntityMap map[int]*Entity
}
func (w *World) Load(file string) {
xmlFile, err := os.Open(file)
if err != nil {
fmt.Println(err)
}
fmt.Println("Successfully Opened users.xml")
defer xmlFile.Close()
byteValue, _ := ioutil.ReadAll(xmlFile)
fmt.Println(len(byteValue))
err = xml.Unmarshal(byteValue, w)
if err != nil {
fmt.Println(err)
}
fmt.Println("World loaded")
}
func (w *World) Process() {
w.RegionMap = make(map[int]*Region)
mapObjects(&w.Regions, &w.RegionMap)
w.UndergroundRegionMap = make(map[int]*UndergroundRegion)
mapObjects(&w.UndergroundRegions, &w.UndergroundRegionMap)
w.LandmassMap = make(map[int]*Landmass)
mapObjects(&w.Landmasses, &w.LandmassMap)
w.SiteMap = make(map[int]*Site)
mapObjects(&w.Sites, &w.SiteMap)
w.WorldConstructionMap = make(map[int]*WorldConstruction)
mapObjects(&w.WorldConstructions, &w.WorldConstructionMap)
w.ArtifactMap = make(map[int]*Artifact)
mapObjects(&w.Artifacts, &w.ArtifactMap)
w.HistoricalFigureMap = make(map[int]*HistoricalFigure)
mapObjects(&w.HistoricalFigures, &w.HistoricalFigureMap)
w.HistoricalEventMap = make(map[int]*HistoricalEvent)
mapObjects(&w.HistoricalEvents, &w.HistoricalEventMap)
w.HistoricalEventCollectionMap = make(map[int]*HistoricalEventCollection)
mapObjects(&w.HistoricalEventCollections, &w.HistoricalEventCollectionMap)
w.EntityMap = make(map[int]*Entity)
mapObjects(&w.Entities, &w.EntityMap)
w.processEvents()
}
func (w *World) processEvents() {
legendFields := make(map[string][]int)
t := reflect.TypeOf(HistoricalEvent{})
for i := 0; i < t.NumField(); i++ {
f := t.Field(i)
l, ok := f.Tag.Lookup("legend")
if ok {
legendFields[l] = append(legendFields[l], i)
}
}
for eventIndex := 0; eventIndex < len(w.HistoricalEvents); eventIndex++ {
e := w.HistoricalEvents[eventIndex]
v := reflect.ValueOf(*e)
processEvent(e, &v, legendFields["entity"], &w.EntityMap)
processEvent(e, &v, legendFields["site"], &w.SiteMap)
processEvent(e, &v, legendFields["hf"], &w.HistoricalFigureMap)
processEvent(e, &v, legendFields["artifact"], &w.ArtifactMap)
// processEvent(e, &v, legendFields["wc"], &w.WorldConstructionMap)
// processEvent(e, &v, legendFields["structure"], &w.St)
}
}
func processEvent[T HasEvents](event *HistoricalEvent, v *reflect.Value, fields []int, objectMap *map[int]T) {
for _, i := range fields {
val := v.Field(i)
if !val.IsZero() {
switch val.Elem().Kind() {
case reflect.Slice:
ids := val.Interface().(*[]int)
for _, id := range *ids {
x, ok := (*objectMap)[id]
if ok {
x.SetEvents(append(x.GetEvents(), event))
}
}
case reflect.Int:
id := int(val.Elem().Int())
x, ok := (*objectMap)[id]
if ok {
x.SetEvents(append(x.GetEvents(), event))
}
default:
fmt.Println("unknown", val.Elem().Kind())
}
}
}
}
func mapObjects[T Identifiable](objects *[]T, objectMap *map[int]T) {
for i, obj := range *objects {
(*objectMap)[obj.Id()] = (*objects)[i]
}
}

43
server/resource.go Normal file
View File

@ -0,0 +1,43 @@
package server
import (
"encoding/json"
"fmt"
"legendsbrowser/model"
"net/http"
"strconv"
"github.com/gorilla/mux"
)
type Info struct {
Id int `json:"id"`
Name string `json:"name"`
}
func RegisterResource[T model.Named](router *mux.Router, resourceName string, resources map[int]T) {
list := func(w http.ResponseWriter, r *http.Request) {
values := make([]Info, 0, len(resources))
for _, v := range resources {
values = append(values, Info{Id: v.Id(), Name: v.Name()})
}
json.NewEncoder(w).Encode(values)
}
get := func(w http.ResponseWriter, r *http.Request) {
id, err := strconv.Atoi(mux.Vars(r)["id"])
if err != nil {
fmt.Println(err)
}
for _, item := range resources {
if item.Id() == id {
json.NewEncoder(w).Encode(item)
}
}
}
router.HandleFunc(fmt.Sprintf("/api/%s", resourceName), list).Methods("GET")
router.HandleFunc(fmt.Sprintf("/api/%s/{id}", resourceName), get).Methods("GET")
}

33
server/servestatic.go Normal file
View File

@ -0,0 +1,33 @@
package server
import (
"net/http"
"os"
"path/filepath"
)
type SpaHandler struct {
StaticPath string
IndexPath string
}
func (h SpaHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
path, err := filepath.Abs(r.URL.Path)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
path = filepath.Join(h.StaticPath, path)
_, err = os.Stat(path)
if os.IsNotExist(err) {
http.ServeFile(w, r, filepath.Join(h.StaticPath, h.IndexPath))
return
} else if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
http.FileServer(http.Dir(h.StaticPath)).ServeHTTP(w, r)
}

28
util/util.go Normal file
View File

@ -0,0 +1,28 @@
package util
import "strings"
func Keys[K comparable, V any](input map[K]V) []K {
keys := make([]K, 0, len(input))
for k := range input {
keys = append(keys, k)
}
return keys
}
func Values[K comparable, V any](input map[K]V) []V {
values := make([]V, 0, len(input))
for _, v := range input {
values = append(values, v)
}
return values
}
func ContainsAny(s string, substrings ...string) bool {
for _, substring := range substrings {
if strings.Contains(s, substring) {
return true
}
}
return false
}