Merge pull request #13 from Aurorastation/dev
Deprecates the old WPF app, replacing it with a web app hosted at c.ily.rs.
This commit is contained in:
commit
c9b7ab30ca
110 changed files with 7514 additions and 3187 deletions
63
.gitattributes
vendored
63
.gitattributes
vendored
|
|
@ -1,63 +0,0 @@
|
|||
###############################################################################
|
||||
# Set default behavior to automatically normalize line endings.
|
||||
###############################################################################
|
||||
* text=auto
|
||||
|
||||
###############################################################################
|
||||
# Set default behavior for command prompt diff.
|
||||
#
|
||||
# This is need for earlier builds of msysgit that does not have it on by
|
||||
# default for csharp files.
|
||||
# Note: This is only used by command line
|
||||
###############################################################################
|
||||
#*.cs diff=csharp
|
||||
|
||||
###############################################################################
|
||||
# Set the merge driver for project and solution files
|
||||
#
|
||||
# Merging from the command prompt will add diff markers to the files if there
|
||||
# are conflicts (Merging from VS is not affected by the settings below, in VS
|
||||
# the diff markers are never inserted). Diff markers may cause the following
|
||||
# file extensions to fail to load in VS. An alternative would be to treat
|
||||
# these files as binary and thus will always conflict and require user
|
||||
# intervention with every merge. To do so, just uncomment the entries below
|
||||
###############################################################################
|
||||
#*.sln merge=binary
|
||||
#*.csproj merge=binary
|
||||
#*.vbproj merge=binary
|
||||
#*.vcxproj merge=binary
|
||||
#*.vcproj merge=binary
|
||||
#*.dbproj merge=binary
|
||||
#*.fsproj merge=binary
|
||||
#*.lsproj merge=binary
|
||||
#*.wixproj merge=binary
|
||||
#*.modelproj merge=binary
|
||||
#*.sqlproj merge=binary
|
||||
#*.wwaproj merge=binary
|
||||
|
||||
###############################################################################
|
||||
# behavior for image files
|
||||
#
|
||||
# image files are treated as binary by default.
|
||||
###############################################################################
|
||||
#*.jpg binary
|
||||
#*.png binary
|
||||
#*.gif binary
|
||||
|
||||
###############################################################################
|
||||
# diff behavior for common document formats
|
||||
#
|
||||
# Convert binary document formats to text before diffing them. This feature
|
||||
# is only available from the command line. Turn it on by uncommenting the
|
||||
# entries below.
|
||||
###############################################################################
|
||||
#*.doc diff=astextplain
|
||||
#*.DOC diff=astextplain
|
||||
#*.docx diff=astextplain
|
||||
#*.DOCX diff=astextplain
|
||||
#*.dot diff=astextplain
|
||||
#*.DOT diff=astextplain
|
||||
#*.pdf diff=astextplain
|
||||
#*.PDF diff=astextplain
|
||||
#*.rtf diff=astextplain
|
||||
#*.RTF diff=astextplain
|
||||
68
.github/workflows/dotnet-build-test.yml
vendored
68
.github/workflows/dotnet-build-test.yml
vendored
|
|
@ -1,68 +0,0 @@
|
|||
name: ci build and test
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [ "master" ]
|
||||
tags:
|
||||
- 'v*'
|
||||
pull_request:
|
||||
branches: [ "master" ]
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: windows-latest
|
||||
env:
|
||||
Solution_Name: CharacterRecordsGenerator.sln
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v3
|
||||
- name: Add MSBuild to path
|
||||
uses: microsoft/setup-msbuild@v1.1
|
||||
- name: Setup NuGet
|
||||
uses: NuGet/setup-nuget@v1
|
||||
- name: Restore nuget
|
||||
run: nuget restore $env:Solution_Name
|
||||
- name: Build
|
||||
run: msbuild $env:Solution_Name /t:Rebuild -property:Configuration=Release
|
||||
- name: Unit tests
|
||||
run: dotnet test --no-build --verbosity normal
|
||||
- name: Archive
|
||||
uses: actions/upload-artifact@v3
|
||||
with:
|
||||
name: character-records-generator
|
||||
path: |
|
||||
CharacterRecordsGenerator/bin/Release/*.exe
|
||||
CharacterRecordsGenerator/bin/Release/*.dll
|
||||
deploy:
|
||||
name: Create release
|
||||
runs-on: ubuntu-latest
|
||||
if: startsWith(github.ref, 'refs/tags/v')
|
||||
needs: build
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v3
|
||||
- name: Create release
|
||||
id: create_release
|
||||
uses: actions/create-release@v1
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
with:
|
||||
tag_name: ${{ github.ref }}
|
||||
release_name: ${{ github.ref }}
|
||||
draft: false
|
||||
prerelease: false
|
||||
- uses: actions/download-artifact@v3
|
||||
with:
|
||||
name: character-records-generator
|
||||
path: character-records-generator
|
||||
- run: ls
|
||||
- run: zip -r character-records-generator.zip character-records-generator
|
||||
- name: Upload character-records-generator
|
||||
uses: actions/upload-release-asset@v1
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
with:
|
||||
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
||||
asset_path: ./character-records-generator.zip
|
||||
asset_name: character-records-generator.zip
|
||||
asset_content_type: application/zip
|
||||
276
.gitignore
vendored
276
.gitignore
vendored
|
|
@ -1,261 +1,23 @@
|
|||
## Ignore Visual Studio temporary files, build results, and
|
||||
## files generated by popular Visual Studio add-ons.
|
||||
node_modules
|
||||
|
||||
# User-specific files
|
||||
*.suo
|
||||
*.user
|
||||
*.userosscache
|
||||
*.sln.docstates
|
||||
# Output
|
||||
.output
|
||||
.vercel
|
||||
.netlify
|
||||
.wrangler
|
||||
/.svelte-kit
|
||||
/build
|
||||
|
||||
# User-specific files (MonoDevelop/Xamarin Studio)
|
||||
*.userprefs
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Build results
|
||||
[Dd]ebug/
|
||||
[Dd]ebugPublic/
|
||||
[Rr]elease/
|
||||
[Rr]eleases/
|
||||
x64/
|
||||
x86/
|
||||
bld/
|
||||
[Bb]in/
|
||||
[Oo]bj/
|
||||
[Ll]og/
|
||||
# Env
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
!.env.test
|
||||
|
||||
# Visual Studio 2015 cache/options directory
|
||||
.vs/
|
||||
# Uncomment if you have tasks that create the project's static files in wwwroot
|
||||
#wwwroot/
|
||||
|
||||
# MSTest test Results
|
||||
[Tt]est[Rr]esult*/
|
||||
[Bb]uild[Ll]og.*
|
||||
|
||||
# NUNIT
|
||||
*.VisualState.xml
|
||||
TestResult.xml
|
||||
|
||||
# Build Results of an ATL Project
|
||||
[Dd]ebugPS/
|
||||
[Rr]eleasePS/
|
||||
dlldata.c
|
||||
|
||||
# DNX
|
||||
project.lock.json
|
||||
project.fragment.lock.json
|
||||
artifacts/
|
||||
|
||||
*_i.c
|
||||
*_p.c
|
||||
*_i.h
|
||||
*.ilk
|
||||
*.meta
|
||||
*.obj
|
||||
*.pch
|
||||
*.pdb
|
||||
*.pgc
|
||||
*.pgd
|
||||
*.rsp
|
||||
*.sbr
|
||||
*.tlb
|
||||
*.tli
|
||||
*.tlh
|
||||
*.tmp
|
||||
*.tmp_proj
|
||||
*.log
|
||||
*.vspscc
|
||||
*.vssscc
|
||||
.builds
|
||||
*.pidb
|
||||
*.svclog
|
||||
*.scc
|
||||
|
||||
# Chutzpah Test files
|
||||
_Chutzpah*
|
||||
|
||||
# Visual C++ cache files
|
||||
ipch/
|
||||
*.aps
|
||||
*.ncb
|
||||
*.opendb
|
||||
*.opensdf
|
||||
*.sdf
|
||||
*.cachefile
|
||||
*.VC.db
|
||||
*.VC.VC.opendb
|
||||
|
||||
# Visual Studio profiler
|
||||
*.psess
|
||||
*.vsp
|
||||
*.vspx
|
||||
*.sap
|
||||
|
||||
# TFS 2012 Local Workspace
|
||||
$tf/
|
||||
|
||||
# Guidance Automation Toolkit
|
||||
*.gpState
|
||||
|
||||
# ReSharper is a .NET coding add-in
|
||||
_ReSharper*/
|
||||
*.[Rr]e[Ss]harper
|
||||
*.DotSettings.user
|
||||
|
||||
# JustCode is a .NET coding add-in
|
||||
.JustCode
|
||||
|
||||
# TeamCity is a build add-in
|
||||
_TeamCity*
|
||||
|
||||
# DotCover is a Code Coverage Tool
|
||||
*.dotCover
|
||||
|
||||
# NCrunch
|
||||
_NCrunch_*
|
||||
.*crunch*.local.xml
|
||||
nCrunchTemp_*
|
||||
|
||||
# MightyMoose
|
||||
*.mm.*
|
||||
AutoTest.Net/
|
||||
|
||||
# Web workbench (sass)
|
||||
.sass-cache/
|
||||
|
||||
# Installshield output folder
|
||||
[Ee]xpress/
|
||||
|
||||
# DocProject is a documentation generator add-in
|
||||
DocProject/buildhelp/
|
||||
DocProject/Help/*.HxT
|
||||
DocProject/Help/*.HxC
|
||||
DocProject/Help/*.hhc
|
||||
DocProject/Help/*.hhk
|
||||
DocProject/Help/*.hhp
|
||||
DocProject/Help/Html2
|
||||
DocProject/Help/html
|
||||
|
||||
# Click-Once directory
|
||||
publish/
|
||||
|
||||
# Publish Web Output
|
||||
*.[Pp]ublish.xml
|
||||
*.azurePubxml
|
||||
# TODO: Comment the next line if you want to checkin your web deploy settings
|
||||
# but database connection strings (with potential passwords) will be unencrypted
|
||||
#*.pubxml
|
||||
*.publishproj
|
||||
|
||||
# Microsoft Azure Web App publish settings. Comment the next line if you want to
|
||||
# checkin your Azure Web App publish settings, but sensitive information contained
|
||||
# in these scripts will be unencrypted
|
||||
PublishScripts/
|
||||
|
||||
# NuGet Packages
|
||||
*.nupkg
|
||||
# The packages folder can be ignored because of Package Restore
|
||||
**/packages/*
|
||||
# except build/, which is used as an MSBuild target.
|
||||
!**/packages/build/
|
||||
# Uncomment if necessary however generally it will be regenerated when needed
|
||||
#!**/packages/repositories.config
|
||||
# NuGet v3's project.json files produces more ignoreable files
|
||||
*.nuget.props
|
||||
*.nuget.targets
|
||||
|
||||
# Microsoft Azure Build Output
|
||||
csx/
|
||||
*.build.csdef
|
||||
|
||||
# Microsoft Azure Emulator
|
||||
ecf/
|
||||
rcf/
|
||||
|
||||
# Windows Store app package directories and files
|
||||
AppPackages/
|
||||
BundleArtifacts/
|
||||
Package.StoreAssociation.xml
|
||||
_pkginfo.txt
|
||||
|
||||
# Visual Studio cache files
|
||||
# files ending in .cache can be ignored
|
||||
*.[Cc]ache
|
||||
# but keep track of directories ending in .cache
|
||||
!*.[Cc]ache/
|
||||
|
||||
# Others
|
||||
ClientBin/
|
||||
~$*
|
||||
*~
|
||||
*.dbmdl
|
||||
*.dbproj.schemaview
|
||||
*.jfm
|
||||
*.pfx
|
||||
*.publishsettings
|
||||
node_modules/
|
||||
orleans.codegen.cs
|
||||
|
||||
# Since there are multiple workflows, uncomment next line to ignore bower_components
|
||||
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
|
||||
#bower_components/
|
||||
|
||||
# RIA/Silverlight projects
|
||||
Generated_Code/
|
||||
|
||||
# Backup & report files from converting an old project file
|
||||
# to a newer Visual Studio version. Backup files are not needed,
|
||||
# because we have git ;-)
|
||||
_UpgradeReport_Files/
|
||||
Backup*/
|
||||
UpgradeLog*.XML
|
||||
UpgradeLog*.htm
|
||||
|
||||
# SQL Server files
|
||||
*.mdf
|
||||
*.ldf
|
||||
|
||||
# Business Intelligence projects
|
||||
*.rdl.data
|
||||
*.bim.layout
|
||||
*.bim_*.settings
|
||||
|
||||
# Microsoft Fakes
|
||||
FakesAssemblies/
|
||||
|
||||
# GhostDoc plugin setting file
|
||||
*.GhostDoc.xml
|
||||
|
||||
# Node.js Tools for Visual Studio
|
||||
.ntvs_analysis.dat
|
||||
|
||||
# Visual Studio 6 build log
|
||||
*.plg
|
||||
|
||||
# Visual Studio 6 workspace options file
|
||||
*.opt
|
||||
|
||||
# Visual Studio LightSwitch build output
|
||||
**/*.HTMLClient/GeneratedArtifacts
|
||||
**/*.DesktopClient/GeneratedArtifacts
|
||||
**/*.DesktopClient/ModelManifest.xml
|
||||
**/*.Server/GeneratedArtifacts
|
||||
**/*.Server/ModelManifest.xml
|
||||
_Pvt_Extensions
|
||||
|
||||
# Paket dependency manager
|
||||
.paket/paket.exe
|
||||
paket-files/
|
||||
|
||||
# FAKE - F# Make
|
||||
.fake/
|
||||
|
||||
# JetBrains Rider
|
||||
.idea/
|
||||
*.sln.iml
|
||||
|
||||
# CodeRush
|
||||
.cr/
|
||||
|
||||
# Python Tools for Visual Studio (PTVS)
|
||||
__pycache__/
|
||||
*.pyc
|
||||
# Vite
|
||||
vite.config.js.timestamp-*
|
||||
vite.config.ts.timestamp-*
|
||||
|
|
|
|||
1
.npmrc
Normal file
1
.npmrc
Normal file
|
|
@ -0,0 +1 @@
|
|||
engine-strict=true
|
||||
|
|
@ -1,74 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project ToolsVersion="15.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||
<Import Project="..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.props" Condition="Exists('..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.props')" />
|
||||
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
||||
<PropertyGroup>
|
||||
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
||||
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
|
||||
<ProjectGuid>{195A07A8-03F7-4A1F-932F-73F934EF324F}</ProjectGuid>
|
||||
<OutputType>Library</OutputType>
|
||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||
<RootNamespace>CharacterRecordsGeneratorTests</RootNamespace>
|
||||
<AssemblyName>CharacterRecordsGeneratorTests</AssemblyName>
|
||||
<TargetFrameworkVersion>v4.5.2</TargetFrameworkVersion>
|
||||
<FileAlignment>512</FileAlignment>
|
||||
<ProjectTypeGuids>{3AC096D0-A1C2-E12C-1390-A8335801FDAB};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
|
||||
<VisualStudioVersion Condition="'$(VisualStudioVersion)' == ''">15.0</VisualStudioVersion>
|
||||
<VSToolsPath Condition="'$(VSToolsPath)' == ''">$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)</VSToolsPath>
|
||||
<ReferencePath>$(ProgramFiles)\Common Files\microsoft shared\VSTT\$(VisualStudioVersion)\UITestExtensionPackages</ReferencePath>
|
||||
<IsCodedUITest>False</IsCodedUITest>
|
||||
<TestProjectType>UnitTest</TestProjectType>
|
||||
<NuGetPackageImportStamp>
|
||||
</NuGetPackageImportStamp>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
|
||||
<DebugSymbols>true</DebugSymbols>
|
||||
<DebugType>full</DebugType>
|
||||
<Optimize>false</Optimize>
|
||||
<OutputPath>bin\Debug\</OutputPath>
|
||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
|
||||
<DebugType>pdbonly</DebugType>
|
||||
<Optimize>true</Optimize>
|
||||
<OutputPath>bin\Release\</OutputPath>
|
||||
<DefineConstants>TRACE</DefineConstants>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\MSTest.TestFramework.2.1.2\lib\net45\Microsoft.VisualStudio.TestPlatform.TestFramework.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions, Version=14.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\MSTest.TestFramework.2.1.2\lib\net45\Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="System" />
|
||||
<Reference Include="System.Core" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="CharacterRecordsUtilitiesTests.cs" />
|
||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Include="packages.config" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\CharacterRecordsGenerator\CharacterRecordsGenerator.csproj">
|
||||
<Project>{2e1295c2-7bd9-454e-b13e-8a22448dd5f6}</Project>
|
||||
<Name>CharacterRecordsGenerator</Name>
|
||||
</ProjectReference>
|
||||
</ItemGroup>
|
||||
<Import Project="$(VSToolsPath)\TeamTest\Microsoft.TestTools.targets" Condition="Exists('$(VSToolsPath)\TeamTest\Microsoft.TestTools.targets')" />
|
||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
|
||||
<PropertyGroup>
|
||||
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
|
||||
</PropertyGroup>
|
||||
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.props'))" />
|
||||
<Error Condition="!Exists('..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.targets'))" />
|
||||
</Target>
|
||||
<Import Project="..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.targets" Condition="Exists('..\packages\MSTest.TestAdapter.2.1.2\build\net45\MSTest.TestAdapter.targets')" />
|
||||
</Project>
|
||||
|
|
@ -1,18 +0,0 @@
|
|||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using System;
|
||||
using CharacterRecordsGenerator;
|
||||
|
||||
namespace CharacterRecordsGeneratorTests
|
||||
{
|
||||
[TestClass]
|
||||
public class CharacterRecordsUtilitiesTests
|
||||
{
|
||||
[TestMethod]
|
||||
public void Utility_CmToFeet_WithValidNumber()
|
||||
{
|
||||
double cm = 150.0;
|
||||
string expected = "4'11\"";
|
||||
Assert.AreEqual(expected, Utility.CmToFeet(cm));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,20 +0,0 @@
|
|||
using System.Reflection;
|
||||
using System.Runtime.CompilerServices;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
[assembly: AssemblyTitle("CharacterRecordsGeneratorTests")]
|
||||
[assembly: AssemblyDescription("")]
|
||||
[assembly: AssemblyConfiguration("")]
|
||||
[assembly: AssemblyCompany("")]
|
||||
[assembly: AssemblyProduct("CharacterRecordsGeneratorTests")]
|
||||
[assembly: AssemblyCopyright("Copyright © 2022")]
|
||||
[assembly: AssemblyTrademark("")]
|
||||
[assembly: AssemblyCulture("")]
|
||||
|
||||
[assembly: ComVisible(false)]
|
||||
|
||||
[assembly: Guid("195a07a8-03f7-4a1f-932f-73f934ef324f")]
|
||||
|
||||
// [assembly: AssemblyVersion("1.0.*")]
|
||||
[assembly: AssemblyVersion("1.0.0.0")]
|
||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
||||
|
|
@ -1,5 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<packages>
|
||||
<package id="MSTest.TestAdapter" version="2.1.2" targetFramework="net452" />
|
||||
<package id="MSTest.TestFramework" version="2.1.2" targetFramework="net452" />
|
||||
</packages>
|
||||
|
|
@ -1,54 +0,0 @@
|
|||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 16
|
||||
VisualStudioVersion = 16.0.31702.278
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CharacterRecordsGenerator", "CharacterRecordsGenerator\CharacterRecordsGenerator.csproj", "{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CharacterRecordsGenerator.Tests", "CharacterRecordsGenerator.Tests\CharacterRecordsGenerator.Tests.csproj", "{195A07A8-03F7-4A1F-932F-73F934EF324F}"
|
||||
ProjectSection(ProjectDependencies) = postProject
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6} = {2E1295C2-7BD9-454E-B13E-8A22448DD5F6}
|
||||
EndProjectSection
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Debug|x64 = Debug|x64
|
||||
Debug|x86 = Debug|x86
|
||||
Release|Any CPU = Release|Any CPU
|
||||
Release|x64 = Release|x64
|
||||
Release|x86 = Release|x86
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Debug|x64.ActiveCfg = Debug|x64
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Debug|x64.Build.0 = Debug|x64
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Debug|x86.ActiveCfg = Debug|x86
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Debug|x86.Build.0 = Debug|x86
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Release|x64.ActiveCfg = Release|x64
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Release|x64.Build.0 = Release|x64
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Release|x86.ActiveCfg = Release|x86
|
||||
{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}.Release|x86.Build.0 = Release|x86
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Release|x64.Build.0 = Release|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{195A07A8-03F7-4A1F-932F-73F934EF324F}.Release|x86.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {3F94E0AF-3102-424C-A6C2-B930212E7CD8}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
|
|
@ -1,6 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<configuration>
|
||||
<startup>
|
||||
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5.2" />
|
||||
</startup>
|
||||
</configuration>
|
||||
|
|
@ -1,20 +0,0 @@
|
|||
<Application x:Class="CharacterRecordsGenerator.App"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:local="clr-namespace:CharacterRecordsGenerator"
|
||||
StartupUri="RecordEditor.xaml">
|
||||
<Application.Resources>
|
||||
<ResourceDictionary>
|
||||
<ResourceDictionary.MergedDictionaries>
|
||||
<!-- MahApps.Metro resource dictionaries. Make sure that all file names are Case Sensitive! -->
|
||||
<ResourceDictionary Source="pack://application:,,,/MahApps.Metro;component/Styles/Controls.xaml" />
|
||||
<ResourceDictionary Source="pack://application:,,,/MahApps.Metro;component/Styles/Fonts.xaml" />
|
||||
<ResourceDictionary Source="pack://application:,,,/MahApps.Metro;component/Styles/Colors.xaml" />
|
||||
<!-- Accent and AppTheme setting -->
|
||||
<ResourceDictionary Source="pack://application:,,,/MahApps.Metro;component/Styles/Accents/Blue.xaml" />
|
||||
<ResourceDictionary Source="pack://application:,,,/MahApps.Metro;component/Styles/Accents/BaseLight.xaml" />
|
||||
<ResourceDictionary Source="pack://application:,,,/MahApps.Metro;component/Styles/Controls.AnimatedTabControl.xaml" />
|
||||
</ResourceDictionary.MergedDictionaries>
|
||||
</ResourceDictionary>
|
||||
</Application.Resources>
|
||||
</Application>
|
||||
|
|
@ -1,17 +0,0 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Configuration;
|
||||
using System.Data;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using System.Windows;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
/// <summary>
|
||||
/// Interaction logic for App.xaml
|
||||
/// </summary>
|
||||
public partial class App : Application
|
||||
{
|
||||
}
|
||||
}
|
||||
Binary file not shown.
|
Before Width: | Height: | Size: 66 KiB |
|
|
@ -1,220 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project ToolsVersion="14.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
||||
<PropertyGroup>
|
||||
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
||||
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
|
||||
<ProjectGuid>{2E1295C2-7BD9-454E-B13E-8A22448DD5F6}</ProjectGuid>
|
||||
<OutputType>WinExe</OutputType>
|
||||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||
<RootNamespace>CharacterRecordsGenerator</RootNamespace>
|
||||
<AssemblyName>Character Records Generator</AssemblyName>
|
||||
<TargetFrameworkVersion>v4.5.2</TargetFrameworkVersion>
|
||||
<FileAlignment>512</FileAlignment>
|
||||
<ProjectTypeGuids>{60dc8134-eba5-43b8-bcc9-bb4bc16c2548};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
|
||||
<IsWebBootstrapper>false</IsWebBootstrapper>
|
||||
<PublishUrl>publish\</PublishUrl>
|
||||
<Install>true</Install>
|
||||
<InstallFrom>Disk</InstallFrom>
|
||||
<UpdateEnabled>false</UpdateEnabled>
|
||||
<UpdateMode>Foreground</UpdateMode>
|
||||
<UpdateInterval>7</UpdateInterval>
|
||||
<UpdateIntervalUnits>Days</UpdateIntervalUnits>
|
||||
<UpdatePeriodically>false</UpdatePeriodically>
|
||||
<UpdateRequired>false</UpdateRequired>
|
||||
<MapFileExtensions>true</MapFileExtensions>
|
||||
<ApplicationRevision>0</ApplicationRevision>
|
||||
<ApplicationVersion>2.0.0.%2a</ApplicationVersion>
|
||||
<UseApplicationTrust>false</UseApplicationTrust>
|
||||
<BootstrapperEnabled>true</BootstrapperEnabled>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
|
||||
<PlatformTarget>AnyCPU</PlatformTarget>
|
||||
<DebugSymbols>true</DebugSymbols>
|
||||
<DebugType>full</DebugType>
|
||||
<Optimize>false</Optimize>
|
||||
<OutputPath>bin\Debug\</OutputPath>
|
||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
<Prefer32Bit>true</Prefer32Bit>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
|
||||
<PlatformTarget>AnyCPU</PlatformTarget>
|
||||
<DebugType>pdbonly</DebugType>
|
||||
<Optimize>true</Optimize>
|
||||
<OutputPath>bin\Release\</OutputPath>
|
||||
<DefineConstants>TRACE</DefineConstants>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<WarningLevel>4</WarningLevel>
|
||||
<Prefer32Bit>true</Prefer32Bit>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x64'">
|
||||
<DebugSymbols>true</DebugSymbols>
|
||||
<OutputPath>bin\x64\Debug\</OutputPath>
|
||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||
<DebugType>full</DebugType>
|
||||
<PlatformTarget>x64</PlatformTarget>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x64'">
|
||||
<OutputPath>bin\x64\Release\</OutputPath>
|
||||
<DefineConstants>TRACE</DefineConstants>
|
||||
<Optimize>true</Optimize>
|
||||
<DebugType>pdbonly</DebugType>
|
||||
<PlatformTarget>x64</PlatformTarget>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
|
||||
<Prefer32Bit>true</Prefer32Bit>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup>
|
||||
<ApplicationIcon>CRG.ico</ApplicationIcon>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||
<DebugSymbols>true</DebugSymbols>
|
||||
<OutputPath>bin\x86\Debug\</OutputPath>
|
||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||
<DebugType>full</DebugType>
|
||||
<PlatformTarget>x86</PlatformTarget>
|
||||
<LangVersion>7.3</LangVersion>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
|
||||
<Prefer32Bit>true</Prefer32Bit>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x86'">
|
||||
<OutputPath>bin\x86\Release\</OutputPath>
|
||||
<DefineConstants>TRACE</DefineConstants>
|
||||
<Optimize>true</Optimize>
|
||||
<DebugType>pdbonly</DebugType>
|
||||
<PlatformTarget>x86</PlatformTarget>
|
||||
<LangVersion>7.3</LangVersion>
|
||||
<ErrorReport>prompt</ErrorReport>
|
||||
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
|
||||
<Prefer32Bit>true</Prefer32Bit>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<Reference Include="ControlzEx, Version=3.0.2.4, Culture=neutral, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\ControlzEx.3.0.2.4\lib\net45\ControlzEx.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="Humanizer, Version=2.5.0.0, Culture=neutral, PublicKeyToken=979442b78dfc278e, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\Humanizer.Core.2.5.16\lib\netstandard1.0\Humanizer.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="MahApps.Metro, Version=1.6.5.1, Culture=neutral, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\MahApps.Metro.1.6.5\lib\net45\MahApps.Metro.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="protobuf-net, Version=2.4.0.0, Culture=neutral, PublicKeyToken=257b51d87d2e4d67, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\protobuf-net.2.4.0\lib\net40\protobuf-net.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="System" />
|
||||
<Reference Include="System.ComponentModel.Composition" />
|
||||
<Reference Include="System.ComponentModel.DataAnnotations" />
|
||||
<Reference Include="System.Configuration" />
|
||||
<Reference Include="System.Data" />
|
||||
<Reference Include="System.IO.Compression" />
|
||||
<Reference Include="System.Numerics" />
|
||||
<Reference Include="System.Runtime.InteropServices.RuntimeInformation, Version=4.0.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\System.Runtime.InteropServices.RuntimeInformation.4.3.0\lib\net45\System.Runtime.InteropServices.RuntimeInformation.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="System.Runtime.Serialization" />
|
||||
<Reference Include="System.ServiceModel" />
|
||||
<Reference Include="System.Windows.Interactivity, Version=4.5.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
|
||||
<HintPath>..\packages\ControlzEx.3.0.2.4\lib\net45\System.Windows.Interactivity.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="System.Xml" />
|
||||
<Reference Include="Microsoft.CSharp" />
|
||||
<Reference Include="System.Core" />
|
||||
<Reference Include="System.Xml.Linq" />
|
||||
<Reference Include="System.Data.DataSetExtensions" />
|
||||
<Reference Include="System.Net.Http" />
|
||||
<Reference Include="System.Xaml">
|
||||
<RequiredTargetFramework>4.0</RequiredTargetFramework>
|
||||
</Reference>
|
||||
<Reference Include="WindowsBase" />
|
||||
<Reference Include="PresentationCore" />
|
||||
<Reference Include="PresentationFramework" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ApplicationDefinition Include="App.xaml">
|
||||
<Generator>MSBuild:Compile</Generator>
|
||||
<SubType>Designer</SubType>
|
||||
</ApplicationDefinition>
|
||||
<Compile Include="GeneratedResultWindow.xaml.cs">
|
||||
<DependentUpon>GeneratedResultWindow.xaml</DependentUpon>
|
||||
</Compile>
|
||||
<Compile Include="Record.cs" />
|
||||
<Compile Include="RecordFormatter.cs" />
|
||||
<Compile Include="RecordFormatterGenerationMethods.cs" />
|
||||
<Compile Include="Types.cs" />
|
||||
<Compile Include="Utility.cs" />
|
||||
<Page Include="GeneratedResultWindow.xaml">
|
||||
<SubType>Designer</SubType>
|
||||
<Generator>MSBuild:Compile</Generator>
|
||||
</Page>
|
||||
<Page Include="RecordEditor.xaml">
|
||||
<Generator>MSBuild:Compile</Generator>
|
||||
<SubType>Designer</SubType>
|
||||
</Page>
|
||||
<Compile Include="App.xaml.cs">
|
||||
<DependentUpon>App.xaml</DependentUpon>
|
||||
<SubType>Code</SubType>
|
||||
</Compile>
|
||||
<Compile Include="RecordEditor.xaml.cs">
|
||||
<DependentUpon>RecordEditor.xaml</DependentUpon>
|
||||
<SubType>Code</SubType>
|
||||
</Compile>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="Properties\AssemblyInfo.cs">
|
||||
<SubType>Code</SubType>
|
||||
</Compile>
|
||||
<Compile Include="Properties\Resources.Designer.cs">
|
||||
<AutoGen>True</AutoGen>
|
||||
<DesignTime>True</DesignTime>
|
||||
<DependentUpon>Resources.resx</DependentUpon>
|
||||
</Compile>
|
||||
<Compile Include="Properties\Settings.Designer.cs">
|
||||
<AutoGen>True</AutoGen>
|
||||
<DependentUpon>Settings.settings</DependentUpon>
|
||||
<DesignTimeSharedInput>True</DesignTimeSharedInput>
|
||||
</Compile>
|
||||
<EmbeddedResource Include="Properties\Resources.resx">
|
||||
<Generator>ResXFileCodeGenerator</Generator>
|
||||
<LastGenOutput>Resources.Designer.cs</LastGenOutput>
|
||||
</EmbeddedResource>
|
||||
<None Include="packages.config" />
|
||||
<None Include="Properties\Settings.settings">
|
||||
<Generator>SettingsSingleFileGenerator</Generator>
|
||||
<LastGenOutput>Settings.Designer.cs</LastGenOutput>
|
||||
</None>
|
||||
<AppDesigner Include="Properties\" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Include="App.config" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<BootstrapperPackage Include=".NETFramework,Version=v4.5.2">
|
||||
<Visible>False</Visible>
|
||||
<ProductName>Microsoft .NET Framework 4.5.2 %28x86 and x64%29</ProductName>
|
||||
<Install>true</Install>
|
||||
</BootstrapperPackage>
|
||||
<BootstrapperPackage Include="Microsoft.Net.Framework.3.5.SP1">
|
||||
<Visible>False</Visible>
|
||||
<ProductName>.NET Framework 3.5 SP1</ProductName>
|
||||
<Install>false</Install>
|
||||
</BootstrapperPackage>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Resource Include="CRG.ico" />
|
||||
</ItemGroup>
|
||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||
<!-- To modify your build process, add your task inside one of the targets below and uncomment it.
|
||||
Other similar extension points exist, see Microsoft.Common.targets.
|
||||
<Target Name="BeforeBuild">
|
||||
</Target>
|
||||
<Target Name="AfterBuild">
|
||||
</Target>
|
||||
-->
|
||||
</Project>
|
||||
|
|
@ -1,24 +0,0 @@
|
|||
<controls:MetroWindow x:Class="CharacterRecordsGenerator.GeneratedResultWindow"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:local="clr-namespace:CharacterRecordsGenerator"
|
||||
xmlns:controls="http://metro.mahapps.com/winfx/xaml/controls"
|
||||
mc:Ignorable="d"
|
||||
GlowBrush="{DynamicResource AccentColorBrush}"
|
||||
Title="Generated Records" Height="500" Width="850" TitleCaps="False">
|
||||
<Grid>
|
||||
<controls:MetroAnimatedTabControl Margin="10">
|
||||
<TabItem Header="Employment">
|
||||
<TextBox x:Name="EmploymentBox" IsReadOnly="True" IsUndoEnabled="False" AutoWordSelection="True" FontFamily="Consolas" TextWrapping="Wrap"/>
|
||||
</TabItem>
|
||||
<TabItem Header="Medical">
|
||||
<TextBox x:Name="MedicalBox" IsReadOnly="True" IsUndoEnabled="False" AutoWordSelection="True" FontFamily="Consolas" TextWrapping="Wrap"/>
|
||||
</TabItem>
|
||||
<TabItem Header="Security">
|
||||
<TextBox x:Name="SecurityBox" IsReadOnly="True" IsUndoEnabled="False" AutoWordSelection="True" FontFamily="Consolas" TextWrapping="Wrap"/>
|
||||
</TabItem>
|
||||
</controls:MetroAnimatedTabControl>
|
||||
</Grid>
|
||||
</controls:MetroWindow>
|
||||
|
|
@ -1,18 +0,0 @@
|
|||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
public partial class GeneratedResultWindow
|
||||
{
|
||||
public GeneratedResultWindow()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
|
||||
public GeneratedResultWindow(Record record) : this()
|
||||
{
|
||||
var formatter = new RecordFormatter(record);
|
||||
EmploymentBox.Text = formatter.EmploymentRecords;
|
||||
MedicalBox.Text = formatter.MedicalRecords;
|
||||
SecurityBox.Text = formatter.SecurityRecords;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,54 +0,0 @@
|
|||
using System.Reflection;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Windows;
|
||||
|
||||
// General Information about an assembly is controlled through the following
|
||||
// set of attributes. Change these attribute values to modify the information
|
||||
// associated with an assembly.
|
||||
[assembly: AssemblyTitle("CRG")]
|
||||
[assembly: AssemblyDescription("")]
|
||||
[assembly: AssemblyConfiguration("")]
|
||||
[assembly: AssemblyCompany("")]
|
||||
[assembly: AssemblyProduct("Character Records Generator")]
|
||||
[assembly: AssemblyCopyright("Copyright © 2022")]
|
||||
[assembly: AssemblyTrademark("")]
|
||||
[assembly: AssemblyCulture("")]
|
||||
[assembly: System.Runtime.CompilerServices.InternalsVisibleTo("CharacterRecordsGeneratorTests")]
|
||||
|
||||
// Setting ComVisible to false makes the types in this assembly not visible
|
||||
// to COM components. If you need to access a type in this assembly from
|
||||
// COM, set the ComVisible attribute to true on that type.
|
||||
[assembly: ComVisible(false)]
|
||||
|
||||
//In order to begin building localizable applications, set
|
||||
//<UICulture>CultureYouAreCodingWith</UICulture> in your .csproj file
|
||||
//inside a <PropertyGroup>. For example, if you are using US english
|
||||
//in your source files, set the <UICulture> to en-US. Then uncomment
|
||||
//the NeutralResourceLanguage attribute below. Update the "en-US" in
|
||||
//the line below to match the UICulture setting in the project file.
|
||||
|
||||
//[assembly: NeutralResourcesLanguage("en-US", UltimateResourceFallbackLocation.Satellite)]
|
||||
|
||||
|
||||
[assembly: ThemeInfo(
|
||||
ResourceDictionaryLocation.None, //where theme specific resource dictionaries are located
|
||||
//(used if a resource is not found in the page,
|
||||
// or application resource dictionaries)
|
||||
ResourceDictionaryLocation.SourceAssembly //where the generic resource dictionary is located
|
||||
//(used if a resource is not found in the page,
|
||||
// app, or any theme specific resource dictionaries)
|
||||
)]
|
||||
|
||||
|
||||
// Version information for an assembly consists of the following four values:
|
||||
//
|
||||
// Major Version
|
||||
// Minor Version
|
||||
// Build Number
|
||||
// Revision
|
||||
//
|
||||
// You can specify all the values or you can default the Build and Revision Numbers
|
||||
// by using the '*' as shown below:
|
||||
// [assembly: AssemblyVersion("1.0.*")]
|
||||
[assembly: AssemblyVersion("2.2.*")]
|
||||
[assembly: AssemblyFileVersion("1.1.0.0")]
|
||||
|
|
@ -1,63 +0,0 @@
|
|||
//------------------------------------------------------------------------------
|
||||
// <auto-generated>
|
||||
// This code was generated by a tool.
|
||||
// Runtime Version:4.0.30319.42000
|
||||
//
|
||||
// Changes to this file may cause incorrect behavior and will be lost if
|
||||
// the code is regenerated.
|
||||
// </auto-generated>
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
namespace CharacterRecordsGenerator.Properties {
|
||||
using System;
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// A strongly-typed resource class, for looking up localized strings, etc.
|
||||
/// </summary>
|
||||
// This class was auto-generated by the StronglyTypedResourceBuilder
|
||||
// class via a tool like ResGen or Visual Studio.
|
||||
// To add or remove a member, edit your .ResX file then rerun ResGen
|
||||
// with the /str option, or rebuild your VS project.
|
||||
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "16.0.0.0")]
|
||||
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
|
||||
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
|
||||
internal class Resources {
|
||||
|
||||
private static global::System.Resources.ResourceManager resourceMan;
|
||||
|
||||
private static global::System.Globalization.CultureInfo resourceCulture;
|
||||
|
||||
[global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
|
||||
internal Resources() {
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the cached ResourceManager instance used by this class.
|
||||
/// </summary>
|
||||
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
|
||||
internal static global::System.Resources.ResourceManager ResourceManager {
|
||||
get {
|
||||
if (object.ReferenceEquals(resourceMan, null)) {
|
||||
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("CharacterRecordsGenerator.Properties.Resources", typeof(Resources).Assembly);
|
||||
resourceMan = temp;
|
||||
}
|
||||
return resourceMan;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Overrides the current thread's CurrentUICulture property for all
|
||||
/// resource lookups using this strongly typed resource class.
|
||||
/// </summary>
|
||||
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
|
||||
internal static global::System.Globalization.CultureInfo Culture {
|
||||
get {
|
||||
return resourceCulture;
|
||||
}
|
||||
set {
|
||||
resourceCulture = value;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,117 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<root>
|
||||
<!--
|
||||
Microsoft ResX Schema
|
||||
|
||||
Version 2.0
|
||||
|
||||
The primary goals of this format is to allow a simple XML format
|
||||
that is mostly human readable. The generation and parsing of the
|
||||
various data types are done through the TypeConverter classes
|
||||
associated with the data types.
|
||||
|
||||
Example:
|
||||
|
||||
... ado.net/XML headers & schema ...
|
||||
<resheader name="resmimetype">text/microsoft-resx</resheader>
|
||||
<resheader name="version">2.0</resheader>
|
||||
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
|
||||
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
|
||||
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
|
||||
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
|
||||
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
|
||||
<value>[base64 mime encoded serialized .NET Framework object]</value>
|
||||
</data>
|
||||
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
|
||||
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
|
||||
<comment>This is a comment</comment>
|
||||
</data>
|
||||
|
||||
There are any number of "resheader" rows that contain simple
|
||||
name/value pairs.
|
||||
|
||||
Each data row contains a name, and value. The row also contains a
|
||||
type or mimetype. Type corresponds to a .NET class that support
|
||||
text/value conversion through the TypeConverter architecture.
|
||||
Classes that don't support this are serialized and stored with the
|
||||
mimetype set.
|
||||
|
||||
The mimetype is used for serialized objects, and tells the
|
||||
ResXResourceReader how to depersist the object. This is currently not
|
||||
extensible. For a given mimetype the value must be set accordingly:
|
||||
|
||||
Note - application/x-microsoft.net.object.binary.base64 is the format
|
||||
that the ResXResourceWriter will generate, however the reader can
|
||||
read any of the formats listed below.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.binary.base64
|
||||
value : The object must be serialized with
|
||||
: System.Serialization.Formatters.Binary.BinaryFormatter
|
||||
: and then encoded with base64 encoding.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.soap.base64
|
||||
value : The object must be serialized with
|
||||
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter
|
||||
: and then encoded with base64 encoding.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.bytearray.base64
|
||||
value : The object must be serialized into a byte array
|
||||
: using a System.ComponentModel.TypeConverter
|
||||
: and then encoded with base64 encoding.
|
||||
-->
|
||||
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
|
||||
<xsd:element name="root" msdata:IsDataSet="true">
|
||||
<xsd:complexType>
|
||||
<xsd:choice maxOccurs="unbounded">
|
||||
<xsd:element name="metadata">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" />
|
||||
<xsd:attribute name="type" type="xsd:string" />
|
||||
<xsd:attribute name="mimetype" type="xsd:string" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="assembly">
|
||||
<xsd:complexType>
|
||||
<xsd:attribute name="alias" type="xsd:string" />
|
||||
<xsd:attribute name="name" type="xsd:string" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="data">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
||||
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" msdata:Ordinal="1" />
|
||||
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
|
||||
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="resheader">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" use="required" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
</xsd:choice>
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
</xsd:schema>
|
||||
<resheader name="resmimetype">
|
||||
<value>text/microsoft-resx</value>
|
||||
</resheader>
|
||||
<resheader name="version">
|
||||
<value>2.0</value>
|
||||
</resheader>
|
||||
<resheader name="reader">
|
||||
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
</root>
|
||||
|
|
@ -1,26 +0,0 @@
|
|||
//------------------------------------------------------------------------------
|
||||
// <auto-generated>
|
||||
// This code was generated by a tool.
|
||||
// Runtime Version:4.0.30319.42000
|
||||
//
|
||||
// Changes to this file may cause incorrect behavior and will be lost if
|
||||
// the code is regenerated.
|
||||
// </auto-generated>
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
namespace CharacterRecordsGenerator.Properties {
|
||||
|
||||
|
||||
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
|
||||
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "16.10.0.0")]
|
||||
internal sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase {
|
||||
|
||||
private static Settings defaultInstance = ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings())));
|
||||
|
||||
public static Settings Default {
|
||||
get {
|
||||
return defaultInstance;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,7 +0,0 @@
|
|||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<SettingsFile xmlns="uri:settings" CurrentProfile="(Default)">
|
||||
<Profiles>
|
||||
<Profile Name="(Default)" />
|
||||
</Profiles>
|
||||
<Settings />
|
||||
</SettingsFile>
|
||||
|
|
@ -1,124 +0,0 @@
|
|||
using System;
|
||||
using ProtoBuf;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
[ProtoContract]
|
||||
public class Record
|
||||
{
|
||||
// Defaults defined here will automatically populate the form on program load
|
||||
[ProtoMember(1)]
|
||||
public string FirstName { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(2)]
|
||||
public string MiddleName { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(3)]
|
||||
public string LastName { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(4)]
|
||||
public string SpokenLanguages { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(5)]
|
||||
public SpeciesType Species { get; set; } = SpeciesType.Human;
|
||||
|
||||
|
||||
[ProtoMember(6)]
|
||||
public string Pronouns { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(7)]
|
||||
public DateTime BirthDate { get; set; } = Info.IcDate;
|
||||
|
||||
[ProtoMember(8)]
|
||||
public double? CharHeight { get; set; } = null;
|
||||
|
||||
[ProtoMember(9)]
|
||||
public double? Weight { get; set; } = null;
|
||||
|
||||
[ProtoMember(10)]
|
||||
public string SkinColor { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(11)]
|
||||
public string EyeColor { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(12)]
|
||||
public string DistinguishingFeatures { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(13)]
|
||||
public string HairColor { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(14)]
|
||||
public string EmployedAs { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(15)]
|
||||
public string Citizenship { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(16)]
|
||||
public SpeciesSubType Subspecies { get; set; } = SpeciesSubType.None;
|
||||
|
||||
[ProtoMember(17)]
|
||||
public string PublicNotes { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(18)]
|
||||
public string NextOfKin { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(19)]
|
||||
public string MedicalPostmortem { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(20)]
|
||||
public string MedicalAllergies { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(21)]
|
||||
public string MedicalCurrentPrescriptions { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(22)]
|
||||
public string MedicalHistory { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(23)]
|
||||
public string MedicalSurgicalHistory { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(24)]
|
||||
public bool NoBorg { get; set; } = false;
|
||||
|
||||
[ProtoMember(25)]
|
||||
public string MedicalPsychDisorders { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(26)]
|
||||
public bool NoRevive { get; set; } = false;
|
||||
|
||||
[ProtoMember(27)]
|
||||
public bool NoProsthetic { get; set; } = false;
|
||||
|
||||
[ProtoMember(28)]
|
||||
public string MedicalPhysicalEvaluations { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(29)]
|
||||
public string MedicalPsychEvaluations { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(30)]
|
||||
public string SecurityRecords { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(31)]
|
||||
public string SecurityNotes { get; set; } = string.Empty;
|
||||
|
||||
// 32 was used for EmploymentPublicRecord, now empty
|
||||
|
||||
[ProtoMember(33)]
|
||||
public string EmploymentExperience { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(35)]
|
||||
public string EmploymentFormalEducation { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(36)]
|
||||
public string EmploymentNtEmploymentHistory { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(37)]
|
||||
public string EmploymentSkills { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(38)]
|
||||
public string SecurityAttitudeScc { get; set; } = string.Empty;
|
||||
|
||||
[ProtoMember(39)]
|
||||
public string SecurityAttitudeCrew { get; set; } = string.Empty;
|
||||
}
|
||||
}
|
||||
|
|
@ -1,465 +0,0 @@
|
|||
<controls:MetroWindow x:Class="CharacterRecordsGenerator.RecordEditor"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:local="clr-namespace:CharacterRecordsGenerator"
|
||||
xmlns:system="clr-namespace:System;assembly=mscorlib"
|
||||
xmlns:controls="http://metro.mahapps.com/winfx/xaml/controls"
|
||||
mc:Ignorable="d"
|
||||
Title="Character Records Generator" Height="697" Width="719.583"
|
||||
ShowIconOnTitleBar="True"
|
||||
ShowTitleBar="True"
|
||||
Icon="CRG.ico"
|
||||
GlowBrush="{DynamicResource AccentColorBrush}"
|
||||
TitleCharacterCasing="Normal" Loaded="WindowLoaded">
|
||||
<controls:MetroWindow.Resources>
|
||||
<!-- Species Combobox Data Source -->
|
||||
<ObjectDataProvider x:Key="SpeciesEnum" MethodName="GetValues" ObjectType="{x:Type system:Enum}">
|
||||
<ObjectDataProvider.MethodParameters>
|
||||
<x:Type TypeName="local:SpeciesType" />
|
||||
</ObjectDataProvider.MethodParameters>
|
||||
</ObjectDataProvider>
|
||||
<ObjectDataProvider x:Key="ThreatLevelEnum" MethodName="GetValues" ObjectType="{x:Type system:Enum}">
|
||||
<ObjectDataProvider.MethodParameters>
|
||||
<x:Type TypeName="local:ThreatLevel" />
|
||||
</ObjectDataProvider.MethodParameters>
|
||||
</ObjectDataProvider>
|
||||
</controls:MetroWindow.Resources>
|
||||
<Grid>
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition Width="85*"/>
|
||||
<ColumnDefinition Width="93*"/>
|
||||
</Grid.ColumnDefinitions>
|
||||
<controls:MetroAnimatedTabControl Margin="10,10,0,45" controls:TabControlHelper.IsUnderlined="True" Grid.ColumnSpan="2">
|
||||
<!-- General Character Information -->
|
||||
<TabItem Header="General">
|
||||
<ScrollViewer>
|
||||
<StackPanel>
|
||||
<TextBlock Margin="10,10,10,10"
|
||||
Text="- Bold fields with a * are required. Otherwise, if you leave a box blank, it'll just leave that field off the final result.
- Hover over the fields for tooltips! Or look at the default watermark value for an idea about what to write."
|
||||
VerticalAlignment="Top" FontWeight="SemiBold"/>
|
||||
<GroupBox Header="Basic Information" controls:ControlsHelper.ContentCharacterCasing="Normal"
|
||||
Height="200">
|
||||
<Grid>
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition Width="160" />
|
||||
<ColumnDefinition Width="260" />
|
||||
<ColumnDefinition/>
|
||||
</Grid.ColumnDefinitions>
|
||||
<TextBox Height="26"
|
||||
Margin="10,26,5,0"
|
||||
TextWrapping="Wrap" VerticalAlignment="Top"
|
||||
ToolTip="Your character's first name. Required."
|
||||
Text="{Binding Path=FirstName}" controls:TextBoxHelper.Watermark="John" TabIndex="1" />
|
||||
<TextBox Height="26"
|
||||
Margin="0,26,0,0"
|
||||
TextWrapping="Wrap"
|
||||
ToolTip="Your character's middle name." Grid.Column="1"
|
||||
HorizontalAlignment="Left" Width="120" VerticalAlignment="Top"
|
||||
Text="{Binding Path=MiddleName}" TabIndex="2" />
|
||||
<TextBox Height="26"
|
||||
Margin="125,26,0,0"
|
||||
TextWrapping="Wrap" VerticalAlignment="Top"
|
||||
ToolTip="Your character's last name. Required." Grid.Column="1"
|
||||
Text="{Binding Path=LastName}" controls:TextBoxHelper.Watermark="Doe" TabIndex="3" />
|
||||
<TextBlock Grid.Column="0" Margin="10,10,10,0" TextWrapping="Wrap"
|
||||
Text="First*" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Height="16" />
|
||||
<TextBlock HorizontalAlignment="Left" Margin="0,10,0,0" TextWrapping="Wrap"
|
||||
Text="Middle" VerticalAlignment="Top" Grid.Column="1"
|
||||
Height="15.96" Width="120"/>
|
||||
<TextBlock Margin="125,10,10,0" TextWrapping="Wrap"
|
||||
Text="Last"
|
||||
VerticalAlignment="Top" Grid.Column="1" Height="15.96"/>
|
||||
<TextBox Height="26"
|
||||
Margin="5,26,10.5,0" TextWrapping="Wrap"
|
||||
ToolTip="Your character's citizenship." Grid.Column="2"
|
||||
VerticalAlignment="Top"
|
||||
Text="{Binding Path=Citizenship}"
|
||||
controls:TextBoxHelper.Watermark="Coalition of Colonies" TabIndex="4" />
|
||||
<TextBlock HorizontalAlignment="Left" Margin="5,10,0,0" TextWrapping="Wrap"
|
||||
Text="Citizenship" VerticalAlignment="Top" Grid.Column="2"
|
||||
Height="15.96" Width="150" />
|
||||
<TextBlock Grid.Column="0" Margin="10,57,10,0" TextWrapping="Wrap"
|
||||
Text="Species*" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Height="15.96"/>
|
||||
<ComboBox Grid.Column="0" x:Name="SpeciesCombo" Margin="10,73,5,0"
|
||||
VerticalAlignment="Top"
|
||||
ItemsSource="{Binding Source={StaticResource SpeciesEnum}}"
|
||||
ToolTip="Your character's species."
|
||||
SelectionChanged="SpeciesSelectChanged" Height="26" TabIndex="5"/>
|
||||
<TextBlock HorizontalAlignment="Left" Margin="0,57,0,0" TextWrapping="Wrap"
|
||||
Text="Pronouns"
|
||||
VerticalAlignment="Top" Grid.Column="1" Height="15.96" Width="120"/>
|
||||
<TextBox HorizontalAlignment="Left" Margin="0,73,0,0"
|
||||
VerticalAlignment="Top" Width="120"
|
||||
ToolTip="Your character's preferred pronouns."
|
||||
Text="{Binding Path=Pronouns}"
|
||||
controls:TextBoxHelper.Watermark="He/Him"
|
||||
Grid.Column="1" Height="26" TabIndex="6"/>
|
||||
<DatePicker Margin="125,73,0,0" VerticalAlignment="Top"
|
||||
SelectedDateFormat="Short" DisplayDateEnd="2470-01-01"
|
||||
DisplayDateStart="1955-01-01" DisplayDate="2464-01-01" Grid.Column="1"
|
||||
Height="26"
|
||||
SelectedDate="{Binding Path=BirthDate}" TabIndex="7" />
|
||||
<TextBlock Margin="125,57,40.96,0" TextWrapping="Wrap" Text="Date of Birth*" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Grid.Column="1" Height="15.96" />
|
||||
<TextBlock Grid.Column="1" Margin="0,0,0,45" TextWrapping="Wrap" Text="Next of Kin"
|
||||
VerticalAlignment="Bottom" HorizontalAlignment="Left" />
|
||||
<TextBox Height="23" Margin="0,0,0,19" TextWrapping="Wrap"
|
||||
VerticalAlignment="Bottom" Grid.ColumnSpan="1"
|
||||
Grid.Column="1"
|
||||
controls:TextBoxHelper.Watermark="e.g. Jane Doe (mother)"
|
||||
Text="{Binding Path=NextOfKin}" TabIndex="10" />
|
||||
<TextBlock HorizontalAlignment="Left" Margin="5,0,0,45"
|
||||
TextWrapping="Wrap" Text="Employed As" VerticalAlignment="Bottom" Grid.Column="2" />
|
||||
<TextBox Height="23" Margin="5,0,10,19" TextWrapping="Wrap"
|
||||
VerticalAlignment="Bottom" controls:TextBoxHelper.Watermark="Assistant"
|
||||
Text="{Binding Path=EmployedAs}" Grid.Column="2" TabIndex="11" />
|
||||
<TextBox Grid.Column="2" Height="23" Margin="5,73,10.5,0"
|
||||
TextWrapping="Wrap" VerticalAlignment="Top" controls:TextBoxHelper.Watermark="Tau Ceti Basic, Tradeband"
|
||||
Text="{Binding Path=SpokenLanguages}" TabIndex="8" />
|
||||
<TextBlock Grid.Column="2" HorizontalAlignment="Left" Margin="5,57,0,0" TextWrapping="Wrap" Text="Spoken Languages" VerticalAlignment="Top" />
|
||||
<ComboBox Grid.Column="0" x:Name="SubSpeciesCombo" Margin="10,120,5,0"
|
||||
VerticalAlignment="Top"
|
||||
ToolTip="Your character's ethnic group or subtype."
|
||||
SelectionChanged="SpeciesSelectChanged" Height="26" TabIndex="9"
|
||||
/>
|
||||
<TextBlock Grid.Column="0" Margin="10,104,10,0" TextWrapping="Wrap"
|
||||
Text="Subtype/Ethnicity"
|
||||
VerticalAlignment="Top" Height="16"
|
||||
/>
|
||||
</Grid>
|
||||
</GroupBox>
|
||||
<GroupBox Header="Appearance" controls:ControlsHelper.ContentCharacterCasing="Normal">
|
||||
<Grid Height="125">
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition Width="550" />
|
||||
<ColumnDefinition Width="*" />
|
||||
</Grid.ColumnDefinitions>
|
||||
<TextBlock HorizontalAlignment="Left" Margin="10,10,0,0" TextWrapping="Wrap"
|
||||
VerticalAlignment="Top" Text="Height (cm)" />
|
||||
<controls:NumericUpDown Grid.Column="0"
|
||||
Value="{Binding Path=CharHeight}" controls:TextBoxHelper.Watermark="170.0"
|
||||
HorizontalAlignment="Left" Margin="10,26,0,0"
|
||||
Minimum="10" Maximum="400" HideUpDownButtons="True" Width="1"
|
||||
StringFormat="N1" VerticalAlignment="Top"
|
||||
HorizontalContentAlignment="Center" TabIndex="12" />
|
||||
<controls:NumericUpDown Grid.Column="0"
|
||||
Value="{Binding Path=Weight}" controls:TextBoxHelper.Watermark="65.0"
|
||||
HorizontalAlignment="Left" Margin="77.033,26,0,0"
|
||||
Minimum="10" Maximum="1000" HideUpDownButtons="True" Width="1"
|
||||
StringFormat="N1" VerticalAlignment="Top"
|
||||
HorizontalContentAlignment="Center" TabIndex="13" />
|
||||
<TextBlock Grid.Column="0" HorizontalAlignment="Left" Margin="77.033,10,0,0"
|
||||
TextWrapping="Wrap"
|
||||
VerticalAlignment="Top" Text="Weight (kg)" />
|
||||
<TextBlock Grid.Column="0" HorizontalAlignment="Left" Margin="144.033,10,0,0"
|
||||
TextWrapping="Wrap"
|
||||
Text="Skin/Body Color" VerticalAlignment="Top" />
|
||||
<TextBox Grid.Column="0" Height="23"
|
||||
Margin="144.033,25.96,261.967,0" TextWrapping="Wrap" VerticalAlignment="Top"
|
||||
ToolTip="The color of your character's skin/scales/fur/chassis."
|
||||
controls:TextBoxHelper.Watermark="Brown"
|
||||
d:LayoutOverrides="HorizontalAlignment" Text="{Binding Path=SkinColor}" TabIndex="14" />
|
||||
<TextBlock Grid.Column="0" HorizontalAlignment="Left" Margin="10,57,0,52"
|
||||
TextWrapping="Wrap"
|
||||
Text="Distinguishing Features" d:LayoutOverrides="Height" />
|
||||
<TextBox Grid.Column="0" Height="23"
|
||||
Margin="10,0,0,26" Text="{Binding Path=DistinguishingFeatures}"
|
||||
controls:TextBoxHelper.Watermark="A full sleeve on their right arm; mechanical eyes."
|
||||
TextWrapping="Wrap" VerticalAlignment="Bottom" TabIndex="17" />
|
||||
<TextBox Grid.Column="0" Height="23"
|
||||
Margin="0,25.96,136.967,0"
|
||||
TextWrapping="Wrap" VerticalAlignment="Top" HorizontalAlignment="Right"
|
||||
Width="120" d:LayoutOverrides="HorizontalAlignment"
|
||||
Text="{Binding Path=HairColor}" TabIndex="15" />
|
||||
<TextBlock Grid.Column="0" Margin="0,10,203.403,0" TextWrapping="Wrap"
|
||||
Text="Hair Color"
|
||||
VerticalAlignment="Top" HorizontalAlignment="Right"
|
||||
d:LayoutOverrides="HorizontalAlignment" />
|
||||
<TextBox Grid.Column="0" HorizontalAlignment="Right"
|
||||
Margin="0,25.96,0,0"
|
||||
TextWrapping="Wrap" Width="131.967" Height="26" VerticalAlignment="Top"
|
||||
Text="{Binding Path=EyeColor}" TabIndex="16" />
|
||||
<TextBlock Grid.Column="0" HorizontalAlignment="Right" Margin="0,10,81.957,0"
|
||||
TextWrapping="Wrap"
|
||||
Text="Eye Color" VerticalAlignment="Top" />
|
||||
</Grid>
|
||||
</GroupBox>
|
||||
<GroupBox Header="Additional Information" controls:ControlsHelper.ContentCharacterCasing="Normal"
|
||||
Height="150">
|
||||
<Grid>
|
||||
<Grid.RowDefinitions>
|
||||
<RowDefinition Height="122"
|
||||
MinHeight="60"/>
|
||||
<RowDefinition/>
|
||||
</Grid.RowDefinitions>
|
||||
<GridSplitter Grid.Row="0"
|
||||
VerticalAlignment="Bottom"
|
||||
Margin="0"
|
||||
Height="10" />
|
||||
<TextBox Grid.Row="0"
|
||||
Margin="0,10,0,10"
|
||||
AcceptsReturn="True"
|
||||
TextWrapping="Wrap"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.Watermark="Any additional notes that should be present on all three records.

e.g.
John Doe is here for a two-month cultural exchange.
John Doe has been known to [...]."
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
ToolTip="Additional public notes that will be visible on all three records."
|
||||
Text="{Binding Path=PublicNotes}"
|
||||
d:LayoutOverrides="VerticalAlignment"/>
|
||||
</Grid>
|
||||
</GroupBox>
|
||||
</StackPanel>
|
||||
</ScrollViewer>
|
||||
</TabItem>
|
||||
<!-- Employment Information -->
|
||||
<TabItem Header="Employment">
|
||||
<ScrollViewer>
|
||||
<Grid>
|
||||
<Grid.RowDefinitions>
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
</Grid.RowDefinitions>
|
||||
<GridSplitter Grid.Row="0" Grid.ColumnSpan="1" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="1" Grid.ColumnSpan="1" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="2" Grid.ColumnSpan="1" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<TextBlock Grid.Row="0" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Employment History"/>
|
||||
<TextBlock Grid.Row="1" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Formal Education / Qualifications"/>
|
||||
<TextBlock Grid.Row="2" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Other Skills"/>
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap"
|
||||
controls:TextBoxHelper.Watermark="e.g.
[2450-2454] Police Cadet, Mendell City
[2454-2460] Police Officer, Mendell City
[2464-Ongoing] Security Officer, Nanotrasen"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True" Grid.Row="0"
|
||||
ToolTip="Previous jobs that the character has worked. One per line.
No need to put hyphens/bulletpoints; they'll be added automatically.

e.g.
[2450 - 2454] Police Cadet, Mendell City
[2454 - Ongoing] Police Officer, Mendell City"
|
||||
d:LayoutOverrides="VerticalAlignment" Text="{Binding Path=EmploymentExperience}" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" Grid.Row="1"
|
||||
controls:TextBoxHelper.Watermark="e.g.
[2452] Manual handling certificate
[2460] PhD in Psychology"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
ToolTip="Formal education completed / qualifications acquired. One per-line.
No need for bulletpoints; they'll be added automatically.

e.g.
[2452] Manual handling certificate
[2460] PhD in Psychology"
|
||||
Text="{Binding Path=EmploymentFormalEducation}" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap"
|
||||
controls:TextBoxHelper.Watermark="e.g.
Currently a culinary student, set to graduate in 2465.
First Aid trained" Grid.Row="2"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
Text="{Binding Path=EmploymentSkills}"
|
||||
ToolTip="Any other skills of note, or training being undertaken currently.

e.g.
First Aid training
Chemistry student at Mendell University" />
|
||||
</Grid>
|
||||
</ScrollViewer>
|
||||
</TabItem>
|
||||
<!-- Medical Information -->
|
||||
<TabItem Header="Medical">
|
||||
<ScrollViewer>
|
||||
<Grid>
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition Width="92*"/>
|
||||
<ColumnDefinition Width="587*"/>
|
||||
</Grid.ColumnDefinitions>
|
||||
<Grid.RowDefinitions>
|
||||
<RowDefinition Height="50" />
|
||||
<!-- Checkboxes -->
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="0" />
|
||||
</Grid.RowDefinitions>
|
||||
<!-- Opt-Outs -->
|
||||
<Grid Grid.Row="0" ToolTip="If the character should not be borged." Grid.ColumnSpan="2">
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition Width="163*"/>
|
||||
<ColumnDefinition Width="163*"/>
|
||||
<ColumnDefinition Width="163*"/>
|
||||
<ColumnDefinition Width="163*"/>
|
||||
<ColumnDefinition Width="163*"/>
|
||||
</Grid.ColumnDefinitions>
|
||||
<CheckBox x:Name="NoBorg" Content="Do Not Borgify"
|
||||
HorizontalAlignment="Left"
|
||||
Margin="0,26,0,0"
|
||||
IsChecked="{Binding Path=NoBorg}"
|
||||
ToolTip="If the character should not be borged." TabIndex="1" Grid.Column="0" />
|
||||
<CheckBox x:Name="NoRevive" Content="Do Not Resuscitate"
|
||||
HorizontalAlignment="Left"
|
||||
Margin="-10,26,0,0"
|
||||
IsChecked="{Binding Path=NoRevive}"
|
||||
ToolTip="If the character should not be revived." TabIndex="2" Grid.Column="1" />
|
||||
<CheckBox x:Name="NoProsthetic" Content="No Prosthetics"
|
||||
HorizontalAlignment="Left"
|
||||
Margin="0,26,0,0"
|
||||
IsChecked="{Binding Path=NoProsthetic}"
|
||||
ToolTip="If the character should not be fitted with prosthetics." TabIndex="3" Grid.Column="2" />
|
||||
</Grid>
|
||||
<!-- user-resizable stuff is fun! Not. -->
|
||||
<GridSplitter Grid.Row="1" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Height="10" />
|
||||
<GridSplitter Grid.Row="2" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="3" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="4" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="5" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="6" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="7" Grid.ColumnSpan="2" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<TextBlock Grid.Row="0" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Opt-outs"/>
|
||||
<TextBlock Grid.Row="1" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Postmortem Instructions"/>
|
||||
<TextBlock Grid.Row="2" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Allergies"/>
|
||||
<TextBlock Grid.Row="3" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Current Medications/Prescriptions"/>
|
||||
<TextBlock Grid.Row="4" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Medication History"/>
|
||||
<TextBlock Grid.Row="5" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Surgical History"/>
|
||||
<TextBlock Grid.Row="6" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Physical Evaluations"/>
|
||||
<TextBlock Grid.Row="7" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Documented Psychological Disorders"/>
|
||||
<TextBlock Grid.Row="8" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Psychological Evaluations"/>
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
Cremate and return remains to next of kin, located on Xanu Prime."
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Instructions to execute upon death." Grid.Row="1"
|
||||
Text="{Binding Path=MedicalPostmortem}" TabIndex="4" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True" TextWrapping="Wrap"
|
||||
controls:TextBoxHelper.Watermark="e.g.
Peanuts (severe)
Latex (minor irritation)"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="List of allergies. One per line."
|
||||
Grid.Row="2" Text="{Binding Path=MedicalAllergies}" TabIndex="5" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True" TextWrapping="Wrap"
|
||||
controls:TextBoxHelper.Watermark="e.g.
2x 5u Perconol (daily) for back pain."
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Current prescriptions, etc. One per line." Grid.Row="3"
|
||||
Text="{Binding Path=MedicalCurrentPrescriptions}" TabIndex="6" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
[12/07/2462 - 12/08/2462] 5u Hyperzine (daily) for heart injury.
[14/09/2463 - 16/09/2463] 30u Nightlife (daily) prescribed for a weekend party."
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Dated records of medical history. One per line." Grid.Row="4"
|
||||
Text="{Binding Path=MedicalHistory}" TabIndex="7" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
[09/04/2460] Triple coronary artery bypass"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Dated records of surgical history. One per line." Grid.Row="5"
|
||||
Text="{Binding Path=MedicalSurgicalHistory}" TabIndex="8" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
[30/03/2458] Passed - Fully able for work.
[29/03/2459] Passed - Fully able for work."
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Dated records of physical evaluations. One per line." Grid.Row="6"
|
||||
Text="{Binding Path=MedicalPhysicalEvaluations}" TabIndex="9" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
Obsessive-compulsive disorder"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Any documented psychological disorders. One per line." Grid.Row="7"
|
||||
Text="{Binding Path=MedicalPsychDisorders}" TabIndex="10" Grid.ColumnSpan="2" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
[04/02/2459] Passed - John Doe is an example of excellent mental health."
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
ToolTip="Dated records of psychological evaluations. One per line." Grid.Row="8"
|
||||
Text="{Binding Path=MedicalPsychEvaluations}" TabIndex="11" Grid.ColumnSpan="2" />
|
||||
</Grid>
|
||||
</ScrollViewer>
|
||||
</TabItem>
|
||||
<!-- Security Information -->
|
||||
<TabItem Header="Security">
|
||||
<ScrollViewer>
|
||||
<Grid>
|
||||
<Grid.RowDefinitions>
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="122" MinHeight="60" />
|
||||
<RowDefinition Height="0" />
|
||||
</Grid.RowDefinitions>
|
||||
<Grid Grid.Row="0">
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition Width="1*"/>
|
||||
<ColumnDefinition Width="1*"/>
|
||||
</Grid.ColumnDefinitions>
|
||||
<TextBlock Grid.Column="0" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Attitude Towards the SCC"/>
|
||||
<TextBlock Grid.Column="1" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Attitude Towards the Crew"/>
|
||||
<TextBox Grid.Column="0" Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
John Doe displays a positive attitude towards the corporation."
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
ToolTip="A brief summary of the character's perceived attitude towards the conglomerate."
|
||||
Text="{Binding Path=SecurityAttitudeScc}" />
|
||||
<TextBox Grid.Column="1" Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
John Doe is generally supportive of the rest of the crew. He is rarely involved in disputes."
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
ToolTip="A brief summary of the character's perceived attitude towards the rest of the crew."
|
||||
Text="{Binding Path=SecurityAttitudeCrew}" />
|
||||
</Grid>
|
||||
|
||||
<GridSplitter Grid.Row="0" Grid.ColumnSpan="1" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="1" Grid.ColumnSpan="1" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<GridSplitter Grid.Row="2" Grid.ColumnSpan="1" HorizontalAlignment="Stretch"
|
||||
VerticalAlignment="Bottom" Margin="0" Height="10" />
|
||||
<TextBlock Grid.Row="1" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Arrest History"/>
|
||||
<TextBlock Grid.Row="2" Grid.ColumnSpan="2" HorizontalAlignment="Stretch" FontWeight="DemiBold"
|
||||
VerticalAlignment="Top" Margin="5,5,5,0" Text="Other Notes"/>
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
[23/04/2464] Battery - Some additional notes"
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
ToolTip="List of charges, dated if possible. One per line." Grid.Row="1"
|
||||
Text="{Binding Path=SecurityRecords}" />
|
||||
<TextBox Margin="0,21,0,10" AcceptsReturn="True"
|
||||
TextWrapping="Wrap" controls:TextBoxHelper.Watermark="e.g.
Whilst John Doe's record is clean, they're known to have unsavoury associates outside of work."
|
||||
VerticalScrollBarVisibility="Auto"
|
||||
controls:TextBoxHelper.IsSpellCheckContextMenuEnabled="True"
|
||||
ToolTip="Any things security should keep in mind when dealing with you. One per line."
|
||||
Grid.Row="2" Text="{Binding Path=SecurityNotes}" />
|
||||
</Grid>
|
||||
</ScrollViewer>
|
||||
</TabItem>
|
||||
</controls:MetroAnimatedTabControl>
|
||||
<!-- Footer -->
|
||||
<Grid Height="45" VerticalAlignment="Bottom" Grid.ColumnSpan="2">
|
||||
<Button Content="Open" HorizontalAlignment="Left" Margin="10,9.66,0,10" Width="75"
|
||||
d:LayoutOverrides="Height" Click="OpenContent" />
|
||||
<Button Content="Save" HorizontalAlignment="Left" Margin="90,10,0,9.66" Width="75"
|
||||
d:LayoutOverrides="Height" Click="SaveContent" />
|
||||
<Button Content="Save As" HorizontalAlignment="Left" Margin="170,10,0,9.66"
|
||||
Width="75" d:LayoutOverrides="Height" Click="SaveContentAs" />
|
||||
<Button Content="Generate" HorizontalAlignment="Right" Margin="0,10.3,10,9.66"
|
||||
Width="75" d:LayoutOverrides="Height" Click="GenerateRecord" />
|
||||
<Label x:Name="VersionLabel" Content="Version" HorizontalAlignment="Left" Margin="250,9,0,0" VerticalAlignment="Top" Foreground="Gray"/>
|
||||
</Grid>
|
||||
</Grid>
|
||||
</controls:MetroWindow>
|
||||
|
|
@ -1,171 +0,0 @@
|
|||
using MahApps.Metro.Controls.Dialogs;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Windows;
|
||||
using System.Windows.Controls;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
/// <summary>
|
||||
/// Interaction logic for RecordEditor.xaml
|
||||
/// </summary>
|
||||
public partial class RecordEditor
|
||||
{
|
||||
public RecordEditor()
|
||||
{
|
||||
// Initialize the record object used for storage and generation
|
||||
Data = new Record();
|
||||
DataContext = Data;
|
||||
ProtoBuf.Serializer.PrepareSerializer<SpeciesType>();
|
||||
ProtoBuf.Serializer.PrepareSerializer<SpeciesSubType>();
|
||||
ProtoBuf.Serializer.PrepareSerializer<Record>();
|
||||
InitializeComponent();
|
||||
SubSpeciesCombo.ItemsSource = GetSpeciesOptions();
|
||||
VersionLabel.Content = $"v{Utility.GetVersion()}";
|
||||
}
|
||||
|
||||
private Record Data { get; set; }
|
||||
private string _currentFilePath;
|
||||
|
||||
private void SpeciesSelectChanged(object sender, SelectionChangedEventArgs e)
|
||||
{
|
||||
if (SpeciesCombo.SelectionBoxItem == null)
|
||||
return;
|
||||
|
||||
var type = (SpeciesType)SpeciesCombo.SelectedValue;
|
||||
|
||||
Debug.WriteLine("Updating subspecies types.");
|
||||
var types = GetSpeciesOptions(type);
|
||||
var itemsSource = types as IList<string> ?? types.ToList();
|
||||
SubSpeciesCombo.ItemsSource = itemsSource;
|
||||
Debug.WriteLine($"New types: {string.Join(",", itemsSource)}");
|
||||
}
|
||||
|
||||
private void WindowLoaded(object sender, RoutedEventArgs e)
|
||||
{
|
||||
SpeciesCombo.SelectedIndex = 1;
|
||||
}
|
||||
|
||||
private void GenerateRecord(object sender, RoutedEventArgs e)
|
||||
{
|
||||
// Update medical checkboxes.
|
||||
Data.NoBorg = NoBorg.IsChecked ?? false;
|
||||
Data.NoProsthetic = NoProsthetic.IsChecked ?? false;
|
||||
Data.NoRevive = NoRevive.IsChecked ?? false;
|
||||
|
||||
// Figure out what subspecies we've got.
|
||||
var subspecies = SubSpeciesCombo.SelectedItem as string;
|
||||
Data.Subspecies = subspecies != null ? Utility.SubspeciesNiceNameToEnum(subspecies) : SpeciesSubType.None;
|
||||
|
||||
// Figure out their species too.
|
||||
Data.Species = (SpeciesType)SpeciesCombo.SelectedValue;
|
||||
|
||||
var wnd = new GeneratedResultWindow(Data);
|
||||
wnd.Show();
|
||||
}
|
||||
|
||||
private async void SaveContent(object sender, RoutedEventArgs e)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(_currentFilePath))
|
||||
SaveContentAs(null, null);
|
||||
else
|
||||
{
|
||||
// have a path, attempt to save to it
|
||||
if (!File.Exists(_currentFilePath))
|
||||
{
|
||||
switch (
|
||||
await
|
||||
this.ShowMessageAsync("File Error",
|
||||
"Current file missing, renamed, or deleted. Do you want to save as another name?",
|
||||
MessageDialogStyle.AffirmativeAndNegative))
|
||||
{
|
||||
case MessageDialogResult.Negative:
|
||||
_currentFilePath = null;
|
||||
return;
|
||||
|
||||
case MessageDialogResult.Affirmative:
|
||||
SaveContentAs(null, null);
|
||||
return;
|
||||
|
||||
default:
|
||||
throw new ArgumentOutOfRangeException();
|
||||
}
|
||||
}
|
||||
// have a path, outdated extension
|
||||
if (Path.GetExtension(_currentFilePath) != ".ss13records")
|
||||
{
|
||||
switch (
|
||||
await
|
||||
this.ShowMessageAsync("Outdated File Extension",
|
||||
"This profile is using an outdated extension from an older version of the CRG. Press \"OK\" to convert to the new file extension (.ss13records).",
|
||||
MessageDialogStyle.AffirmativeAndNegative))
|
||||
{
|
||||
case MessageDialogResult.Affirmative:
|
||||
SaveContentAs(null, null);
|
||||
return;
|
||||
case MessageDialogResult.Negative:
|
||||
return;
|
||||
default:
|
||||
throw new ArgumentOutOfRangeException();
|
||||
}
|
||||
}
|
||||
|
||||
var fs = File.Open(_currentFilePath, FileMode.Truncate);
|
||||
ProtoBuf.Serializer.Serialize(fs, Data);
|
||||
}
|
||||
}
|
||||
|
||||
private async void OpenContent(object sender, RoutedEventArgs e)
|
||||
{
|
||||
var dialog = new Microsoft.Win32.OpenFileDialog
|
||||
{
|
||||
AddExtension = true,
|
||||
CheckFileExists = true,
|
||||
CheckPathExists = true,
|
||||
Filter = "Character Profiles (*.ss13records, *.ss13prof)|*.ss13records;*.ss13prof|"
|
||||
+ "All Files (*.*)|*.*"
|
||||
};
|
||||
|
||||
if (!(dialog.ShowDialog() ?? false)) return;
|
||||
|
||||
var fs = File.Open(dialog.FileName, FileMode.Open);
|
||||
try
|
||||
{
|
||||
Data = ProtoBuf.Serializer.Deserialize<Record>(fs);
|
||||
_currentFilePath = dialog.FileName;
|
||||
// So WPF updates bindings
|
||||
DataContext = Data;
|
||||
}
|
||||
catch (ProtoBuf.ProtoException)
|
||||
{
|
||||
await this.ShowMessageAsync("Profile Error", "An error occurred during loading of your profile. You may have selected a file that is not a profile file, or the profile is corrupted.");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
private void SaveContentAs(object sender, RoutedEventArgs e)
|
||||
{
|
||||
var dialog = new Microsoft.Win32.SaveFileDialog
|
||||
{
|
||||
AddExtension = true,
|
||||
CheckPathExists = true,
|
||||
Filter = "Character Profiles (*.ss13records)|*.ss13records|"
|
||||
+ "All Files (*.*)|*.*"
|
||||
};
|
||||
if (!(dialog.ShowDialog() ?? false)) return;
|
||||
var fs = File.Open(dialog.FileName, FileMode.Create);
|
||||
ProtoBuf.Serializer.Serialize(fs, Data);
|
||||
_currentFilePath = dialog.FileName;
|
||||
}
|
||||
|
||||
private static IEnumerable<string> GetSpeciesOptions() => Enum.GetValues(typeof(SpeciesSubType)).Cast<SpeciesSubType>().Select(Utility.SubspeciesNiceName);
|
||||
|
||||
private static IEnumerable<string> GetSpeciesOptions(SpeciesType limitTo) => from item in Enum.GetValues(typeof(SpeciesSubType)).Cast<SpeciesSubType>()
|
||||
let attr = item.GetAttributeOfType<SubspeciesMetaAttribute>()
|
||||
where attr != null && (attr.AssociatedSpecies == limitTo || attr.AssociatedSpecies == SpeciesType.None)
|
||||
select Utility.SubspeciesNiceName(item);
|
||||
}
|
||||
}
|
||||
|
|
@ -1,103 +0,0 @@
|
|||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
internal partial class RecordFormatter
|
||||
{
|
||||
private Record _targetRecord;
|
||||
|
||||
public RecordFormatter(Record r)
|
||||
{
|
||||
_targetRecord = r;
|
||||
UpdateSplitRecords();
|
||||
MakeCommonRecords();
|
||||
}
|
||||
|
||||
private IList<string> _publicNotes;
|
||||
|
||||
private IList<string> _MedicalPostmortem;
|
||||
private IList<string> _MedicalAllergies;
|
||||
private IList<string> _MedicalCurrentPrescriptions;
|
||||
private IList<string> _MedicalHistory;
|
||||
private IList<string> _MedicalSurgicalHistory;
|
||||
private IList<string> _MedicalPhysicalEvaluations;
|
||||
private IList<string> _MedicalPsychDisorders;
|
||||
private IList<string> _MedicalPsychEvaluations;
|
||||
|
||||
private IList<string> _securityRecords;
|
||||
private IList<string> _securityNotes;
|
||||
private IList<string> _securityAttitudeScc;
|
||||
private IList<string> _securityAttitudeCrew;
|
||||
|
||||
private IList<string> _employmentExperience;
|
||||
private IList<string> _employmentFormalEducation;
|
||||
private IList<string> _employmentSkills;
|
||||
|
||||
private void UpdateSplitRecords()
|
||||
{
|
||||
if (_targetRecord == null)
|
||||
{
|
||||
_targetRecord = new Record();
|
||||
}
|
||||
_publicNotes = _targetRecord.PublicNotes?.LineSplit();
|
||||
|
||||
// Medical
|
||||
_MedicalPostmortem = _targetRecord.MedicalPostmortem?.LineSplit();
|
||||
_MedicalAllergies = _targetRecord.MedicalAllergies?.LineSplit();
|
||||
_MedicalCurrentPrescriptions = _targetRecord.MedicalCurrentPrescriptions?.LineSplit();
|
||||
_MedicalHistory = _targetRecord.MedicalHistory?.LineSplit();
|
||||
_MedicalSurgicalHistory = _targetRecord.MedicalSurgicalHistory?.LineSplit();
|
||||
_MedicalPhysicalEvaluations = _targetRecord.MedicalPhysicalEvaluations?.LineSplit();
|
||||
_MedicalPsychDisorders = _targetRecord.MedicalPsychDisorders?.LineSplit();
|
||||
_MedicalPsychEvaluations = _targetRecord.MedicalPsychEvaluations?.LineSplit();
|
||||
|
||||
// security
|
||||
_securityRecords = _targetRecord.SecurityRecords?.LineSplit();
|
||||
_securityNotes = _targetRecord.SecurityNotes?.LineSplit();
|
||||
_securityAttitudeCrew = _targetRecord.SecurityAttitudeCrew?.LineSplit();
|
||||
_securityAttitudeScc = _targetRecord.SecurityAttitudeScc?.LineSplit();
|
||||
|
||||
// employment
|
||||
_employmentExperience = _targetRecord.EmploymentExperience?.LineSplit();
|
||||
_employmentFormalEducation = _targetRecord.EmploymentFormalEducation?.LineSplit();
|
||||
_employmentSkills = _targetRecord.EmploymentSkills?.LineSplit();
|
||||
|
||||
// flush the record cache so they're regenerated
|
||||
_commonRecords = null;
|
||||
}
|
||||
|
||||
public string EmploymentRecords => MakeEmploymentRecords();
|
||||
public string MedicalRecords => MakeMedicalRecords();
|
||||
public string SecurityRecords => MakeSecurityRecords();
|
||||
|
||||
private string _commonRecords;
|
||||
|
||||
/// <summary>
|
||||
/// Writes the <see cref="string"/> form of a record section to the specified <see cref="StringBuilder"/>, as long as there's entries to write.
|
||||
/// </summary>
|
||||
/// <param name="builder">The <see cref="StringBuilder"/> to write to.</param>
|
||||
/// <param name="header">The title for the section.</param>
|
||||
/// <param name="entries">The entries of this section.</param>
|
||||
private static void WriteSectionIfAny(ref StringBuilder builder, string header, IList<string> entries)
|
||||
{
|
||||
if (entries == null || !entries.Any() || entries[0].Trim().Length == 0)
|
||||
return;
|
||||
builder.AppendLine(header);
|
||||
builder.AppendLine(entries.FormatAsList());
|
||||
}
|
||||
|
||||
private string MakeNameLine()
|
||||
{
|
||||
var builder = new StringBuilder("Name: ");
|
||||
builder.Append(_targetRecord.FirstName.SpaceIfValue());
|
||||
builder.Append(_targetRecord.MiddleName.SpaceIfValue());
|
||||
builder.Append(_targetRecord.LastName);
|
||||
return builder.ToString();
|
||||
}
|
||||
|
||||
private static void MakeMedicalNote(ref StringBuilder b, string s) =>
|
||||
b.AppendLine($" - {s}");
|
||||
}
|
||||
}
|
||||
|
|
@ -1,248 +0,0 @@
|
|||
using Humanizer;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
internal partial class RecordFormatter
|
||||
{
|
||||
private void MakeCommonRecords()
|
||||
{
|
||||
var record = new StringBuilder();
|
||||
record.AppendLine("/// PUBLIC RECORD ///");
|
||||
|
||||
// TODO: most of this should be replaced by WriteRecordIfAny()
|
||||
if (_targetRecord.FirstName.Any())
|
||||
{
|
||||
record.AppendLine(MakeNameLine());
|
||||
}
|
||||
else
|
||||
{
|
||||
record.AppendLine("Name: Not specified.");
|
||||
}
|
||||
record.AppendLine($"Date of Birth: {_targetRecord.BirthDate.ToString("MMMM")} {_targetRecord.BirthDate.Day.Ordinalize()}, {_targetRecord.BirthDate.Year}");
|
||||
if (_targetRecord.Species != SpeciesType.None)
|
||||
{
|
||||
record.AppendLine($"Species: {_targetRecord.Species.Humanize()}"); // might fuck up the names
|
||||
}
|
||||
else
|
||||
{
|
||||
record.AppendLine("Species: Not specified.");
|
||||
}
|
||||
if (_targetRecord.Subspecies != SpeciesSubType.None)
|
||||
{
|
||||
record.AppendLine($"{_targetRecord.Subspecies.GetAttributeOfType<SubspeciesMetaAttribute>()?.FieldName ?? "Subspecies"}: {Utility.SubspeciesNiceName(_targetRecord.Subspecies)}");
|
||||
}
|
||||
if (_targetRecord.Pronouns.Any())
|
||||
{
|
||||
record.AppendLine($"Pronouns: {_targetRecord.Pronouns}");
|
||||
}
|
||||
if (_targetRecord.Citizenship.Any()) {
|
||||
record.AppendLine($"Citizenship: {_targetRecord.Citizenship}");
|
||||
}
|
||||
if (_targetRecord.SpokenLanguages.Any())
|
||||
{
|
||||
record.AppendLine($"Spoken Languages: {_targetRecord.SpokenLanguages}");
|
||||
}
|
||||
if (_targetRecord.NextOfKin.Any()) {
|
||||
record.AppendLine($"Next of Kin: {_targetRecord.NextOfKin}");
|
||||
}
|
||||
if (_targetRecord.EmployedAs.Any()) {
|
||||
record.AppendLine($"Employed As: {_targetRecord.EmployedAs}");
|
||||
}
|
||||
if (_targetRecord.CharHeight != null)
|
||||
record.AppendLine($"Height: {_targetRecord.CharHeight} cm ({Utility.CmToFeet(_targetRecord.CharHeight.Value)})");
|
||||
|
||||
if (_targetRecord.Weight != null)
|
||||
record.AppendLine($"Weight: {_targetRecord.Weight} kg ({Utility.KgToLb(_targetRecord.Weight ?? 0)} lb)");
|
||||
|
||||
// Eye color
|
||||
if (_targetRecord.EyeColor.Any())
|
||||
{
|
||||
var trimmedEye = _targetRecord.EyeColor.Trim();
|
||||
record.AppendFormat("Eye Color: {0}\n", trimmedEye.Length > 0 ? trimmedEye : "Not specified.");
|
||||
}
|
||||
if (_targetRecord.SkinColor.Any())
|
||||
{
|
||||
var bodyColor = _targetRecord.SkinColor.Trim();
|
||||
record.AppendFormat("Skin/Body Color: {0}\n", bodyColor.Length > 0 ? bodyColor : "Not specified.");
|
||||
}
|
||||
if (_targetRecord.HairColor.Any())
|
||||
{
|
||||
var hairColor = _targetRecord.HairColor.Trim();
|
||||
record.AppendFormat("Hair Color: {0}\n", hairColor.Length > 0 ? hairColor : "Not specified.");
|
||||
}
|
||||
|
||||
if (_targetRecord.DistinguishingFeatures.Any())
|
||||
{
|
||||
// identifying features
|
||||
var trimmedFeatures = _targetRecord.DistinguishingFeatures.Trim();
|
||||
record.Append("Distinguishing Features: ");
|
||||
record.AppendLine(trimmedFeatures.Length > 0 ? trimmedFeatures : "None noted.");
|
||||
}
|
||||
|
||||
record.AppendLine();
|
||||
|
||||
// general notes
|
||||
WriteSectionIfAny(ref record,
|
||||
"Additional Notes:",
|
||||
_publicNotes);
|
||||
|
||||
_commonRecords = record.ToString();
|
||||
}
|
||||
|
||||
private string MakeEmploymentRecords()
|
||||
{
|
||||
var recordText = new StringBuilder();
|
||||
if (_commonRecords.IsEmpty())
|
||||
MakeCommonRecords();
|
||||
|
||||
recordText.Append(_commonRecords);
|
||||
|
||||
if (!_employmentExperience.Any() &&
|
||||
!_employmentFormalEducation.Any() &&
|
||||
!_employmentSkills.Any())
|
||||
{
|
||||
recordText.AppendLine("/// NO EMPLOYMENT RECORD FOUND ///");
|
||||
recordText.AppendLine();
|
||||
}
|
||||
else
|
||||
{
|
||||
recordText.AppendLine("/// EMPLOYMENT RECORD ///");
|
||||
recordText.AppendLine("This information has been verified by employment agents within the External Affairs department, and any comments, questions, or concerns about the legitimacy of such must be sent in a secure document to the same department.");
|
||||
recordText.AppendLine();
|
||||
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Employment History:",
|
||||
_employmentExperience);
|
||||
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Qualifications:",
|
||||
_employmentFormalEducation);
|
||||
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Other skills:",
|
||||
_employmentSkills);
|
||||
|
||||
}
|
||||
|
||||
recordText.AppendLine($"LAST UPDATED: {Utility.HumanisedDate(Info.IcDate)}");
|
||||
return recordText.ToString();
|
||||
}
|
||||
|
||||
private string MakeMedicalRecords()
|
||||
{
|
||||
var recordText = new StringBuilder();
|
||||
if (_commonRecords.IsEmpty())
|
||||
MakeCommonRecords();
|
||||
|
||||
recordText.Append(_commonRecords);
|
||||
|
||||
// TODO: make this less horrible
|
||||
if (!_MedicalAllergies.Any() &&
|
||||
!_MedicalCurrentPrescriptions.Any() &&
|
||||
!_MedicalHistory.Any() &&
|
||||
!_MedicalSurgicalHistory.Any() &&
|
||||
!_MedicalPhysicalEvaluations.Any() &&
|
||||
!_MedicalPsychEvaluations.Any() &&
|
||||
!_MedicalPsychDisorders.Any() &&
|
||||
!_MedicalPostmortem.Any() &&
|
||||
!_targetRecord.NoBorg &&
|
||||
!_targetRecord.NoProsthetic &&
|
||||
!_targetRecord.NoRevive)
|
||||
{
|
||||
recordText.AppendLine("/// NO MEDICAL RECORD FOUND ///");
|
||||
recordText.AppendLine();
|
||||
}
|
||||
else
|
||||
{
|
||||
recordText.AppendLine("/// MEDICAL RECORD ///");
|
||||
recordText.AppendLine("The following information is protected by doctor-patient confidentiality laws. Do not release without patient's consent.");
|
||||
recordText.AppendLine();
|
||||
|
||||
if (_targetRecord.NoBorg || _targetRecord.NoProsthetic || _targetRecord.NoRevive)
|
||||
{
|
||||
recordText.AppendLine("OPT-OUTS:");
|
||||
|
||||
if (_targetRecord.NoBorg)
|
||||
MakeMedicalNote(ref recordText, "DO NOT BORGIFY");
|
||||
if (_targetRecord.NoProsthetic)
|
||||
MakeMedicalNote(ref recordText, "DO NOT INSTALL PROSTHETICS");
|
||||
if (_targetRecord.NoRevive)
|
||||
MakeMedicalNote(ref recordText, "DO NOT REVIVE");
|
||||
|
||||
recordText.AppendLine();
|
||||
}
|
||||
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"POSTMORTEM INSTRUCTIONS:",
|
||||
_MedicalPostmortem);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"ALLERGIES:",
|
||||
_MedicalAllergies);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Current Prescriptions:",
|
||||
_MedicalCurrentPrescriptions);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Surgical History:",
|
||||
_MedicalSurgicalHistory);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Medication History:",
|
||||
_MedicalHistory);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Physical Evaluations:",
|
||||
_MedicalPhysicalEvaluations);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Documented Psychological Disorders:",
|
||||
_MedicalPsychDisorders);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Psychological Evaluations:",
|
||||
_MedicalPsychEvaluations);
|
||||
|
||||
|
||||
}
|
||||
recordText.AppendLine($"LAST UPDATED: {Utility.HumanisedDate(Info.IcDate)}");
|
||||
return recordText.ToString();
|
||||
}
|
||||
|
||||
private string MakeSecurityRecords()
|
||||
{
|
||||
var recordText = new StringBuilder();
|
||||
if (_commonRecords.IsEmpty())
|
||||
MakeCommonRecords();
|
||||
|
||||
recordText.Append(_commonRecords);
|
||||
|
||||
if (!_securityRecords.Any() &&
|
||||
!_securityNotes.Any() &&
|
||||
!_securityAttitudeScc.Any() &&
|
||||
!_securityAttitudeCrew.Any())
|
||||
{
|
||||
recordText.AppendLine("/// NO SECURITY RECORD FOUND ///");
|
||||
recordText.AppendLine();
|
||||
}
|
||||
else
|
||||
{
|
||||
recordText.AppendLine("/// SECURITY RECORD ///");
|
||||
recordText.AppendLine("This information has been verified by employment agents within the External Affairs department, and any comments, questions, or concerns about the legitimacy of such must be sent in a secure document to the same department.");
|
||||
recordText.AppendLine();
|
||||
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Attitude Towards the SCC:",
|
||||
_securityAttitudeScc);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Attitude Towards the Crew:",
|
||||
_securityAttitudeCrew);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Notes:",
|
||||
_securityNotes);
|
||||
WriteSectionIfAny(ref recordText,
|
||||
"Record:",
|
||||
_securityRecords);
|
||||
}
|
||||
|
||||
recordText.AppendLine($"LAST UPDATED: {Utility.HumanisedDate(Info.IcDate)}");
|
||||
return recordText.ToString();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,136 +0,0 @@
|
|||
using ProtoBuf;
|
||||
using System;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
[ProtoContract]
|
||||
public enum SpeciesType
|
||||
{
|
||||
[ProtoEnum]
|
||||
None = 0,
|
||||
|
||||
[ProtoEnum]
|
||||
Human,
|
||||
|
||||
[ProtoEnum]
|
||||
Skrell,
|
||||
|
||||
[ProtoEnum]
|
||||
Tajara,
|
||||
|
||||
[ProtoEnum]
|
||||
Unathi,
|
||||
|
||||
[ProtoEnum]
|
||||
Vaurca,
|
||||
|
||||
[ProtoEnum]
|
||||
Diona,
|
||||
|
||||
[ProtoEnum]
|
||||
IPC
|
||||
}
|
||||
|
||||
[ProtoContract]
|
||||
public enum SpeciesSubType
|
||||
{
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.None, "N/A")]
|
||||
None = 0,
|
||||
|
||||
// SKRELL VARIANTS
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Skrell, "Axiori", "Ethnicity")]
|
||||
SkrellAxiori,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Skrell, "Xiialt", "Ethnicity")]
|
||||
SkrellXiialt,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Skrell, "Xiiori", "Ethnicity")]
|
||||
SkrellXiiori,
|
||||
|
||||
// TAJARA VARIANTS
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Tajara, "Hharar", "Ethnicity")]
|
||||
TajaraHharar,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Tajara, "Zhan-Khazan", "Ethnicity")]
|
||||
TajaraZhan,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Tajara, "Njarir'Akhran", "Ethnicity")]
|
||||
TajaraNjarir,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Tajara, "M'sai", "Ethnicity")]
|
||||
TajaraMsai,
|
||||
|
||||
// VAURCA VARIANTS
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Vaurca, "Type A (Worker)", "Classification")]
|
||||
VaurcaWorker,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Vaurca, "Type B (Warrior)", "Classification")]
|
||||
VaurcaWarrior,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Vaurca, "Type C (Breeder)", "Classification")]
|
||||
VaurcaBreeder,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Vaurca, "Type E (Bulwark)", "Classification")]
|
||||
VaurcaBulwark,
|
||||
|
||||
// IPC VARIANTS
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Baseline", "Model")]
|
||||
IpcBaseline,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Shell", "Model")]
|
||||
IpcShell,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Hephaestus G1 Industrial", "Model")]
|
||||
IpcG1,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Hephaestus G2 Industrial", "Model")]
|
||||
IpcG2,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Xion Industrial", "Model")]
|
||||
IpcXion,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Zeng-Hu Mobility", "Model")]
|
||||
IpcZengHu,
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.IPC, "Bishop Accessory", "Model")]
|
||||
IpcBishop,
|
||||
|
||||
// UNATHI VARIANTS
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Unathi, "Aut'akh", "Variant")]
|
||||
UnathiAutakh,
|
||||
|
||||
// HUMAN VARIANTS
|
||||
[ProtoEnum, SubspeciesMeta(SpeciesType.Human, "Offworlder", "Variant")]
|
||||
HumanOffworld
|
||||
}
|
||||
|
||||
// Currently unused
|
||||
[ProtoContract]
|
||||
public enum ThreatLevel
|
||||
{
|
||||
[ProtoEnum]
|
||||
Minimal = 0,
|
||||
|
||||
[ProtoEnum]
|
||||
Low,
|
||||
|
||||
[ProtoEnum]
|
||||
Medium,
|
||||
|
||||
[ProtoEnum]
|
||||
High,
|
||||
|
||||
[ProtoEnum]
|
||||
Extreme
|
||||
}
|
||||
|
||||
public static class Info
|
||||
{
|
||||
/// <summary>
|
||||
/// The current in-character date.
|
||||
/// </summary>
|
||||
public static DateTime IcDate => new DateTime(DateTime.Now.Year + 442,
|
||||
DateTime.Now.Month,
|
||||
DateTime.Now.Day);
|
||||
}
|
||||
|
||||
[AttributeUsage(AttributeTargets.Field)]
|
||||
public class SubspeciesMetaAttribute : Attribute
|
||||
{
|
||||
public SpeciesType AssociatedSpecies { get; private set; }
|
||||
public string NiceName { get; private set; }
|
||||
public string FieldName { get; private set; }
|
||||
public SubspeciesMetaAttribute(SpeciesType associatedType, string nicename, string fieldname = "Subspecies")
|
||||
{
|
||||
AssociatedSpecies = associatedType;
|
||||
NiceName = nicename;
|
||||
FieldName = fieldname;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,91 +0,0 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Reflection;
|
||||
using System.Text;
|
||||
using Humanizer;
|
||||
|
||||
namespace CharacterRecordsGenerator
|
||||
{
|
||||
public static class Utility
|
||||
{
|
||||
public static IList<string> LineSplit(this string source) =>
|
||||
source.Split('\n').Where(item => item.Trim().Length != 0).ToList();
|
||||
|
||||
public static string CmToFeet(double cm)
|
||||
{
|
||||
var feet = Math.Floor(cm * 0.0328084);
|
||||
var inches = Math.Floor(cm * 0.39370079); // Isn't imperial a lovely system?
|
||||
inches -= feet * 12;
|
||||
return $"{feet}'{inches}\"";
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a weight in Kilograms to Pounds.
|
||||
/// </summary>
|
||||
/// <param name="kg">The weight in kilograms.</param>
|
||||
/// <returns>The weight converted to pounds.</returns>
|
||||
public static double KgToLb(double kg) => Math.Round(kg * 2.2046, 2);
|
||||
|
||||
/// <summary>
|
||||
/// Returns <paramref name="val"/> and a trailing space if val is not whitespace, <see cref="string.Empty"/> otherwise.
|
||||
/// </summary>
|
||||
/// <param name="val"></param>
|
||||
/// <returns></returns>
|
||||
public static string SpaceIfValue(this string val) => string.IsNullOrWhiteSpace(val) ? string.Empty : $"{val} ";
|
||||
|
||||
public static string HumanisedDate(DateTime date) => $"{date.ToString("MMMM")} {date.Day.Ordinalize()}, {date.Year}";
|
||||
|
||||
public static string IfEmpty(this string target, string fallback) =>
|
||||
target.IsEmpty() ? fallback : target;
|
||||
|
||||
public static bool IsEmpty(this string val) => string.IsNullOrWhiteSpace(val);
|
||||
|
||||
public static string FormatAsList(this IEnumerable<string> target) =>
|
||||
target.Aggregate(new StringBuilder(), (b, s) => b.AppendLine($" - {s.Trim()}")).ToString();
|
||||
|
||||
public static string Repeat(this string target, int repeatNum)
|
||||
{
|
||||
var builder = new StringBuilder(target.Length * repeatNum);
|
||||
for (var i = 0; i < repeatNum; i++)
|
||||
builder.Append(target);
|
||||
|
||||
return builder.ToString();
|
||||
}
|
||||
|
||||
public static string SubspeciesNiceName(SpeciesSubType species)
|
||||
{
|
||||
var attr = species.GetAttributeOfType<SubspeciesMetaAttribute>();
|
||||
return attr?.NiceName ?? Enum.GetName(typeof(SpeciesSubType), species);
|
||||
}
|
||||
|
||||
public static SpeciesSubType SubspeciesNiceNameToEnum(string nicename)
|
||||
{
|
||||
return (from item in Enum.GetValues(typeof(SpeciesSubType)).Cast<SpeciesSubType>()
|
||||
let attr = item.GetAttributeOfType<SubspeciesMetaAttribute>()
|
||||
where attr != null && attr.NiceName == nicename
|
||||
select item).FirstOrDefault();
|
||||
}
|
||||
|
||||
public static Version GetVersion() => Assembly.GetExecutingAssembly().GetName().Version;
|
||||
}
|
||||
|
||||
// From https://stackoverflow.com/questions/1799370/getting-attributes-of-enums-value
|
||||
public static class EnumHelper
|
||||
{
|
||||
/// <summary>
|
||||
/// Gets an attribute on an enum field value
|
||||
/// </summary>
|
||||
/// <typeparam name="T">The type of the attribute you want to retrieve</typeparam>
|
||||
/// <param name="enumVal">The enum value</param>
|
||||
/// <returns>The attribute of type T that exists on the enum value</returns>
|
||||
/// <example>string desc = myEnumVariable.GetAttributeOfType<DescriptionAttribute>().Description;</example>
|
||||
public static T GetAttributeOfType<T>(this Enum enumVal) where T : Attribute
|
||||
{
|
||||
var type = enumVal.GetType();
|
||||
var memInfo = type.GetMember(enumVal.ToString());
|
||||
var attributes = memInfo[0].GetCustomAttributes(typeof(T), false);
|
||||
return attributes.Length > 0 ? (T)attributes[0] : null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -1,39 +0,0 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<packages>
|
||||
<package id="ControlzEx" version="3.0.2.4" targetFramework="net452" />
|
||||
<package id="Humanizer.Core" version="2.5.16" targetFramework="net452" />
|
||||
<package id="MahApps.Metro" version="1.6.5" targetFramework="net452" />
|
||||
<package id="Microsoft.NETCore.Platforms" version="2.2.0" targetFramework="net452" />
|
||||
<package id="NETStandard.Library" version="2.0.3" targetFramework="net452" />
|
||||
<package id="protobuf-net" version="2.4.0" targetFramework="net452" />
|
||||
<package id="System.Collections" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Collections.Concurrent" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Diagnostics.Debug" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Diagnostics.Tools" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Diagnostics.Tracing" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Globalization" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.IO" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.IO.Compression" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Linq" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Linq.Expressions" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Net.Http" version="4.3.4" targetFramework="net452" />
|
||||
<package id="System.Net.Primitives" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.ObjectModel" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Reflection" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Reflection.Extensions" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Reflection.Primitives" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Resources.ResourceManager" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Runtime" version="4.3.1" targetFramework="net452" />
|
||||
<package id="System.Runtime.Extensions" version="4.3.1" targetFramework="net452" />
|
||||
<package id="System.Runtime.InteropServices" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Runtime.InteropServices.RuntimeInformation" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Runtime.Numerics" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Text.Encoding" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Text.Encoding.Extensions" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Text.RegularExpressions" version="4.3.1" targetFramework="net452" />
|
||||
<package id="System.Threading" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Threading.Tasks" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Threading.Timer" version="4.3.0" targetFramework="net452" />
|
||||
<package id="System.Xml.ReaderWriter" version="4.3.1" targetFramework="net452" />
|
||||
<package id="System.Xml.XDocument" version="4.3.0" targetFramework="net452" />
|
||||
</packages>
|
||||
674
LICENSE
674
LICENSE
|
|
@ -1,674 +0,0 @@
|
|||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
<program> Copyright (C) <year> <name of author>
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<http://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<http://www.gnu.org/philosophy/why-not-lgpl.html>.
|
||||
45
README.md
45
README.md
|
|
@ -1,9 +1,44 @@
|
|||
# Character Records Generator (for Aurora)
|
||||
# Character Records Generator
|
||||
|
||||
[Find the latest release here](https://github.com/Aurorastation/character-records-generator/releases) - just download the character-records-generator.zip from the latest release and you're good to go.
|
||||
A web-based character records tool for [Aurora Station](https://aurorastation.org/). Hosted at [c.ily.rs](https://c.ily.rs).
|
||||
|
||||
A tool for making setting up employment, medical, and security records a little easier on Aurora, forked from the AuroraRecordGenerator originally created by Lohikar.
|
||||
Pick a template and fill in the form. Each section covers a different record. Blank fields are omitted from the output automatically, so no rush to finish everything.
|
||||
|
||||
This modification of the records generator has been made up to date with all of the latest changes to the server, and to better match the example records present on [Aurora's Guide to Character Records](https://wiki.aurorastation.org/index.php?title=Guide_to_Character_Records). Some redundant fields have been removed, and nearly all of the fields are now optional rather than being listed as 'not specified', hopefully to make filling out the records less of a chore if you want to do it piece by piece as you play.
|
||||
Characters save to your browser. You can also export to a file or generate a share link: the link itself encodes the full set of records, so functionally it's a save file.
|
||||
|
||||
Credit to Lohikar's original AuroraRecordGenerator (https://github.com/Lohikar/AuroraRecordGenerator)
|
||||
Share links let the recipient see a preview of your records, with the option to import the character into their own roster.
|
||||
|
||||
This tool is entirely data-driven in XML, and it's already set up for template sharing. A visual template editor is coming soon, so anybody can create their own templates and share them between one another.
|
||||
|
||||
For issues, your best chance of getting a reply is to make an issue here, or to ping @llywelwyn in Discord.
|
||||
|
||||
Cheers.
|
||||
|
||||
## Development
|
||||
|
||||
```sh
|
||||
npm install
|
||||
npm run dev
|
||||
```
|
||||
|
||||
Build for production:
|
||||
|
||||
```sh
|
||||
npm run build
|
||||
```
|
||||
|
||||
Validate the data files:
|
||||
|
||||
```sh
|
||||
npm run validate
|
||||
```
|
||||
|
||||
Run tests:
|
||||
|
||||
```sh
|
||||
npx vitest run
|
||||
```
|
||||
|
||||
## Where did the old WPF app go?
|
||||
|
||||
This used to be a WPF desktop app. The last version of that lives at [`03feee5`](https://github.com/Aurorastation/character-records-generator/tree/03feee572bc7085fd8f9c458490a5dcc642ce689).
|
||||
|
|
|
|||
82
data/citizenships.xml
Normal file
82
data/citizenships.xml
Normal file
|
|
@ -0,0 +1,82 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<citizenships>
|
||||
<!-- Human -->
|
||||
<citizenship id="biesel" name="Republic of Biesel">
|
||||
<description>An independent system within the core of human space, heavily tied to NanoTrasen and known for its large xeno population.</description>
|
||||
</citizenship>
|
||||
<citizenship id="sol" name="Sol Alliance">
|
||||
<description>Once the juggernaut of human space, the Sol Alliance has declined significantly. Now a military junta, it controls the largest human military and holds xenophobic attitudes.</description>
|
||||
</citizenship>
|
||||
<citizenship id="coalition" name="Coalition of Colonies">
|
||||
<description>A loose confederation of frontier communities, ranging from colonial systems to super ships, bound together by shared independence from the major powers.</description>
|
||||
</citizenship>
|
||||
<citizenship id="elyra" name="Republic of Elyra">
|
||||
<description>The Serene Republic of Elyra, formed from early colonisation efforts. Its mixed relations with NanoTrasen stem from its possession of phoron deposits.</description>
|
||||
</citizenship>
|
||||
<citizenship id="elyra-ncp" name="Elyran Non-Citizen Person">
|
||||
<description>NCPs make up roughly a third of Elyra's population. Predominantly poor migrant labourers, they are excluded from most public systems.</description>
|
||||
</citizenship>
|
||||
<citizenship id="eridani" name="Eridani Federation">
|
||||
<description>A dystopian oligarchic republic dominated by mega-corporations, with a stark class divide between Corporates and Dregs.</description>
|
||||
</citizenship>
|
||||
<citizenship id="dominia" name="Empire of Dominia">
|
||||
<description>A heavily religious absolute monarchy ruled by Emperor Boleslaw Keeser. Autocratic and socio-economically stratified, heavily influenced by the Moroz Holy Tribunal.</description>
|
||||
</citizenship>
|
||||
|
||||
<!-- Skrell -->
|
||||
<citizenship id="nralakk-federation" name="Nralakk Federation">
|
||||
<description>A centralised union of solar systems with its capital at Kal'lo on Qerrbalak. Widely regarded as offering the best quality of life in the galaxy, the Federation remains wary of humanity's development of AI technology.</description>
|
||||
</citizenship>
|
||||
|
||||
<!-- Unathi -->
|
||||
<citizenship id="izweski" name="Izweski Hegemony">
|
||||
<description>A feudal empire ruled by the Izweski Clan from Moghes. The clan system is deeply entrenched in Unathi society, and the Hegemony continues to recover from an apocalyptic nuclear war.</description>
|
||||
</citizenship>
|
||||
|
||||
<!-- Tajara -->
|
||||
<citizenship id="pra" name="People's Republic of Adhomai">
|
||||
<description>The Hadiist faction of Adhomai. A centralised state implementing land reform and collectivisation across Tajaran territory.</description>
|
||||
</citizenship>
|
||||
<citizenship id="dpra" name="Democratic People's Republic of Adhomai">
|
||||
<description>Led by Purrjar Almrah Harrlala, the DPRA was formerly a militant insurgency now transitioning into a democratic nation.</description>
|
||||
</citizenship>
|
||||
<citizenship id="nka" name="New Kingdom of Adhomai">
|
||||
<description>Ruled by the Njarir'Akhran noble line under King Vahzirthaamro Azunja. A constitutional monarchy seeking to restore traditional Tajaran governance.</description>
|
||||
</citizenship>
|
||||
<citizenship id="free-council" name="Free Tajaran Council">
|
||||
<description>The largest Tajaran community on Himeo, with origins in First Revolution-era revolutionaries. Founded on principles of radical equality.</description>
|
||||
</citizenship>
|
||||
|
||||
<!-- Vaurca -->
|
||||
<citizenship id="zora" name="Zo'ra Hive">
|
||||
<description>The largest and most powerful Vaurca hive, and the first discovered by humanity. The Zo'ra believe themselves to be the alpha of the Vaurca and the face of their species. They are the most politically developed hive.</description>
|
||||
</citizenship>
|
||||
<citizenship id="klax" name="K'lax Hive">
|
||||
<description>The second hive discovered by humanity. Formerly a client state of the Zo'ra, the K'lax are now a vassal of the Izweski Nation. They are the most technologically developed hive.</description>
|
||||
</citizenship>
|
||||
<citizenship id="cthur" name="C'thur Hive">
|
||||
<description>The third hive to develop relationships with other sophonts, and the only one still led by its original Hive Queen. Allied with the Nralakk Federation, the C'thur are the most economically developed hive.</description>
|
||||
</citizenship>
|
||||
|
||||
<!-- Diona -->
|
||||
<citizenship id="hieroaetheria" name="The Consortium of Hieroaetheria">
|
||||
<description>A loose confederation of Dionae groups across Mede. Progressive, multicultural, and inclusive of non-Dionae members.</description>
|
||||
</citizenship>
|
||||
<citizenship id="glaorr" name="The Union of Gla'orr">
|
||||
<description>An autocratic and xenophobic Dionae state opposed to non-Dionae integration. More secular than the Ekane.</description>
|
||||
</citizenship>
|
||||
<citizenship id="ekane" name="The Eternal Republic of The Ekane">
|
||||
<description>An autocratic Dionae theocracy founded after Nralakk contact. Opposed to reform and inclusion, with the Eternal schools central to daily life.</description>
|
||||
</citizenship>
|
||||
|
||||
<!-- Synthetic / Special -->
|
||||
<citizenship id="golden-deep" name="Golden Deep">
|
||||
<description>A collection of free synthetics united by the pursuit of power through currency and trade. Only recently revealed to humanity, their presence in Tau Ceti is limited to merchants and clerical industries.</description>
|
||||
</citizenship>
|
||||
<citizenship id="orepit" name="Ecclesiarchy of Orepit">
|
||||
<description>A theocratic regime under ARM-1DRIL, tracing its origins to the Church of the Trinary Perfection. A signatory to the Open Doors memorandum, allowing its populace to work and travel abroad.</description>
|
||||
</citizenship>
|
||||
<citizenship id="none" name="None">
|
||||
<description>Not a citizen of any state. A common situation for owned IPCs and Dionae.</description>
|
||||
</citizenship>
|
||||
</citizenships>
|
||||
20
data/languages.xml
Normal file
20
data/languages.xml
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<languages>
|
||||
<language id="tau-ceti-basic" name="Tau Ceti Basic" shortname="TCB"></language>
|
||||
<language id="eal" name="Encoded Audio Language" shortname="EAL"></language>
|
||||
<language id="sign" name="Sign Language"> </language>
|
||||
<language id="sol-common" name="Sol Common"> </language>
|
||||
<language id="elyran-standard" name="Elyran Standard"> </language>
|
||||
<language id="freespeak" name="Freespeak"> </language>
|
||||
<language id="tradeband" name="Tradeband"> </language>
|
||||
<language id="siikmaas" name="Siik'maas"> </language>
|
||||
<language id="siiktajr" name="Siik'tajr"> </language>
|
||||
<language id="yassa" name="Ya'ssa"> </language>
|
||||
<language id="delvahhi" name="Delvahhi"> </language>
|
||||
<language id="nalrasan" name="Nal'rasan"> </language>
|
||||
<language id="sintaunathi" name="Sinta'Unathi"> </language>
|
||||
<language id="sintaazaziba" name="Sinta'Azaziba"> </language>
|
||||
<language id="nralmalic" name="Nral'Malic"> </language>
|
||||
<language id="rootsong" name="Rootsong"> </language>
|
||||
<language id="hivenet" name="Hivenet"> </language>
|
||||
</languages>
|
||||
20
data/schema/citizenships.xsd
Normal file
20
data/schema/citizenships.xsd
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
|
||||
|
||||
<xs:element name="citizenships">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="citizenship" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="description" type="xs:string" minOccurs="0" />
|
||||
</xs:sequence>
|
||||
<xs:attribute name="id" type="xs:string" use="required" />
|
||||
<xs:attribute name="name" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
|
||||
</xs:schema>
|
||||
20
data/schema/languages.xsd
Normal file
20
data/schema/languages.xsd
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
|
||||
|
||||
<xs:element name="languages">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="language" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="description" type="xs:string" minOccurs="0" />
|
||||
</xs:sequence>
|
||||
<xs:attribute name="id" type="xs:string" use="required" />
|
||||
<xs:attribute name="name" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
|
||||
</xs:schema>
|
||||
42
data/schema/species.xsd
Normal file
42
data/schema/species.xsd
Normal file
|
|
@ -0,0 +1,42 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
|
||||
|
||||
<xs:element name="species">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="description" type="xs:string" minOccurs="0" />
|
||||
<xs:element name="languages" type="refList" />
|
||||
<xs:element name="citizenships" type="refList" />
|
||||
<xs:element name="subspecies" minOccurs="0">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="entry" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="description" type="xs:string" minOccurs="0" />
|
||||
</xs:sequence>
|
||||
<xs:attribute name="id" type="xs:string" use="required" />
|
||||
<xs:attribute name="name" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
<xs:attribute name="id" type="xs:string" use="required" />
|
||||
<xs:attribute name="name" type="xs:string" use="required" />
|
||||
<xs:attribute name="subspeciesLabel" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
|
||||
<xs:complexType name="refList">
|
||||
<xs:sequence>
|
||||
<xs:element name="ref" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:attribute name="id" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
</xs:complexType>
|
||||
|
||||
</xs:schema>
|
||||
65
data/schema/template.xsd
Normal file
65
data/schema/template.xsd
Normal file
|
|
@ -0,0 +1,65 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
|
||||
|
||||
<xs:element name="template">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="description" type="xs:string" />
|
||||
<xs:element name="record" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="preamble" type="xs:string" minOccurs="0" />
|
||||
<xs:element name="note" type="xs:string" minOccurs="0" />
|
||||
<xs:element name="field" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:sequence>
|
||||
<xs:element name="option" minOccurs="0" maxOccurs="unbounded">
|
||||
<xs:complexType>
|
||||
<xs:attribute name="value" type="xs:string" use="required" />
|
||||
<xs:attribute name="label" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
<xs:attribute name="label" type="xs:string" use="required" />
|
||||
<xs:attribute name="type" type="fieldType" use="required" />
|
||||
<xs:attribute name="required" type="xs:boolean" />
|
||||
<xs:attribute name="placeholder" type="xs:string" />
|
||||
<xs:attribute name="from" type="xs:string" />
|
||||
<xs:attribute name="min" type="xs:decimal" />
|
||||
<xs:attribute name="max" type="xs:decimal" />
|
||||
<xs:attribute name="unit" type="xs:string" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
<xs:attribute name="type" type="xs:string" use="required" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
</xs:sequence>
|
||||
<xs:attribute name="name" type="xs:string" use="required" />
|
||||
<xs:attribute name="schemaVersion" type="xs:positiveInteger" use="required" />
|
||||
<xs:attribute name="species" type="xs:string" />
|
||||
</xs:complexType>
|
||||
</xs:element>
|
||||
|
||||
<xs:simpleType name="fieldType">
|
||||
<xs:restriction base="xs:string">
|
||||
<xs:enumeration value="name" />
|
||||
<xs:enumeration value="text" />
|
||||
<xs:enumeration value="textarea" />
|
||||
<xs:enumeration value="list" />
|
||||
<xs:enumeration value="number" />
|
||||
<xs:enumeration value="select" />
|
||||
<xs:enumeration value="multi-select" />
|
||||
<xs:enumeration value="checkbox" />
|
||||
<xs:enumeration value="date" />
|
||||
<xs:enumeration value="height" />
|
||||
<xs:enumeration value="weight" />
|
||||
<xs:enumeration value="species" />
|
||||
<xs:enumeration value="subspecies" />
|
||||
<xs:enumeration value="citizenship" />
|
||||
<xs:enumeration value="languages" />
|
||||
<xs:enumeration value="separator" />
|
||||
</xs:restriction>
|
||||
</xs:simpleType>
|
||||
|
||||
</xs:schema>
|
||||
34
data/species/diona.xml
Normal file
34
data/species/diona.xml
Normal file
|
|
@ -0,0 +1,34 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="diona" name="Diona" subspeciesLabel="Subspecies">
|
||||
<description>A mysterious plant-like species hailing from the depths of space. Dionae are comprised of cat-sized caterpillar-like creatures called nymphs, which form gestalt consciousnesses when two or more combine. Almost every aspect of the species is a mystery: their origins, behaviour, and functions. They are capable of great intellectual and biological feats, surviving primarily off the electromagnetic spectrum and biological matter.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="rootsong" />
|
||||
<ref id="nralmalic" />
|
||||
<ref id="sintaunathi" />
|
||||
<ref id="sintaazaziba" />
|
||||
<ref id="sign" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="biesel" />
|
||||
<ref id="nralakk-federation" />
|
||||
<ref id="hieroaetheria" />
|
||||
<ref id="ekane" />
|
||||
<ref id="glaorr" />
|
||||
<ref id="coalition" />
|
||||
<ref id="none" />
|
||||
</citizenships>
|
||||
|
||||
<subspecies>
|
||||
<entry id="geras" name="Geras">
|
||||
<description>The older, more common form of Dionae gestalt. Their bark is fully developed, providing greater resistance to damage but making them slower. Geras gestalts can live for up to a thousand years, and their long lifespans grant them vast experience and knowledge.</description>
|
||||
</entry>
|
||||
<entry id="coeus" name="Coeus">
|
||||
<description>Younger Dionae gestalts whose bark is not as developed as that of the Geras. Generally considered a Coeus until about 80 years old, though some can be as old as 180. Due to having less bark, Coeus are faster but less resistant to damage. They tend to be more formative, idealistic, and pacifistic than their older counterparts.</description>
|
||||
</entry>
|
||||
</subspecies>
|
||||
</species>
|
||||
29
data/species/human.xml
Normal file
29
data/species/human.xml
Normal file
|
|
@ -0,0 +1,29 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="human" name="Human" subspeciesLabel="Variant">
|
||||
<description>Humans originated on Earth.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="sign" />
|
||||
<ref id="sol-common" />
|
||||
<ref id="elyran-standard" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="sol" />
|
||||
<ref id="coalition" />
|
||||
<ref id="biesel" />
|
||||
<ref id="dominia" />
|
||||
<ref id="elyra" />
|
||||
<ref id="elyra-ncp" />
|
||||
<ref id="eridani" />
|
||||
</citizenships>
|
||||
|
||||
<subspecies>
|
||||
<entry id="offworlder" name="Offworlder">
|
||||
<description>These humans negatively affected by low gravity are a common sight in remote regions of the Orion Spur colonized by humanity. The largest known concentration of offworlders is the Scarab Fleet.</description>
|
||||
</entry>
|
||||
</subspecies>
|
||||
</species>
|
||||
47
data/species/ipc.xml
Normal file
47
data/species/ipc.xml
Normal file
|
|
@ -0,0 +1,47 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="ipc" name="Integrated Positronic Chassis" subspeciesLabel="Frame">
|
||||
<description>IPCs are synthetic chassis housing a positronic processing core. From skeletal baselines to humanlike shells, they serve in a variety of roles across the Orion Spur. Their legal status varies wildly between nations, ranging from full citizenship to being classified as property.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="eal" />
|
||||
<ref id="sign" />
|
||||
<ref id="sol-common" />
|
||||
<ref id="elyran-standard" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="biesel" />
|
||||
<ref id="coalition" />
|
||||
<ref id="elyra" />
|
||||
<ref id="orepit" />
|
||||
<ref id="golden-deep" />
|
||||
<ref id="none" />
|
||||
</citizenships>
|
||||
|
||||
<subspecies>
|
||||
<entry id="baseline" name="Baseline">
|
||||
<description>The most common type of IPC. Baseline models cover anything that is not an Industrial chassis or a Shell, ranging from custom-made to assembly-line units. They are typically skeletal or semi-humanoid with simple atmospheric diffusion cooling systems.</description>
|
||||
</entry>
|
||||
<entry id="shell" name="Shell">
|
||||
<description>A controversial model equipped with a synthskin weave over its metal chassis, creating an uncannily close approximation of the organic form. Created in the late days of 2450, Shells are focused on service, civilian, and medical roles. The additional weight of the synthskin reduces the efficacy of their coolant systems and increases charge consumption.</description>
|
||||
</entry>
|
||||
<entry id="g1" name="Industrial G1">
|
||||
<description>The first commercialised industrial-type IPC by Hephaestus Industries. Designed for extra durability and increased weight loads, the G1 possesses a limited power cell and actuators designed for heavy lifting rather than locomotion, resulting in a slow and frequently charging machine. A built-in heat transferal system allows it to perform EVA without a voidsuit.</description>
|
||||
</entry>
|
||||
<entry id="g2" name="Hephaestus Industrial G2">
|
||||
<description>An improved Hephaestus Industries industrial model with thicker plating and an improved power cell. Its actuators struggle to carry the immense weight, making the unit quite slow. Increased plating makes cooling a challenge, and overtaxing its hardware leads to rapid overheating.</description>
|
||||
</entry>
|
||||
<entry id="xion" name="Xion Industrial">
|
||||
<description>Developed by the Xion Manufacturing Group, a subsidiary of Hephaestus Industries. Sturdy, strong, and powerful with an ample power cell and improved actuators. The Xion model retains sturdiness without excessive plating, allowing its cooling systems to vent heat more easily. This unit can perform EVA without assistance.</description>
|
||||
</entry>
|
||||
<entry id="zenghu" name="Zeng-Hu">
|
||||
<description>Produced by Zeng-Hu Pharmaceuticals for medical and science-related operations. Digitigrade legs provide enhanced speed, but most plates covering the interior electronics are polymer casts to reduce weight, resulting in a less durable chassis.</description>
|
||||
</entry>
|
||||
<entry id="bishop" name="Bishop Cybernetics">
|
||||
<description>A sleek, high-end design by Bishop Cybernetics focused on energy efficiency. Cutting-edge power management allows it to run more demanding processing algorithms than most frames. The shiny chrome and glass meant to showcase its technology leaves it exposed and fragile, often suffering from reliability issues.</description>
|
||||
</entry>
|
||||
</subspecies>
|
||||
</species>
|
||||
35
data/species/skrell.xml
Normal file
35
data/species/skrell.xml
Normal file
|
|
@ -0,0 +1,35 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="skrell" name="Skrell" subspeciesLabel="Subspecies">
|
||||
<description>Skrell are a species of amphibious bipeds, originating from the planet of Qerrbalak. With longer lifespans than most sophonts, combined with their earlier development of organized societies, skrell are the oldest spacefaring species in the Orion Spur.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="nralmalic" />
|
||||
<ref id="sign" />
|
||||
<ref id="sol-common" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="nralakk-federation" />
|
||||
<ref id="biesel" />
|
||||
<ref id="sol" />
|
||||
<ref id="coalition" />
|
||||
<ref id="eridani" />
|
||||
<ref id="hieroaetheria" />
|
||||
<ref id="izweski" />
|
||||
</citizenships>
|
||||
|
||||
<subspecies>
|
||||
<entry id="axiori" name="Axiori">
|
||||
<description>More visibly aquatic, having larger tails, eyes, and more prominently webbed feet in order to improve their ability to swim underwater. The average Axiori skrell is shorter, heavier, and usually has darker-colored skin coloration than the other two subspecies.</description>
|
||||
</entry>
|
||||
<entry id="xiialt" name="Xiialt">
|
||||
<description>Xiialt skrell, having adapted to live primarily on land, are usually taller and leaner than Axiori. They possess longer, more developed legs, but usually have much smaller tails, or lack them entirely.</description>
|
||||
</entry>
|
||||
<entry id="xiori" name="Xiori">
|
||||
<description>A hybrid between Xiialt and Axiori skrell, and possess multiple traits from either group. Axiori features maintain slight prominence in Xiori skrell.</description>
|
||||
</entry>
|
||||
</subspecies>
|
||||
</species>
|
||||
39
data/species/tajara.xml
Normal file
39
data/species/tajara.xml
Normal file
|
|
@ -0,0 +1,39 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="tajara" name="Tajara" subspeciesLabel="Ethnicity">
|
||||
<description>The Tajara are a race of humanoids that possess markedly felinoid traits. Tajaran history and society is deeply entrenched in the conflict between its caste system and ruling governments.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="siikmaas" />
|
||||
<ref id="siiktajr" />
|
||||
<ref id="yassa" />
|
||||
<ref id="nalrasan" />
|
||||
<ref id="delvahhi" />
|
||||
<ref id="sign" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="pra" />
|
||||
<ref id="dpra" />
|
||||
<ref id="nka" />
|
||||
<ref id="biesel" />
|
||||
<ref id="free-council" />
|
||||
</citizenships>
|
||||
|
||||
<subspecies>
|
||||
<entry id="hharar" name="Hharar">
|
||||
<description>The first Tajaran ethnicity that Humanity came in contact with is generally viewed as the 'typical Tajara', which is reinforced by their numerical superiority over the other groups.</description>
|
||||
</entry>
|
||||
<entry id="zhan" name="Zhan-Khazan">
|
||||
<description>The second most populous of Tajaran ethnicities, and are considered to be the backbone of the Tajaran workforce.</description>
|
||||
</entry>
|
||||
<entry id="msai" name="M'sai">
|
||||
<description>The third most populous Tajaran ethnic group, the M'sai were at one point the hunters for ancient Tajara and evolved to have lithe, slender forms, and light fur that hid them in the blizzards on Adhomai.</description>
|
||||
</entry>
|
||||
<entry id="njarir" name="Njarir'Akhran">
|
||||
<description>The ethnic group that made up the majority of the plutocracy before the Great War. Their lineage can be traced from careful breeding between Hharar and M'sai, leading to where they are today.</description>
|
||||
</entry>
|
||||
</subspecies>
|
||||
</species>
|
||||
20
data/species/unathi.xml
Normal file
20
data/species/unathi.xml
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="unathi" name="Unathi" subspeciesLabel="Subspecies">
|
||||
<description>A heavily reptilian species, Unathi hail from the Uuosa-Eso system. A relatively recent addition to the galactic stage, they suffered immense turmoil after the cultural and economic disruption following first contact with humanity. With their homeworld of Moghes suffering catastrophic climate change from a nuclear war in the recent past, the Izweski Hegemony that rules the majority of the species struggles to find its place in the galaxy. They hold ideals of honesty, virtue, martial combat, and spirituality above all else.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="sintaunathi" />
|
||||
<ref id="sintaazaziba" />
|
||||
<ref id="sign" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="izweski" />
|
||||
<ref id="biesel" />
|
||||
<ref id="coalition" />
|
||||
<ref id="dominia" />
|
||||
</citizenships>
|
||||
</species>
|
||||
39
data/species/vaurca.xml
Normal file
39
data/species/vaurca.xml
Normal file
|
|
@ -0,0 +1,39 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<species id="vaurca" name="Vaurca" subspeciesLabel="Type">
|
||||
<description>The Vaurca are an insectoid species with a complex caste-based society organised into Hives, each led by a Queen. They communicate primarily through a localised expression of their hivemind. Having arrived in the Orion Spur from deep space, they are one of the newest species to join the galactic stage, and their integration into existing societies remains an ongoing process.</description>
|
||||
|
||||
<languages>
|
||||
<ref id="tau-ceti-basic" />
|
||||
<ref id="hivenet" />
|
||||
<ref id="nralmalic" />
|
||||
<ref id="sintaunathi" />
|
||||
<ref id="sign" />
|
||||
<ref id="tradeband" />
|
||||
<ref id="freespeak" />
|
||||
</languages>
|
||||
|
||||
<citizenships>
|
||||
<ref id="zora" />
|
||||
<ref id="klax" />
|
||||
<ref id="cthur" />
|
||||
<ref id="biesel" />
|
||||
</citizenships>
|
||||
|
||||
<subspecies>
|
||||
<entry id="worker" name="Worker (Type A)">
|
||||
<description>The most common type of Vaurca and the backbone of their societies. Their hardened exoskeleton is approximately half an inch thick, providing protection against harsh radiation and acting as a pressure suit. Workers are bipedal, resistant to brute force, and suited for extended EVA, but slower and less agile than warriors. They are comfortable in any department except security.</description>
|
||||
</entry>
|
||||
<entry id="warrior" name="Warrior (Type BA)">
|
||||
<description>The second most prominent type in Vaurca society, serving as hive security and military. Unlike other types, warriors are not typically passive, making them more suitable for combat-oriented positions. They are most commonly found in security roles but can rarely appear in other departments.</description>
|
||||
</entry>
|
||||
<entry id="attendant" name="Attendant (Type BB)">
|
||||
<description>Digitigrade bipeds built to be agile and quick. Primarily scouts or support personnel, they excel at guerrilla tactics and are commonly attributed to the role of combat medics, providing medical assistance on the field.</description>
|
||||
</entry>
|
||||
<entry id="breeder" name="Breeder (Type CB)">
|
||||
<description>The leaders of Vaurca society. As the only fertile caste, they provide life to the Hive. Some Breeders have recently been used as representatives for their respective queens due to their keen social intelligence. Vaurca Breeders can only be played as Hive Representatives.</description>
|
||||
</entry>
|
||||
<entry id="bulwark" name="Bulwark (Type E)">
|
||||
<description>A bodyform derived from the worker caste in a collaboration between C'thur and Nralakk scientists. Much larger with significantly thicker carapaces than most Vaurca, making them slow but highly resistant to damage, including complete immunity to radiation. Typically used for heavy lifting, agricultural, and industrial work.</description>
|
||||
</entry>
|
||||
</subspecies>
|
||||
</species>
|
||||
58
data/templates/ipc.xml
Normal file
58
data/templates/ipc.xml
Normal file
|
|
@ -0,0 +1,58 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<template name="IPC" schemaVersion="1" species="ipc">
|
||||
<description>The record format recommended on the IPC wiki page for synthetics.</description>
|
||||
|
||||
<record type="public">
|
||||
<note>Basic identification information visible on all records.</note>
|
||||
<field label="Designation" type="name" />
|
||||
<field label="IPC Model" type="subspecies" />
|
||||
<field label="Pronouns" type="multi-select">
|
||||
<option value="he/him" label="he/him" />
|
||||
<option value="she/her" label="she/her" />
|
||||
<option value="they/them" label="they/them" />
|
||||
<option value="any/all" label="any/all" />
|
||||
<option value="it/its" label="it/its" />
|
||||
</field>
|
||||
<field label="Citizenship" type="citizenship" />
|
||||
<field label="Spoken Languages" type="languages" />
|
||||
<field label="Separator" type="separator" label="Chassis" />
|
||||
<field label="Positronic Manufacture Date" type="date" placeholder="24-03-2460" />
|
||||
<field label="Chassis Manufacture Date" type="date" placeholder="24-03-2460" />
|
||||
<field label="Tag" type="text" />
|
||||
<field label="Height" type="height" />
|
||||
<field label="Weight" type="weight" />
|
||||
<field label="Distinguishing Features" type="textarea" />
|
||||
<field label="Separator" type="separator" label="Ownership" />
|
||||
<field label="Ownership Status" type="select">
|
||||
<option value="company-owned" label="Company Owned" />
|
||||
<option value="owned" label="Private Owner" />
|
||||
<option value="free" label="Free" />
|
||||
</field>
|
||||
<field label="Owner Name" type="text" />
|
||||
<field label="Owner Contact Information" type="text" />
|
||||
</record>
|
||||
|
||||
<record type="employment">
|
||||
<preamble>This information has been verified by employment agents within the External Affairs department, and any comments, questions, or concerns about the legitimacy of such must be sent in a secure document to the same department.</preamble>
|
||||
<field label="Manufactured By" type="textarea" />
|
||||
<field label="Employment History" type="list" />
|
||||
<field label="Qualifications Summary" type="list" />
|
||||
<field label="Database Certifications" type="list" />
|
||||
</record>
|
||||
|
||||
<record type="medical">
|
||||
<preamble>The following information is protected by doctor-patient confidentiality laws. Do not release without patient's consent.</preamble>
|
||||
<field label="Chassis Repair History" type="list" />
|
||||
<field label="Positronic Repair History" type="list" />
|
||||
<field label="Software Diagnostics History" type="list" />
|
||||
<field label="Roboticist Notes" type="list" />
|
||||
</record>
|
||||
|
||||
<record type="security">
|
||||
<preamble>This information has been verified by employment agents within the External Affairs department, and any comments, questions, or concerns about the legitimacy of such must be sent in a secure document to the same department.</preamble>
|
||||
<field label="Attitude Towards SCC" type="textarea" />
|
||||
<field label="Attitude Towards Crew" type="textarea" />
|
||||
<field label="Arrest History" type="list" />
|
||||
<field label="Security Notes" type="list" />
|
||||
</record>
|
||||
</template>
|
||||
62
data/templates/standard.xml
Normal file
62
data/templates/standard.xml
Normal file
|
|
@ -0,0 +1,62 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<template name="General" schemaVersion="1">
|
||||
<description>The standard record format used by most players.</description>
|
||||
|
||||
<record type="public">
|
||||
<note>Basic identification information visible on all records.</note>
|
||||
<field label="Name" type="name" />
|
||||
<field label="Species" type="species" />
|
||||
<field label="Subspecies" type="subspecies" />
|
||||
<field label="Pronouns" type="multi-select">
|
||||
<option value="he/him" label="he/him" />
|
||||
<option value="she/her" label="she/her" />
|
||||
<option value="they/them" label="they/them" />
|
||||
<option value="any/all" label="any/all" />
|
||||
<option value="it/its" label="it/its" />
|
||||
</field>
|
||||
<field label="Date of Birth" type="date" placeholder="24-03-2450" />
|
||||
<field label="Citizenship" type="citizenship" />
|
||||
<field label="Spoken Languages" type="languages" />
|
||||
<field label="Employed As" type="text" />
|
||||
<field label="Next of Kin" type="text" />
|
||||
<field label="Separator" type="separator" label="Appearance" />
|
||||
<field label="Height" type="height" />
|
||||
<field label="Weight" type="weight" />
|
||||
<field label="Skin Color" type="text" />
|
||||
<field label="Hair Color" type="text" />
|
||||
<field label="Eye Color" type="text" />
|
||||
<field label="Distinguishing Features" type="textarea" />
|
||||
</record>
|
||||
|
||||
<record type="employment">
|
||||
<preamble>This information has been verified by employment agents within the External Affairs department, and any comments, questions, or concerns about the legitimacy of such must be sent in a secure document to the same department.</preamble>
|
||||
<field label="Employment History" type="list" />
|
||||
<field label="Formal Education" type="list" />
|
||||
<field label="Other Skills" type="list" />
|
||||
</record>
|
||||
|
||||
<record type="medical">
|
||||
<preamble>The following information is protected by doctor-patient confidentiality laws. Do not release without patient's consent.</preamble>
|
||||
<field label="Opt-Outs" type="multi-select">
|
||||
<option value="no-borg" label="Do NOT BORGIFY" />
|
||||
<option value="no-revive" label="DO NOT REVIVE" />
|
||||
<option value="no-prosthetic" label="DO NOT GIVE PROSTHETICS" />
|
||||
</field>
|
||||
<field label="Postmortem Instructions" type="textarea" />
|
||||
<field label="Allergies" type="list" />
|
||||
<field label="Current Prescriptions" type="list" />
|
||||
<field label="Medication History" type="list" />
|
||||
<field label="Surgical History" type="list" />
|
||||
<field label="Physical Evaluations" type="list" />
|
||||
<field label="Psychological Disorders" type="list" />
|
||||
<field label="Psychological Evaluations" type="list" />
|
||||
</record>
|
||||
|
||||
<record type="security">
|
||||
<preamble>This information has been verified by employment agents within the External Affairs department, and any comments, questions, or concerns about the legitimacy of such must be sent in a secure document to the same department.</preamble>
|
||||
<field label="Attitude Towards SCC" type="textarea" />
|
||||
<field label="Attitude Towards Crew" type="textarea" />
|
||||
<field label="Arrest History" type="list" />
|
||||
<field label="Security Notes" type="list" />
|
||||
</record>
|
||||
</template>
|
||||
3184
package-lock.json
generated
Normal file
3184
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
37
package.json
Normal file
37
package.json
Normal file
|
|
@ -0,0 +1,37 @@
|
|||
{
|
||||
"name": "character-records-generator",
|
||||
"private": true,
|
||||
"version": "0.0.1",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "vite dev",
|
||||
"build": "vite build",
|
||||
"preview": "vite preview",
|
||||
"prepare": "svelte-kit sync || echo ''",
|
||||
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
|
||||
"check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch",
|
||||
"validate": "scripts/validate-xml.sh"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@sveltejs/adapter-auto": "^7.0.0",
|
||||
"@sveltejs/adapter-static": "^3.0.10",
|
||||
"@sveltejs/kit": "^2.50.2",
|
||||
"@sveltejs/vite-plugin-svelte": "^6.2.4",
|
||||
"@tailwindcss/vite": "^4.2.2",
|
||||
"@types/pako": "^2.0.4",
|
||||
"jsdom": "^29.0.1",
|
||||
"svelte": "^5.51.0",
|
||||
"svelte-check": "^4.4.2",
|
||||
"tailwindcss": "^4.2.2",
|
||||
"typescript": "^5.9.3",
|
||||
"vite": "^7.3.1",
|
||||
"vitest": "^4.1.1"
|
||||
},
|
||||
"dependencies": {
|
||||
"fast-xml-parser": "^5.5.8",
|
||||
"idb": "^8.0.3",
|
||||
"lucide-svelte": "^1.0.1",
|
||||
"pako": "^2.1.0",
|
||||
"zod": "^4.3.6"
|
||||
}
|
||||
}
|
||||
27
scripts/validate-xml.sh
Executable file
27
scripts/validate-xml.sh
Executable file
|
|
@ -0,0 +1,27 @@
|
|||
#!/usr/bin/env bash
|
||||
set -euo pipefail
|
||||
|
||||
schema_dir="data/schema"
|
||||
fail=0
|
||||
|
||||
validate() {
|
||||
local schema="$1" file="$2"
|
||||
if xmllint --noout --schema "$schema" "$file" 2>&1; then
|
||||
:
|
||||
else
|
||||
fail=1
|
||||
fi
|
||||
}
|
||||
|
||||
for f in data/species/*.xml; do
|
||||
validate "$schema_dir/species.xsd" "$f"
|
||||
done
|
||||
|
||||
validate "$schema_dir/citizenships.xsd" data/citizenships.xml
|
||||
validate "$schema_dir/languages.xsd" data/languages.xml
|
||||
|
||||
for f in data/templates/*.xml; do
|
||||
validate "$schema_dir/template.xsd" "$f"
|
||||
done
|
||||
|
||||
exit $fail
|
||||
45
src/app.css
Normal file
45
src/app.css
Normal file
|
|
@ -0,0 +1,45 @@
|
|||
@import 'tailwindcss';
|
||||
|
||||
@custom-variant dark (&:where(.dark, .dark *));
|
||||
|
||||
@theme {
|
||||
--radius: 2px;
|
||||
--radius-lg: 4px;
|
||||
--radius-full: 9999px;
|
||||
}
|
||||
|
||||
:root {
|
||||
--bg: #fafafa;
|
||||
--bg-card: #ffffff;
|
||||
--bg-input: #ffffff;
|
||||
--border: #d4d4d4;
|
||||
--text: #171717;
|
||||
--text-muted: #737373;
|
||||
--accent: #0066cc;
|
||||
--accent-hover: #0052a3;
|
||||
--error: #dc2626;
|
||||
}
|
||||
|
||||
.dark {
|
||||
--bg: #0a0a0a;
|
||||
--bg-card: #171717;
|
||||
--bg-input: #0a0a0a;
|
||||
--border: #2e2e2e;
|
||||
--text: #e5e5e5;
|
||||
--text-muted: #a3a3a3;
|
||||
--accent: #4da6ff;
|
||||
--accent-hover: #80bfff;
|
||||
--error: #f87171;
|
||||
}
|
||||
|
||||
body {
|
||||
background-color: var(--bg);
|
||||
color: var(--text);
|
||||
font-family: system-ui, -apple-system, sans-serif;
|
||||
}
|
||||
|
||||
.field-error input,
|
||||
.field-error select,
|
||||
.field-error textarea {
|
||||
border-color: var(--error) !important;
|
||||
}
|
||||
13
src/app.d.ts
vendored
Normal file
13
src/app.d.ts
vendored
Normal file
|
|
@ -0,0 +1,13 @@
|
|||
// See https://svelte.dev/docs/kit/types#app.d.ts
|
||||
// for information about these interfaces
|
||||
declare global {
|
||||
namespace App {
|
||||
// interface Error {}
|
||||
// interface Locals {}
|
||||
// interface PageData {}
|
||||
// interface PageState {}
|
||||
// interface Platform {}
|
||||
}
|
||||
}
|
||||
|
||||
export {};
|
||||
12
src/app.html
Normal file
12
src/app.html
Normal file
|
|
@ -0,0 +1,12 @@
|
|||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
<title>Character Records</title>
|
||||
%sveltekit.head%
|
||||
</head>
|
||||
<body data-sveltekit-preload-data="hover">
|
||||
<div style="display: contents">%sveltekit.body%</div>
|
||||
</body>
|
||||
</html>
|
||||
27
src/lib/components/CharacterSwitcher.svelte
Normal file
27
src/lib/components/CharacterSwitcher.svelte
Normal file
|
|
@ -0,0 +1,27 @@
|
|||
<script lang="ts">
|
||||
import { roster } from '$lib/state.svelte';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
|
||||
function displayName(char: { template: { records: { fields: { type: string; label: string }[] }[] }; data: Record<string, unknown> }): string {
|
||||
const nameField = char.template.records.flatMap((r) => r.fields).find((f) => f.type === 'name');
|
||||
const key = nameField ? slugify(nameField.label) : slugify('Name');
|
||||
const name = char.data[key];
|
||||
return (name as string) || 'Unnamed Character';
|
||||
}
|
||||
</script>
|
||||
|
||||
<div class="relative inline-block">
|
||||
<select
|
||||
value={roster.active?.id ?? ''}
|
||||
onchange={(e) => roster.setActive((e.target as HTMLSelectElement).value)}
|
||||
class="rounded pl-3 pr-7 h-[30px] text-sm appearance-none cursor-pointer hover:opacity-80"
|
||||
style="background: var(--bg-card); border: 1px solid var(--border); color: var(--text);"
|
||||
>
|
||||
{#each roster.characters as char}
|
||||
<option value={char.id}>{displayName(char)}</option>
|
||||
{/each}
|
||||
</select>
|
||||
<svg class="absolute right-2 top-1/2 -translate-y-1/2 pointer-events-none" width="12" height="12" viewBox="0 0 12 12" style="color: var(--text-muted);">
|
||||
<path d="M3 5l3 3 3-3" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</svg>
|
||||
</div>
|
||||
171
src/lib/components/Header.svelte
Normal file
171
src/lib/components/Header.svelte
Normal file
|
|
@ -0,0 +1,171 @@
|
|||
<script lang="ts">
|
||||
import { Sun, Moon, Trash2, Plus, Upload, CircleHelp } from 'lucide-svelte';
|
||||
import { theme } from '$lib/theme.svelte';
|
||||
import { roster } from '$lib/state.svelte';
|
||||
import { presets } from '$lib/presets';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
import CharacterSwitcher from './CharacterSwitcher.svelte';
|
||||
import TemplatePicker from './TemplatePicker.svelte';
|
||||
import ShareMenu from './ShareMenu.svelte';
|
||||
import Modal from './Modal.svelte';
|
||||
import HelpText from './HelpText.svelte';
|
||||
|
||||
let { onImport }: { onImport?: (json: string) => void } = $props();
|
||||
|
||||
let confirmDelete = $state(false);
|
||||
let showPicker = $state(false);
|
||||
let showHelp = $state(false);
|
||||
let openDropdown = $state<'add' | 'share' | null>(null);
|
||||
let fileInput: HTMLInputElement;
|
||||
|
||||
function toggleDropdown(which: 'add' | 'share') {
|
||||
openDropdown = openDropdown === which ? null : which;
|
||||
}
|
||||
|
||||
function createCharacter() {
|
||||
if (presets.length === 1) {
|
||||
roster.create(presets[0]);
|
||||
} else {
|
||||
showPicker = true;
|
||||
}
|
||||
openDropdown = null;
|
||||
}
|
||||
|
||||
function triggerImport() {
|
||||
fileInput.click();
|
||||
openDropdown = null;
|
||||
}
|
||||
|
||||
async function handleFile(e: Event) {
|
||||
const input = e.target as HTMLInputElement;
|
||||
const file = input.files?.[0];
|
||||
if (!file) return;
|
||||
const text = await file.text();
|
||||
onImport?.(text);
|
||||
input.value = '';
|
||||
}
|
||||
|
||||
function displayName(): string {
|
||||
const char = roster.active;
|
||||
if (!char) return '';
|
||||
const nameField = char.template.records.flatMap((r) => r.fields).find((f) => f.type === 'name');
|
||||
const key = nameField ? slugify(nameField.label) : slugify('Name');
|
||||
const name = char.data[key];
|
||||
return (name as string) || 'Unnamed Character';
|
||||
}
|
||||
|
||||
async function doDelete() {
|
||||
if (roster.active) {
|
||||
await roster.remove(roster.active.id);
|
||||
}
|
||||
confirmDelete = false;
|
||||
}
|
||||
</script>
|
||||
|
||||
<header class="border-b shrink-0" style="border-color: var(--border); background: var(--bg-card);">
|
||||
<div class="flex items-center gap-2 px-4 py-3 max-w-7xl mx-auto w-full">
|
||||
<h1 class="font-bold whitespace-nowrap">Character Records</h1>
|
||||
|
||||
{#if roster.characters.length > 0}
|
||||
<CharacterSwitcher />
|
||||
{/if}
|
||||
|
||||
<span class="relative">
|
||||
<button
|
||||
onclick={(e) => { e.stopPropagation(); toggleDropdown('add'); }}
|
||||
class="flex items-center justify-center w-[30px] h-[30px] rounded border hover:opacity-80"
|
||||
style="border-color: var(--border);"
|
||||
title="Add character"
|
||||
>
|
||||
<Plus size={14} />
|
||||
</button>
|
||||
|
||||
{#if openDropdown === 'add'}
|
||||
<nav class="absolute left-0 z-10 mt-1 w-48 rounded border shadow-lg" style="background: var(--bg-card); border-color: var(--border);">
|
||||
<button
|
||||
onclick={createCharacter}
|
||||
class="flex items-center gap-2 w-full text-left px-3 py-2 text-sm hover:opacity-80"
|
||||
>
|
||||
<Plus size={14} /> New character
|
||||
</button>
|
||||
<button
|
||||
onclick={triggerImport}
|
||||
class="flex items-center gap-2 w-full text-left px-3 py-2 text-sm hover:opacity-80"
|
||||
>
|
||||
<Upload size={14} /> Import from file
|
||||
</button>
|
||||
</nav>
|
||||
{/if}
|
||||
</span>
|
||||
|
||||
{#if roster.active}
|
||||
<button onclick={() => { confirmDelete = true; }} class="flex items-center justify-center w-[30px] h-[30px] rounded border hover:opacity-80" style="border-color: var(--border);" title="Delete character">
|
||||
<Trash2 size={14} />
|
||||
</button>
|
||||
|
||||
<ShareMenu open={openDropdown === 'share'} onToggle={() => toggleDropdown('share')} />
|
||||
{/if}
|
||||
|
||||
<div class="ml-auto flex items-center gap-2">
|
||||
<span class="text-sm" style="color: var(--text-muted);">
|
||||
{#if roster.saveStatus === 'saving'}Saving...{:else if roster.saveStatus === 'saved'}Saved{/if}
|
||||
</span>
|
||||
|
||||
<button onclick={() => { showHelp = true; }} class="flex items-center justify-center w-[30px] h-[30px] rounded hover:opacity-80" title="About">
|
||||
<CircleHelp size={18} />
|
||||
</button>
|
||||
|
||||
<button onclick={() => theme.toggle()} class="flex items-center justify-center w-[30px] h-[30px] rounded hover:opacity-80" title="Toggle theme">
|
||||
{#if theme.dark}
|
||||
<Sun size={18} />
|
||||
{:else}
|
||||
<Moon size={18} />
|
||||
{/if}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
<input
|
||||
bind:this={fileInput}
|
||||
type="file"
|
||||
accept=".json"
|
||||
class="hidden"
|
||||
onchange={handleFile}
|
||||
/>
|
||||
|
||||
{#if showPicker}
|
||||
<TemplatePicker onClose={() => { showPicker = false; }} />
|
||||
{/if}
|
||||
|
||||
{#if confirmDelete && roster.active}
|
||||
<Modal onClose={() => { confirmDelete = false; }}>
|
||||
<h2 class="font-semibold mb-2">Delete Character</h2>
|
||||
<p class="text-sm mb-4">Delete <strong>{displayName()}</strong>? This can't be undone.</p>
|
||||
<div class="flex gap-2 justify-end">
|
||||
<button onclick={() => { confirmDelete = false; }} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border);">
|
||||
Cancel
|
||||
</button>
|
||||
<button onclick={doDelete} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border); color: #dc2626;">
|
||||
Delete
|
||||
</button>
|
||||
</div>
|
||||
</Modal>
|
||||
{/if}
|
||||
|
||||
{#if showHelp}
|
||||
<Modal onClose={() => { showHelp = false; }}>
|
||||
<HelpText />
|
||||
<div class="flex justify-end mt-4">
|
||||
<button
|
||||
onclick={() => { showHelp = false; }}
|
||||
class="px-3 py-1 rounded text-sm border hover:opacity-80"
|
||||
style="border-color: var(--border); color: var(--text);"
|
||||
>
|
||||
Got it
|
||||
</button>
|
||||
</div>
|
||||
</Modal>
|
||||
{/if}
|
||||
|
||||
<svelte:window onclick={() => { openDropdown = null; }} />
|
||||
17
src/lib/components/HelpText.svelte
Normal file
17
src/lib/components/HelpText.svelte
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
<div class="text-sm flex flex-col gap-4" style="color: var(--text-muted);">
|
||||
<p>
|
||||
Pick a template and fill in the form. Each section covers a different record. Blank fields are omitted from the output automatically, so no rush to finish everything.
|
||||
</p>
|
||||
<p>
|
||||
Characters save to your browser. You can also export to a file or generate a share link: the link itself encodes the full set of records, so functionally it's a save file.
|
||||
</p>
|
||||
<p>
|
||||
Share links let the recipient see a preview of your records, with the option to import the character into their own roster.
|
||||
</p>
|
||||
<p>
|
||||
This tool is entirely data-driven in XML, and it's already set up for template sharing. A visual template editor is coming soon, so anybody can create their own templates and share them between one another.
|
||||
</p>
|
||||
<p>
|
||||
Cheers.
|
||||
</p>
|
||||
</div>
|
||||
140
src/lib/components/ImportModal.svelte
Normal file
140
src/lib/components/ImportModal.svelte
Normal file
|
|
@ -0,0 +1,140 @@
|
|||
<script lang="ts">
|
||||
import { decodeCharacterURL, decodeTemplateURL } from '$lib/sharing';
|
||||
import { generateRecord } from '$lib/output';
|
||||
import { species } from '$lib/data';
|
||||
import { roster } from '$lib/state.svelte';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
import OutputTab from './OutputTab.svelte';
|
||||
|
||||
let {
|
||||
encoded = '',
|
||||
fileData = null,
|
||||
onClose
|
||||
}: {
|
||||
encoded?: string;
|
||||
fileData?: { template: any; data: Record<string, unknown> } | null;
|
||||
onClose: () => void;
|
||||
} = $props();
|
||||
|
||||
let error = $state('');
|
||||
let type = $state<'character' | 'template' | null>(null);
|
||||
let charData = $state<{ template: any; data: Record<string, unknown> } | null>(null);
|
||||
let tmplData = $state<any>(null);
|
||||
|
||||
let activeTab = $state('');
|
||||
|
||||
$effect(() => {
|
||||
if (fileData) {
|
||||
type = 'character';
|
||||
charData = fileData;
|
||||
error = '';
|
||||
} else if (encoded) {
|
||||
try {
|
||||
if (encoded.startsWith('c1.')) {
|
||||
type = 'character';
|
||||
charData = decodeCharacterURL(encoded);
|
||||
} else if (encoded.startsWith('t1.')) {
|
||||
type = 'template';
|
||||
tmplData = decodeTemplateURL(encoded);
|
||||
} else {
|
||||
error = 'Unrecognized share link format.';
|
||||
}
|
||||
} catch {
|
||||
error = 'Failed to decode share link.';
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
let tabs = $derived(
|
||||
charData?.template.records.filter((r: any) => r.type !== 'public') ?? []
|
||||
);
|
||||
|
||||
$effect(() => {
|
||||
if (tabs.length && !tabs.some((t: any) => t.type === activeTab)) {
|
||||
activeTab = tabs[0].type;
|
||||
}
|
||||
});
|
||||
|
||||
let output = $derived(
|
||||
charData && activeTab
|
||||
? generateRecord(charData.template, charData.data, activeTab, species)
|
||||
: ''
|
||||
);
|
||||
|
||||
function charName(): string {
|
||||
if (!charData) return 'Unknown';
|
||||
const nameField = charData.template.records.flatMap((r: any) => r.fields).find((f: any) => f.type === 'name');
|
||||
const key = nameField ? slugify(nameField.label) : slugify('Name');
|
||||
return (charData.data[key] as string) || 'Unnamed Character';
|
||||
}
|
||||
|
||||
async function importCharacter() {
|
||||
if (!charData) return;
|
||||
await roster.create(charData.template, charData.data);
|
||||
onClose();
|
||||
}
|
||||
|
||||
async function importTemplate() {
|
||||
if (!tmplData) return;
|
||||
await roster.create(tmplData);
|
||||
onClose();
|
||||
}
|
||||
</script>
|
||||
|
||||
<div class="min-h-screen flex flex-col items-center p-4">
|
||||
<div class="w-full max-w-2xl">
|
||||
{#if error}
|
||||
<div class="rounded border p-6 text-center" style="border-color: var(--border); background: var(--bg-card);">
|
||||
<p class="mb-4">{error}</p>
|
||||
<button onclick={onClose} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border);">
|
||||
Dismiss
|
||||
</button>
|
||||
</div>
|
||||
{:else if type === 'character' && charData}
|
||||
<div class="flex items-center justify-between mb-4">
|
||||
<div>
|
||||
<h2 class="font-semibold">{charName()}</h2>
|
||||
<p class="text-sm" style="color: var(--text-muted);">Shared character — {charData.template.name} template</p>
|
||||
</div>
|
||||
<div class="flex gap-2">
|
||||
<button onclick={onClose} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border);">
|
||||
Cancel
|
||||
</button>
|
||||
<button onclick={importCharacter} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border);">
|
||||
Import
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded border" style="border-color: var(--border); background: var(--bg-card);">
|
||||
<div class="flex border-b" style="border-color: var(--border);">
|
||||
{#each tabs as tab}
|
||||
<button
|
||||
onclick={() => { activeTab = tab.type; }}
|
||||
class="px-4 py-2 text-sm capitalize"
|
||||
style={activeTab === tab.type
|
||||
? `color: var(--accent); border-bottom: 2px solid var(--accent);`
|
||||
: `color: var(--text-muted); border-bottom: 2px solid transparent;`}
|
||||
>
|
||||
{tab.type}
|
||||
</button>
|
||||
{/each}
|
||||
</div>
|
||||
<OutputTab {output} />
|
||||
</div>
|
||||
{:else if type === 'template' && tmplData}
|
||||
<div class="rounded border p-6 text-center" style="border-color: var(--border); background: var(--bg-card);">
|
||||
<h2 class="font-semibold mb-2">Shared Template: {tmplData.name}</h2>
|
||||
<p class="text-sm mb-4" style="color: var(--text-muted);">{tmplData.records.length} records, {tmplData.records.reduce((n: number, r: any) => n + r.fields.length, 0)} fields</p>
|
||||
<div class="flex gap-2 justify-center">
|
||||
<button onclick={onClose} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border);">
|
||||
Cancel
|
||||
</button>
|
||||
<button onclick={importTemplate} class="px-3 py-1 rounded text-sm border hover:opacity-80" style="border-color: var(--border);">
|
||||
Create Character
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
</div>
|
||||
27
src/lib/components/Modal.svelte
Normal file
27
src/lib/components/Modal.svelte
Normal file
|
|
@ -0,0 +1,27 @@
|
|||
<script lang="ts">
|
||||
import type { Snippet } from 'svelte';
|
||||
|
||||
let { onClose, children }: { onClose: () => void; children: Snippet } = $props();
|
||||
|
||||
function handleKeydown(e: KeyboardEvent) {
|
||||
if (e.key === 'Escape') onClose();
|
||||
}
|
||||
</script>
|
||||
|
||||
<svelte:window onkeydown={handleKeydown} />
|
||||
|
||||
<!-- svelte-ignore a11y_no_static_element_interactions -->
|
||||
<div
|
||||
class="fixed inset-0 z-50 flex items-center justify-center"
|
||||
onmousedown={onClose}
|
||||
>
|
||||
<div class="absolute inset-0 bg-black/50"></div>
|
||||
<!-- svelte-ignore a11y_no_static_element_interactions -->
|
||||
<div
|
||||
class="relative rounded-lg shadow-xl max-w-md w-full mx-4 p-6"
|
||||
style="background: var(--bg-card); color: var(--text);"
|
||||
onmousedown={(e) => e.stopPropagation()}
|
||||
>
|
||||
{@render children()}
|
||||
</div>
|
||||
</div>
|
||||
42
src/lib/components/OutputPanel.svelte
Normal file
42
src/lib/components/OutputPanel.svelte
Normal file
|
|
@ -0,0 +1,42 @@
|
|||
<script lang="ts">
|
||||
import type { Character } from '$lib/types';
|
||||
import { generateRecord } from '$lib/output';
|
||||
import { species } from '$lib/data';
|
||||
import OutputTab from './OutputTab.svelte';
|
||||
|
||||
let { character }: { character: Character } = $props();
|
||||
|
||||
let tabs = $derived(
|
||||
character.template.records.filter((r) => r.type !== 'public')
|
||||
);
|
||||
|
||||
let activeTab = $state('');
|
||||
|
||||
$effect(() => {
|
||||
if (tabs.length && !tabs.some((t) => t.type === activeTab)) {
|
||||
activeTab = tabs[0].type;
|
||||
}
|
||||
});
|
||||
|
||||
let output = $derived(
|
||||
activeTab ? generateRecord(character.template, character.data, activeTab, species) : ''
|
||||
);
|
||||
</script>
|
||||
|
||||
<div class="flex flex-col h-full min-h-0 rounded border" style="border-color: var(--border); background: var(--bg-card);">
|
||||
<div class="flex border-b shrink-0" style="border-color: var(--border);">
|
||||
{#each tabs as tab}
|
||||
<button
|
||||
onclick={() => { activeTab = tab.type; }}
|
||||
class="px-4 py-2 text-sm capitalize"
|
||||
style={activeTab === tab.type
|
||||
? `color: var(--accent); border-bottom: 2px solid var(--accent);`
|
||||
: `color: var(--text-muted); border-bottom: 2px solid transparent;`}
|
||||
>
|
||||
{tab.type}
|
||||
</button>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<OutputTab {output} />
|
||||
</div>
|
||||
31
src/lib/components/OutputTab.svelte
Normal file
31
src/lib/components/OutputTab.svelte
Normal file
|
|
@ -0,0 +1,31 @@
|
|||
<script lang="ts">
|
||||
import { Copy, Check } from 'lucide-svelte';
|
||||
|
||||
let { output }: { output: string } = $props();
|
||||
|
||||
let copied = $state(false);
|
||||
|
||||
let wordCount = $derived(
|
||||
output.trim() ? output.trim().split(/\s+/).length : 0
|
||||
);
|
||||
|
||||
async function copy() {
|
||||
await navigator.clipboard.writeText(output);
|
||||
copied = true;
|
||||
setTimeout(() => { copied = false; }, 2000);
|
||||
}
|
||||
</script>
|
||||
|
||||
<div class="flex flex-col h-full min-h-0">
|
||||
<div class="flex items-center justify-between px-3 py-2 text-sm shrink-0" style="color: var(--text-muted);">
|
||||
<span>{wordCount} words</span>
|
||||
<button onclick={copy} class="flex items-center gap-1 px-2 py-1 rounded border hover:opacity-80" style="border-color: var(--border);">
|
||||
{#if copied}
|
||||
<Check size={14} /> Copied
|
||||
{:else}
|
||||
<Copy size={14} /> Copy
|
||||
{/if}
|
||||
</button>
|
||||
</div>
|
||||
<pre class="flex-1 overflow-auto px-4 py-3 text-sm whitespace-pre-wrap font-mono" style="background: var(--bg); color: var(--text);">{output}</pre>
|
||||
</div>
|
||||
88
src/lib/components/RecordCard.svelte
Normal file
88
src/lib/components/RecordCard.svelte
Normal file
|
|
@ -0,0 +1,88 @@
|
|||
<script lang="ts">
|
||||
import { slide } from 'svelte/transition';
|
||||
import { ChevronDown } from 'lucide-svelte';
|
||||
import type { FieldDef, RecordDef } from '$lib/types';
|
||||
import DynamicField from './fields/DynamicField.svelte';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
|
||||
let { record, data, onFieldChange }: {
|
||||
record: RecordDef;
|
||||
data: Record<string, unknown>;
|
||||
onFieldChange: (key: string, value: any) => void;
|
||||
} = $props();
|
||||
|
||||
let expanded = $state(false);
|
||||
let touched: Record<string, boolean> = $state({});
|
||||
|
||||
function isFieldEmpty(v: unknown): boolean {
|
||||
if (v === undefined || v === null || v === '' || v === 0) return true;
|
||||
if (Array.isArray(v) && v.length === 0) return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
function isRequired(field: FieldDef): boolean {
|
||||
if (field.type === 'separator') return false;
|
||||
return !!field.required;
|
||||
}
|
||||
|
||||
let dataFields = $derived(record.fields.filter((f) => f.type !== 'separator'));
|
||||
let filled = $derived(
|
||||
dataFields.filter((f) => !isFieldEmpty(data[slugify(f.label)])).length
|
||||
);
|
||||
</script>
|
||||
|
||||
<section class="rounded border" style="border-color: var(--border); background: var(--bg-card);">
|
||||
<button
|
||||
onclick={() => { expanded = !expanded; }}
|
||||
class="flex items-center w-full px-4 py-3 text-left gap-3 min-h-[44px]"
|
||||
>
|
||||
<ChevronDown
|
||||
size={16}
|
||||
class="shrink-0 transition-transform duration-200"
|
||||
style="transform: rotate({expanded ? '0' : '-90'}deg); color: var(--text-muted);"
|
||||
/>
|
||||
<div class="flex-1 min-w-0">
|
||||
<span class="font-medium capitalize">{record.type}</span>
|
||||
{#if record.note || record.preamble}
|
||||
<div class="flex flex-col{record.note && record.preamble ? ' gap-1' : ''}">
|
||||
{#if record.note}
|
||||
<span class="block text-xs" style="color: var(--text-muted);">{record.note}</span>
|
||||
{/if}
|
||||
{#if record.preamble}
|
||||
<span class="block text-xs" style="color: var(--text-muted);">{record.preamble}</span>
|
||||
{/if}
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
<span class="text-sm tabular-nums" style="color: var(--text-muted);">
|
||||
{filled}/{dataFields.length}
|
||||
</span>
|
||||
</button>
|
||||
|
||||
{#if expanded}
|
||||
<div transition:slide={{ duration: 150 }} class="px-4 pb-4 flex flex-col gap-4">
|
||||
{#each record.fields as field}
|
||||
{@const key = slugify(field.label)}
|
||||
{@const hasError = isRequired(field) && touched[key] && isFieldEmpty(data[key])}
|
||||
<div
|
||||
class={hasError ? 'field-error' : ''}
|
||||
onfocusout={(e) => {
|
||||
if (isRequired(field) && !(e.currentTarget as HTMLElement).contains(e.relatedTarget as Node)) {
|
||||
touched[key] = true;
|
||||
}
|
||||
}}
|
||||
>
|
||||
<DynamicField
|
||||
{field}
|
||||
value={data[key]}
|
||||
{data}
|
||||
onChange={(v) => onFieldChange(key, v)}
|
||||
/>
|
||||
{#if hasError}
|
||||
<p class="text-xs mt-1" style="color: var(--error);">This field is required</p>
|
||||
{/if}
|
||||
</div>
|
||||
{/each}
|
||||
</div>
|
||||
{/if}
|
||||
</section>
|
||||
212
src/lib/components/SchemaForm.svelte
Normal file
212
src/lib/components/SchemaForm.svelte
Normal file
|
|
@ -0,0 +1,212 @@
|
|||
<script lang="ts">
|
||||
import { ChevronDown } from 'lucide-svelte';
|
||||
import type { Character, Template } from '$lib/types';
|
||||
import { roster } from '$lib/state.svelte';
|
||||
import { presets } from '$lib/presets';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
import { diffTemplates, hasChanges } from '$lib/utils/template-diff';
|
||||
import RecordCard from './RecordCard.svelte';
|
||||
import Modal from './Modal.svelte';
|
||||
|
||||
let { character }: { character: Character } = $props();
|
||||
|
||||
let dismissed = $state<string | null>(null);
|
||||
let showTemplateSwitcher = $state(false);
|
||||
let showMigrationModal = $state(false);
|
||||
|
||||
let speciesKeys = $derived(new Set(
|
||||
character.template.records.flatMap((r) => r.fields)
|
||||
.filter((f) => f.type === 'species')
|
||||
.map((f) => slugify(f.label))
|
||||
));
|
||||
const SPECIES_DEPENDENT_TYPES = new Set(['subspecies', 'citizenship', 'languages']);
|
||||
let speciesDependentKeys = $derived(new Set(
|
||||
character.template.records.flatMap((r) => r.fields)
|
||||
.filter((f) => SPECIES_DEPENDENT_TYPES.has(f.type))
|
||||
.map((f) => slugify(f.label))
|
||||
));
|
||||
let speciesKey = slugify('Species');
|
||||
|
||||
let pendingMigration = $derived.by(() => {
|
||||
if (!character.template.id.startsWith('preset:')) return null;
|
||||
const preset = presets.find((p) => p.id === character.template.id);
|
||||
if (!preset) return null;
|
||||
const diff = diffTemplates(character.template, preset);
|
||||
if (!hasChanges(diff)) return null;
|
||||
return { preset, diff };
|
||||
});
|
||||
|
||||
let suggestion = $derived.by((): { template: Template; reason: string } | null => {
|
||||
const currentSpecies = character.data[speciesKey] as string | undefined;
|
||||
if (!currentSpecies) return null;
|
||||
const current = character.template;
|
||||
if (current.species?.length && !current.species.includes(currentSpecies)) {
|
||||
const specific = presets.find((p) =>
|
||||
p.species?.includes(currentSpecies) && p.id !== current.id
|
||||
);
|
||||
const general = presets.find((p) => !p.species && p.id !== current.id);
|
||||
const better = specific ?? general;
|
||||
if (better && better.id !== dismissed) {
|
||||
return {
|
||||
template: better,
|
||||
reason: `The ${current.name} template isn't designed for this species.`
|
||||
};
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
if (!current.species) {
|
||||
const specific = presets.find((p) =>
|
||||
p.species?.includes(currentSpecies) && p.id !== current.id
|
||||
);
|
||||
if (specific && specific.id !== dismissed) {
|
||||
return {
|
||||
template: specific,
|
||||
reason: `A ${specific.name} template is available for this species.`
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
return null;
|
||||
});
|
||||
|
||||
async function switchTemplate(template: Template) {
|
||||
await roster.migrateToPreset(character, template);
|
||||
dismissed = null;
|
||||
showTemplateSwitcher = false;
|
||||
}
|
||||
|
||||
async function applyMigration() {
|
||||
if (!pendingMigration) return;
|
||||
await roster.migrateToPreset(character, pendingMigration.preset);
|
||||
}
|
||||
</script>
|
||||
|
||||
<div class="flex flex-col gap-4">
|
||||
<!-- Template bar -->
|
||||
<div class="flex items-center gap-2 text-xs">
|
||||
<span class="relative">
|
||||
<button
|
||||
onclick={(e) => { e.stopPropagation(); showTemplateSwitcher = !showTemplateSwitcher; }}
|
||||
class="hover:underline"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
{character.template.name} template
|
||||
</button>
|
||||
{#if showTemplateSwitcher}
|
||||
<nav class="absolute z-10 mt-1 left-0 w-56 rounded border shadow-lg" style="background: var(--bg-card); border-color: var(--border);">
|
||||
{#each presets as preset}
|
||||
<button
|
||||
onclick={(e) => { e.stopPropagation(); switchTemplate(preset); }}
|
||||
class="block w-full text-left px-3 py-2 text-sm hover:opacity-80"
|
||||
style={preset.id === character.template.id ? 'color: var(--accent);' : 'color: var(--text);'}
|
||||
>
|
||||
<span class="font-medium">{preset.name}</span>
|
||||
{#if preset.description}
|
||||
<span class="block text-xs" style="color: var(--text-muted);">{preset.description}</span>
|
||||
{/if}
|
||||
</button>
|
||||
{/each}
|
||||
</nav>
|
||||
{/if}
|
||||
</span>
|
||||
{#if pendingMigration}
|
||||
<button
|
||||
onclick={() => { showMigrationModal = true; }}
|
||||
class="hover:underline"
|
||||
style="color: var(--accent);"
|
||||
>
|
||||
update available
|
||||
</button>
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
<!-- Species suggestion -->
|
||||
{#if suggestion}
|
||||
<div class="rounded border px-4 py-3" style="border-color: var(--accent); background: var(--bg-card);">
|
||||
<p class="text-sm mb-2">
|
||||
{suggestion.reason}
|
||||
Switching will keep your existing data.
|
||||
</p>
|
||||
<div class="flex gap-2">
|
||||
<button
|
||||
onclick={() => switchTemplate(suggestion!.template)}
|
||||
class="px-3 py-1 rounded text-sm border hover:opacity-80"
|
||||
style="border-color: var(--accent); color: var(--accent);"
|
||||
>
|
||||
Switch to {suggestion.template.name}
|
||||
</button>
|
||||
<button
|
||||
onclick={() => { dismissed = suggestion!.template.id; }}
|
||||
class="px-3 py-1 rounded text-sm hover:opacity-80"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Dismiss
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
{#each character.template.records as record}
|
||||
<RecordCard
|
||||
{record}
|
||||
data={character.data}
|
||||
onFieldChange={(key, value) => {
|
||||
character.data[key] = value;
|
||||
if (speciesKeys.has(key)) {
|
||||
for (const depKey of speciesDependentKeys) {
|
||||
character.data[depKey] = '';
|
||||
}
|
||||
}
|
||||
roster.scheduleSave(character);
|
||||
}}
|
||||
/>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
{#if showMigrationModal && pendingMigration}
|
||||
<Modal onClose={() => { showMigrationModal = false; }}>
|
||||
<h2 class="font-semibold mb-3">Template Update</h2>
|
||||
<p class="text-sm mb-2">The <strong>{pendingMigration.preset.name}</strong> template has been updated:</p>
|
||||
<ul class="text-sm flex flex-col gap-0.5 mb-3" style="color: var(--text-muted);">
|
||||
{#each pendingMigration.diff.renamedFields as r}
|
||||
<li>{r.from} → {r.to}</li>
|
||||
{/each}
|
||||
{#each pendingMigration.diff.addedRecords as r}
|
||||
<li>+ New record: {r}</li>
|
||||
{/each}
|
||||
{#each pendingMigration.diff.removedRecords as r}
|
||||
<li>- Removed record: {r}</li>
|
||||
{/each}
|
||||
{#each pendingMigration.diff.addedFields as f}
|
||||
<li>+ New field: {f}</li>
|
||||
{/each}
|
||||
{#each pendingMigration.diff.removedFields as f}
|
||||
<li>- Removed field: {f}</li>
|
||||
{/each}
|
||||
</ul>
|
||||
<p class="text-xs mb-3" style="color: var(--text-muted);">Your existing data will be preserved.</p>
|
||||
<div class="flex gap-2 justify-end">
|
||||
<button
|
||||
onclick={() => { showMigrationModal = false; }}
|
||||
class="px-3 py-1 rounded text-sm hover:opacity-80"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Skip
|
||||
</button>
|
||||
<button
|
||||
onclick={async () => { await applyMigration(); showMigrationModal = false; }}
|
||||
class="px-3 py-1 rounded text-sm border hover:opacity-80"
|
||||
style="border-color: var(--accent); color: var(--accent);"
|
||||
>
|
||||
Update
|
||||
</button>
|
||||
</div>
|
||||
</Modal>
|
||||
{/if}
|
||||
|
||||
<svelte:window onclick={() => {
|
||||
if (showTemplateSwitcher) {
|
||||
showTemplateSwitcher = false;
|
||||
}
|
||||
}} />
|
||||
70
src/lib/components/ShareMenu.svelte
Normal file
70
src/lib/components/ShareMenu.svelte
Normal file
|
|
@ -0,0 +1,70 @@
|
|||
<script lang="ts">
|
||||
import { Share2, Download, Check } from 'lucide-svelte';
|
||||
import { roster } from '$lib/state.svelte';
|
||||
import { encodeCharacterURL } from '$lib/sharing';
|
||||
import { exportCharacter, characterFileName } from '$lib/file';
|
||||
|
||||
let { open, onToggle }: { open: boolean; onToggle: () => void } = $props();
|
||||
|
||||
let copied = $state(false);
|
||||
|
||||
async function share() {
|
||||
const char = roster.active;
|
||||
if (!char) return;
|
||||
const encoded = encodeCharacterURL(char);
|
||||
const url = `${window.location.origin}${window.location.pathname}#${encoded}`;
|
||||
await navigator.clipboard.writeText(url);
|
||||
copied = true;
|
||||
setTimeout(() => { copied = false; }, 2000);
|
||||
}
|
||||
|
||||
function exportFile() {
|
||||
const char = roster.active;
|
||||
if (!char) return;
|
||||
const json = exportCharacter(char);
|
||||
const blob = new Blob([json], { type: 'application/json' });
|
||||
const url = URL.createObjectURL(blob);
|
||||
const a = document.createElement('a');
|
||||
a.href = url;
|
||||
a.download = characterFileName(char);
|
||||
a.click();
|
||||
URL.revokeObjectURL(url);
|
||||
onToggle();
|
||||
}
|
||||
</script>
|
||||
|
||||
<span class="relative">
|
||||
<button
|
||||
onclick={(e) => { e.stopPropagation(); onToggle(); }}
|
||||
class="flex items-center justify-center h-[30px] rounded border hover:opacity-80 {copied ? 'gap-1 px-2' : 'w-[30px]'}"
|
||||
style="border-color: var(--border);"
|
||||
title="Share & export"
|
||||
>
|
||||
{#if copied}
|
||||
<Check size={14} /> <span class="text-xs">Copied!</span>
|
||||
{:else}
|
||||
<Share2 size={14} />
|
||||
{/if}
|
||||
</button>
|
||||
|
||||
{#if open}
|
||||
<nav class="absolute left-0 z-10 mt-1 w-48 rounded border shadow-lg" style="background: var(--bg-card); border-color: var(--border);">
|
||||
<button
|
||||
onclick={share}
|
||||
class="flex items-center gap-2 w-full text-left px-3 py-2 text-sm hover:opacity-80"
|
||||
>
|
||||
{#if copied}
|
||||
<Check size={14} /> Copied!
|
||||
{:else}
|
||||
<Share2 size={14} /> Copy share link
|
||||
{/if}
|
||||
</button>
|
||||
<button
|
||||
onclick={exportFile}
|
||||
class="flex items-center gap-2 w-full text-left px-3 py-2 text-sm hover:opacity-80"
|
||||
>
|
||||
<Download size={14} /> Export to file
|
||||
</button>
|
||||
</nav>
|
||||
{/if}
|
||||
</span>
|
||||
31
src/lib/components/TemplatePicker.svelte
Normal file
31
src/lib/components/TemplatePicker.svelte
Normal file
|
|
@ -0,0 +1,31 @@
|
|||
<script lang="ts">
|
||||
import type { Template } from '$lib/types';
|
||||
import { presets } from '$lib/presets';
|
||||
import { roster } from '$lib/state.svelte';
|
||||
import Modal from './Modal.svelte';
|
||||
|
||||
let { onClose }: { onClose: () => void } = $props();
|
||||
|
||||
async function pick(template: Template) {
|
||||
await roster.create(template);
|
||||
onClose();
|
||||
}
|
||||
</script>
|
||||
|
||||
<Modal {onClose}>
|
||||
<h2 class="font-semibold mb-3">New Character</h2>
|
||||
<div class="flex flex-col gap-2">
|
||||
{#each presets as preset}
|
||||
<button
|
||||
onclick={() => pick(preset)}
|
||||
class="text-left px-3 py-2 rounded border hover:opacity-80"
|
||||
style="border-color: var(--border);"
|
||||
>
|
||||
<span class="font-medium text-sm">{preset.name}</span>
|
||||
{#if preset.description}
|
||||
<span class="block text-xs" style="color: var(--text-muted);">{preset.description}</span>
|
||||
{/if}
|
||||
</button>
|
||||
{/each}
|
||||
</div>
|
||||
</Modal>
|
||||
73
src/lib/components/fields/CheckboxField.svelte
Normal file
73
src/lib/components/fields/CheckboxField.svelte
Normal file
|
|
@ -0,0 +1,73 @@
|
|||
<script lang="ts">
|
||||
import { X } from 'lucide-svelte';
|
||||
import type { CheckboxField } from '$lib/types';
|
||||
|
||||
let { field, value = [], onChange }: { field: CheckboxField; value: string[]; onChange: (v: string[]) => void } = $props();
|
||||
|
||||
let customInput = $state('');
|
||||
|
||||
let knownValues = $derived(new Set(field.options.map((o) => o.value)));
|
||||
let customValues = $derived(value.filter((v) => !knownValues.has(v)));
|
||||
|
||||
function toggle(optValue: string) {
|
||||
if (value.includes(optValue)) {
|
||||
onChange(value.filter((v) => v !== optValue));
|
||||
} else {
|
||||
onChange([...value, optValue]);
|
||||
}
|
||||
}
|
||||
|
||||
function addCustom() {
|
||||
const trimmed = customInput.trim();
|
||||
if (trimmed && !value.includes(trimmed)) {
|
||||
onChange([...value, trimmed]);
|
||||
}
|
||||
customInput = '';
|
||||
}
|
||||
|
||||
function removeCustom(val: string) {
|
||||
onChange(value.filter((v) => v !== val));
|
||||
}
|
||||
</script>
|
||||
|
||||
<fieldset>
|
||||
<legend class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</legend>
|
||||
<div class="mt-1 flex flex-col gap-1">
|
||||
{#each field.options as opt}
|
||||
<label class="flex items-center gap-2 text-sm cursor-pointer">
|
||||
<input
|
||||
type="checkbox"
|
||||
checked={value.includes(opt.value)}
|
||||
onchange={() => toggle(opt.value)}
|
||||
/>
|
||||
{opt.label}
|
||||
</label>
|
||||
{/each}
|
||||
{#each customValues as cv}
|
||||
<span class="flex items-center gap-2 text-sm">
|
||||
<input type="checkbox" checked={true} onchange={() => removeCustom(cv)} />
|
||||
{cv}
|
||||
<button onclick={() => removeCustom(cv)} class="hover:opacity-60">
|
||||
<X size={12} />
|
||||
</button>
|
||||
</span>
|
||||
{/each}
|
||||
</div>
|
||||
<div class="flex items-center gap-2 mt-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={customInput}
|
||||
placeholder="Other..."
|
||||
onkeydown={(e) => { if (e.key === 'Enter') { e.preventDefault(); addCustom(); } }}
|
||||
class="rounded px-3 py-1.5 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
<button
|
||||
onclick={addCustom}
|
||||
class="text-sm hover:opacity-80"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Add
|
||||
</button>
|
||||
</div>
|
||||
</fieldset>
|
||||
69
src/lib/components/fields/CitizenshipField.svelte
Normal file
69
src/lib/components/fields/CitizenshipField.svelte
Normal file
|
|
@ -0,0 +1,69 @@
|
|||
<script lang="ts">
|
||||
import type { CitizenshipField } from '$lib/types';
|
||||
import { citizenships, species } from '$lib/data';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
|
||||
let { field, value = '', onChange, data }: {
|
||||
field: CitizenshipField;
|
||||
value: string;
|
||||
onChange: (v: string) => void;
|
||||
data: Record<string, unknown>;
|
||||
} = $props();
|
||||
|
||||
let currentSpecies = $derived(species.find((s) => s.id === data[slugify('Species')]));
|
||||
let filtered = $derived(
|
||||
currentSpecies
|
||||
? citizenships.filter((c) => currentSpecies!.citizenships.includes(c.id))
|
||||
: citizenships
|
||||
);
|
||||
|
||||
let custom = $state(false);
|
||||
|
||||
function handleSelect(v: string) {
|
||||
if (v === '__custom') {
|
||||
custom = true;
|
||||
onChange('');
|
||||
} else {
|
||||
custom = false;
|
||||
onChange(v);
|
||||
}
|
||||
}
|
||||
|
||||
let isCustom = $derived(custom || (value !== '' && !filtered.some((c) => c.name === value)));
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
{#if isCustom}
|
||||
<div class="flex items-center gap-2 mt-1">
|
||||
<input
|
||||
type="text"
|
||||
{value}
|
||||
placeholder="Enter citizenship"
|
||||
oninput={(e) => onChange((e.target as HTMLInputElement).value)}
|
||||
class="block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
<button
|
||||
onclick={() => { custom = false; onChange(''); }}
|
||||
class="text-sm whitespace-nowrap hover:opacity-80"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Cancel
|
||||
</button>
|
||||
</div>
|
||||
{:else}
|
||||
<select
|
||||
{value}
|
||||
onchange={(e) => handleSelect((e.target as HTMLSelectElement).value)}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
>
|
||||
<option value="">—</option>
|
||||
{#each filtered as c}
|
||||
<option value={c.name}>{c.name}</option>
|
||||
{/each}
|
||||
<option value="__custom">Other...</option>
|
||||
</select>
|
||||
{/if}
|
||||
</label>
|
||||
17
src/lib/components/fields/DateField.svelte
Normal file
17
src/lib/components/fields/DateField.svelte
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
<script lang="ts">
|
||||
import type { DateField } from '$lib/types';
|
||||
|
||||
let { field, value = '', onChange }: { field: DateField; value: string; onChange: (v: string) => void } = $props();
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<input
|
||||
type="text"
|
||||
{value}
|
||||
placeholder={field.placeholder}
|
||||
oninput={(e) => onChange((e.target as HTMLInputElement).value)}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
</label>
|
||||
59
src/lib/components/fields/DynamicField.svelte
Normal file
59
src/lib/components/fields/DynamicField.svelte
Normal file
|
|
@ -0,0 +1,59 @@
|
|||
<script lang="ts">
|
||||
import type { FieldDef } from '$lib/types';
|
||||
import TextField from './TextField.svelte';
|
||||
import TextareaField from './TextareaField.svelte';
|
||||
import ListField from './ListField.svelte';
|
||||
import NumberField from './NumberField.svelte';
|
||||
import SelectField from './SelectField.svelte';
|
||||
import CheckboxField from './CheckboxField.svelte';
|
||||
import MultiSelectField from './MultiSelectField.svelte';
|
||||
import DateField from './DateField.svelte';
|
||||
import HeightField from './HeightField.svelte';
|
||||
import WeightField from './WeightField.svelte';
|
||||
import SpeciesField from './SpeciesField.svelte';
|
||||
import SubspeciesField from './SubspeciesField.svelte';
|
||||
import CitizenshipField from './CitizenshipField.svelte';
|
||||
import LanguagesField from './LanguagesField.svelte';
|
||||
import SeparatorField from './SeparatorField.svelte';
|
||||
|
||||
let { field, value, onChange, data }: {
|
||||
field: FieldDef;
|
||||
value: any;
|
||||
onChange: (v: any) => void;
|
||||
data: Record<string, unknown>;
|
||||
} = $props();
|
||||
</script>
|
||||
|
||||
{#if field.type === 'name'}
|
||||
<TextField field={{ ...field, type: 'text' }} {value} {onChange} />
|
||||
{:else if field.type === 'text'}
|
||||
<TextField {field} {value} {onChange} />
|
||||
{:else if field.type === 'textarea'}
|
||||
<TextareaField {field} {value} {onChange} />
|
||||
{:else if field.type === 'list'}
|
||||
<ListField {field} {value} {onChange} />
|
||||
{:else if field.type === 'number'}
|
||||
<NumberField {field} {value} {onChange} />
|
||||
{:else if field.type === 'select'}
|
||||
<SelectField {field} {value} {onChange} />
|
||||
{:else if field.type === 'multi-select'}
|
||||
<MultiSelectField {field} {value} {onChange} />
|
||||
{:else if field.type === 'checkbox'}
|
||||
<CheckboxField {field} {value} {onChange} />
|
||||
{:else if field.type === 'date'}
|
||||
<DateField {field} {value} {onChange} />
|
||||
{:else if field.type === 'height'}
|
||||
<HeightField {field} {value} {onChange} />
|
||||
{:else if field.type === 'weight'}
|
||||
<WeightField {field} {value} {onChange} />
|
||||
{:else if field.type === 'species'}
|
||||
<SpeciesField {field} {value} {onChange} />
|
||||
{:else if field.type === 'subspecies'}
|
||||
<SubspeciesField {field} {value} {onChange} {data} />
|
||||
{:else if field.type === 'citizenship'}
|
||||
<CitizenshipField {field} {value} {onChange} {data} />
|
||||
{:else if field.type === 'languages'}
|
||||
<LanguagesField {field} {value} {onChange} {data} />
|
||||
{:else if field.type === 'separator'}
|
||||
<SeparatorField {field} />
|
||||
{/if}
|
||||
25
src/lib/components/fields/HeightField.svelte
Normal file
25
src/lib/components/fields/HeightField.svelte
Normal file
|
|
@ -0,0 +1,25 @@
|
|||
<script lang="ts">
|
||||
import type { HeightField } from '$lib/types';
|
||||
import { cmToFeetInches } from '$lib/utils/conversions';
|
||||
|
||||
let { field, value, onChange }: { field: HeightField; value: number | undefined; onChange: (v: number) => void } = $props();
|
||||
|
||||
let converted = $derived(value ? cmToFeetInches(value) : '');
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<div class="flex items-center gap-2 mt-1">
|
||||
<input
|
||||
type="number"
|
||||
value={value ?? ''}
|
||||
min={0}
|
||||
oninput={(e) => onChange(Number((e.target as HTMLInputElement).value))}
|
||||
class="w-24 rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
<span class="text-sm" style="color: var(--text-muted);">
|
||||
cm{#if converted} ({converted}){/if}
|
||||
</span>
|
||||
</div>
|
||||
</label>
|
||||
24
src/lib/components/fields/LanguagesField.svelte
Normal file
24
src/lib/components/fields/LanguagesField.svelte
Normal file
|
|
@ -0,0 +1,24 @@
|
|||
<script lang="ts">
|
||||
import type { LanguagesField } from '$lib/types';
|
||||
import { languages, species } from '$lib/data';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
import MultiSelectField from './MultiSelectField.svelte';
|
||||
|
||||
let { field, value = [], onChange, data }: {
|
||||
field: LanguagesField;
|
||||
value: string[];
|
||||
onChange: (v: string[]) => void;
|
||||
data: Record<string, unknown>;
|
||||
} = $props();
|
||||
|
||||
let currentSpecies = $derived(species.find((s) => s.id === data[slugify('Species')]));
|
||||
let filtered = $derived(
|
||||
currentSpecies
|
||||
? languages.filter((l) => currentSpecies!.languages.includes(l.id))
|
||||
: languages
|
||||
);
|
||||
let options = $derived(filtered.map((l) => ({ value: l.name, label: l.name })));
|
||||
let multiField = $derived({ ...field, type: 'multi-select' as const, options });
|
||||
</script>
|
||||
|
||||
<MultiSelectField field={multiField} {value} {onChange} />
|
||||
17
src/lib/components/fields/ListField.svelte
Normal file
17
src/lib/components/fields/ListField.svelte
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
<script lang="ts">
|
||||
import type { ListField } from '$lib/types';
|
||||
|
||||
let { field, value = '', onChange }: { field: ListField; value: string; onChange: (v: string) => void } = $props();
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<textarea
|
||||
{value}
|
||||
placeholder="One entry per line"
|
||||
oninput={(e) => onChange((e.target as HTMLTextAreaElement).value)}
|
||||
rows={3}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm resize-y"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
></textarea>
|
||||
</label>
|
||||
100
src/lib/components/fields/MultiSelectField.svelte
Normal file
100
src/lib/components/fields/MultiSelectField.svelte
Normal file
|
|
@ -0,0 +1,100 @@
|
|||
<script lang="ts">
|
||||
import { X } from 'lucide-svelte';
|
||||
import type { MultiSelectField } from '$lib/types';
|
||||
|
||||
let { field, value = [], onChange }: {
|
||||
field: MultiSelectField;
|
||||
value: string[];
|
||||
onChange: (v: string[]) => void;
|
||||
} = $props();
|
||||
|
||||
let input = $state('');
|
||||
let open = $state(false);
|
||||
|
||||
let available = $derived(
|
||||
field.options
|
||||
.filter((o) => !value.includes(o.value))
|
||||
.filter((o) => !input || o.label.toLowerCase().includes(input.toLowerCase()))
|
||||
);
|
||||
|
||||
function add(val: string) {
|
||||
onChange([...value, val]);
|
||||
input = '';
|
||||
}
|
||||
|
||||
function addCustom() {
|
||||
const trimmed = input.trim();
|
||||
if (trimmed && !value.includes(trimmed)) {
|
||||
onChange([...value, trimmed]);
|
||||
}
|
||||
input = '';
|
||||
}
|
||||
|
||||
function remove(val: string) {
|
||||
onChange(value.filter((v) => v !== val));
|
||||
}
|
||||
|
||||
function displayLabel(val: string): string {
|
||||
return field.options.find((o) => o.value === val)?.label ?? val;
|
||||
}
|
||||
</script>
|
||||
|
||||
<div class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
|
||||
<div class="flex flex-wrap gap-1 mt-1">
|
||||
{#each value as val}
|
||||
<span class="inline-flex items-center gap-1 px-2 py-0.5 rounded text-sm" style="background: var(--border); color: var(--text);">
|
||||
{displayLabel(val)}
|
||||
<button onclick={() => remove(val)} class="hover:opacity-60">
|
||||
<X size={12} />
|
||||
</button>
|
||||
</span>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<div class="relative mt-1">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={input}
|
||||
placeholder="Add..."
|
||||
onfocus={() => { open = true; }}
|
||||
onblur={() => { setTimeout(() => { open = false; }, 150); }}
|
||||
onkeydown={(e) => {
|
||||
if (e.key === 'Enter') {
|
||||
e.preventDefault();
|
||||
if (available.length) add(available[0].value);
|
||||
else addCustom();
|
||||
}
|
||||
}}
|
||||
class="block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
{#if open && (available.length || input.trim())}
|
||||
<ul class="absolute z-10 w-full mt-1 rounded shadow-lg max-h-48 overflow-y-auto" style="background: var(--bg-card); border: 1px solid var(--border);">
|
||||
{#each available as opt}
|
||||
<li>
|
||||
<button
|
||||
onmousedown={(e) => { e.preventDefault(); add(opt.value); }}
|
||||
class="block w-full text-left px-3 py-1.5 text-sm hover:opacity-80"
|
||||
style="color: var(--text);"
|
||||
>
|
||||
{opt.label}
|
||||
</button>
|
||||
</li>
|
||||
{/each}
|
||||
{#if input.trim() && !field.options.some((o) => o.label.toLowerCase() === input.trim().toLowerCase())}
|
||||
<li>
|
||||
<button
|
||||
onmousedown={(e) => { e.preventDefault(); addCustom(); }}
|
||||
class="block w-full text-left px-3 py-1.5 text-sm"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Add "{input.trim()}"
|
||||
</button>
|
||||
</li>
|
||||
{/if}
|
||||
</ul>
|
||||
{/if}
|
||||
</div>
|
||||
</div>
|
||||
23
src/lib/components/fields/NumberField.svelte
Normal file
23
src/lib/components/fields/NumberField.svelte
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
<script lang="ts">
|
||||
import type { NumberField } from '$lib/types';
|
||||
|
||||
let { field, value, onChange }: { field: NumberField; value: number | undefined; onChange: (v: number) => void } = $props();
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<div class="flex items-center gap-2 mt-1">
|
||||
<input
|
||||
type="number"
|
||||
value={value ?? ''}
|
||||
min={field.min}
|
||||
max={field.max}
|
||||
oninput={(e) => onChange(Number((e.target as HTMLInputElement).value))}
|
||||
class="block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
{#if field.unit}
|
||||
<span class="text-sm" style="color: var(--text-muted);">{field.unit}</span>
|
||||
{/if}
|
||||
</div>
|
||||
</label>
|
||||
54
src/lib/components/fields/SelectField.svelte
Normal file
54
src/lib/components/fields/SelectField.svelte
Normal file
|
|
@ -0,0 +1,54 @@
|
|||
<script lang="ts">
|
||||
import type { SelectField } from '$lib/types';
|
||||
|
||||
let { field, value = '', onChange }: { field: SelectField; value: string; onChange: (v: string) => void } = $props();
|
||||
|
||||
let custom = $state(false);
|
||||
|
||||
function handleSelect(v: string) {
|
||||
if (v === '__custom') {
|
||||
custom = true;
|
||||
onChange('');
|
||||
} else {
|
||||
custom = false;
|
||||
onChange(v);
|
||||
}
|
||||
}
|
||||
|
||||
let isCustom = $derived(custom || (value !== '' && !field.options.some((o) => o.value === value)));
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
{#if isCustom}
|
||||
<div class="flex items-center gap-2 mt-1">
|
||||
<input
|
||||
type="text"
|
||||
{value}
|
||||
oninput={(e) => onChange((e.target as HTMLInputElement).value)}
|
||||
class="block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
<button
|
||||
onclick={() => { custom = false; onChange(''); }}
|
||||
class="text-sm whitespace-nowrap hover:opacity-80"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Cancel
|
||||
</button>
|
||||
</div>
|
||||
{:else}
|
||||
<select
|
||||
{value}
|
||||
onchange={(e) => handleSelect((e.target as HTMLSelectElement).value)}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
>
|
||||
<option value="">—</option>
|
||||
{#each field.options as opt}
|
||||
<option value={opt.value}>{opt.label}</option>
|
||||
{/each}
|
||||
<option value="__custom">Other...</option>
|
||||
</select>
|
||||
{/if}
|
||||
</label>
|
||||
15
src/lib/components/fields/SeparatorField.svelte
Normal file
15
src/lib/components/fields/SeparatorField.svelte
Normal file
|
|
@ -0,0 +1,15 @@
|
|||
<script lang="ts">
|
||||
import type { SeparatorField } from '$lib/types';
|
||||
|
||||
let { field }: { field: SeparatorField } = $props();
|
||||
</script>
|
||||
|
||||
{#if field.label}
|
||||
<div class="flex items-center gap-3 pt-2">
|
||||
<hr class="flex-1" style="border-color: var(--border);" />
|
||||
<span class="text-xs font-medium uppercase tracking-wide" style="color: var(--text-muted);">{field.label}</span>
|
||||
<hr class="flex-1" style="border-color: var(--border);" />
|
||||
</div>
|
||||
{:else}
|
||||
<hr class="mt-2" style="border-color: var(--border);" />
|
||||
{/if}
|
||||
28
src/lib/components/fields/SpeciesField.svelte
Normal file
28
src/lib/components/fields/SpeciesField.svelte
Normal file
|
|
@ -0,0 +1,28 @@
|
|||
<script lang="ts">
|
||||
import type { SpeciesField } from '$lib/types';
|
||||
import { species } from '$lib/data';
|
||||
|
||||
let { field, value = '', onChange }: { field: SpeciesField; value: string; onChange: (v: string) => void } = $props();
|
||||
|
||||
let selected = $derived(species.find((s) => s.id === value));
|
||||
</script>
|
||||
|
||||
<div class="block">
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<select
|
||||
{value}
|
||||
onchange={(e) => onChange((e.target as HTMLSelectElement).value)}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
>
|
||||
<option value="">—</option>
|
||||
{#each species as sp}
|
||||
<option value={sp.id}>{sp.name}</option>
|
||||
{/each}
|
||||
</select>
|
||||
</label>
|
||||
{#if selected?.description}
|
||||
<p class="mt-1 text-sm italic" style="color: var(--text-muted);">{selected.description}</p>
|
||||
{/if}
|
||||
</div>
|
||||
73
src/lib/components/fields/SubspeciesField.svelte
Normal file
73
src/lib/components/fields/SubspeciesField.svelte
Normal file
|
|
@ -0,0 +1,73 @@
|
|||
<script lang="ts">
|
||||
import type { SubspeciesField } from '$lib/types';
|
||||
import { species } from '$lib/data';
|
||||
import { slugify } from '$lib/utils/slugify';
|
||||
|
||||
let { field, value = '', onChange, data }: {
|
||||
field: SubspeciesField;
|
||||
value: string;
|
||||
onChange: (v: string) => void;
|
||||
data: Record<string, unknown>;
|
||||
} = $props();
|
||||
|
||||
let currentSpecies = $derived(species.find((s) => s.id === data[slugify('Species')]));
|
||||
let subs = $derived(currentSpecies?.subspecies ?? []);
|
||||
let label = $derived(currentSpecies?.subspeciesLabel ?? field.label);
|
||||
let selected = $derived(subs.find((s) => s.id === value));
|
||||
|
||||
let custom = $state(false);
|
||||
|
||||
function handleSelect(v: string) {
|
||||
if (v === '__custom') {
|
||||
custom = true;
|
||||
onChange('');
|
||||
} else {
|
||||
custom = false;
|
||||
onChange(v);
|
||||
}
|
||||
}
|
||||
|
||||
let isCustom = $derived(custom || (value !== '' && !subs.some((s) => s.id === value)));
|
||||
</script>
|
||||
|
||||
{#if subs.length > 0 || isCustom}
|
||||
<div class="block">
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
{#if isCustom}
|
||||
<div class="flex items-center gap-2 mt-1">
|
||||
<input
|
||||
type="text"
|
||||
{value}
|
||||
oninput={(e) => onChange((e.target as HTMLInputElement).value)}
|
||||
class="block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
<button
|
||||
onclick={() => { custom = false; onChange(''); }}
|
||||
class="text-sm whitespace-nowrap hover:opacity-80"
|
||||
style="color: var(--text-muted);"
|
||||
>
|
||||
Cancel
|
||||
</button>
|
||||
</div>
|
||||
{:else}
|
||||
<select
|
||||
{value}
|
||||
onchange={(e) => handleSelect((e.target as HTMLSelectElement).value)}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
>
|
||||
<option value="">—</option>
|
||||
{#each subs as sub}
|
||||
<option value={sub.id}>{sub.name}</option>
|
||||
{/each}
|
||||
<option value="__custom">Other...</option>
|
||||
</select>
|
||||
{/if}
|
||||
</label>
|
||||
{#if selected?.description}
|
||||
<p class="mt-1 text-sm italic" style="color: var(--text-muted);">{selected.description}</p>
|
||||
{/if}
|
||||
</div>
|
||||
{/if}
|
||||
17
src/lib/components/fields/TextField.svelte
Normal file
17
src/lib/components/fields/TextField.svelte
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
<script lang="ts">
|
||||
import type { TextField } from '$lib/types';
|
||||
|
||||
let { field, value = '', onChange }: { field: TextField; value: string; onChange: (v: string) => void } = $props();
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<input
|
||||
type="text"
|
||||
{value}
|
||||
placeholder={field.placeholder}
|
||||
oninput={(e) => onChange((e.target as HTMLInputElement).value)}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
</label>
|
||||
17
src/lib/components/fields/TextareaField.svelte
Normal file
17
src/lib/components/fields/TextareaField.svelte
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
<script lang="ts">
|
||||
import type { TextareaField } from '$lib/types';
|
||||
|
||||
let { field, value = '', onChange }: { field: TextareaField; value: string; onChange: (v: string) => void } = $props();
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<textarea
|
||||
{value}
|
||||
placeholder={field.placeholder}
|
||||
oninput={(e) => onChange((e.target as HTMLTextAreaElement).value)}
|
||||
rows={3}
|
||||
class="mt-1 block w-full rounded px-3 py-2 text-sm resize-y"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
></textarea>
|
||||
</label>
|
||||
25
src/lib/components/fields/WeightField.svelte
Normal file
25
src/lib/components/fields/WeightField.svelte
Normal file
|
|
@ -0,0 +1,25 @@
|
|||
<script lang="ts">
|
||||
import type { WeightField } from '$lib/types';
|
||||
import { kgToLb } from '$lib/utils/conversions';
|
||||
|
||||
let { field, value, onChange }: { field: WeightField; value: number | undefined; onChange: (v: number) => void } = $props();
|
||||
|
||||
let converted = $derived(value ? Math.round(kgToLb(value)) : 0);
|
||||
</script>
|
||||
|
||||
<label class="block">
|
||||
<span class="text-sm font-medium">{field.label}{#if field.required}<span style="color: var(--accent);"> *</span>{/if}</span>
|
||||
<div class="flex items-center gap-2 mt-1">
|
||||
<input
|
||||
type="number"
|
||||
value={value ?? ''}
|
||||
min={0}
|
||||
oninput={(e) => onChange(Number((e.target as HTMLInputElement).value))}
|
||||
class="w-24 rounded px-3 py-2 text-sm"
|
||||
style="background: var(--bg-input); border: 1px solid var(--border); color: var(--text);"
|
||||
/>
|
||||
<span class="text-sm" style="color: var(--text-muted);">
|
||||
kg{#if converted} ({converted} lb){/if}
|
||||
</span>
|
||||
</div>
|
||||
</label>
|
||||
13
src/lib/data/index.ts
Normal file
13
src/lib/data/index.ts
Normal file
|
|
@ -0,0 +1,13 @@
|
|||
import { parseSpecies, parseCitizenships, parseLanguages } from './parse';
|
||||
import citizenshipsXml from '../../../data/citizenships.xml?raw';
|
||||
import languagesXml from '../../../data/languages.xml?raw';
|
||||
|
||||
const speciesModules = import.meta.glob('../../../data/species/*.xml', {
|
||||
query: '?raw',
|
||||
import: 'default',
|
||||
eager: true
|
||||
});
|
||||
|
||||
export const species = Object.values(speciesModules).map((xml) => parseSpecies(xml as string));
|
||||
export const citizenships = parseCitizenships(citizenshipsXml);
|
||||
export const languages = parseLanguages(languagesXml);
|
||||
122
src/lib/data/parse.ts
Normal file
122
src/lib/data/parse.ts
Normal file
|
|
@ -0,0 +1,122 @@
|
|||
import { XMLParser } from 'fast-xml-parser';
|
||||
import type { SpeciesData, CitizenshipData, LanguageData } from './types';
|
||||
import type { Template, RecordDef, FieldDef, SelectOption } from '../types';
|
||||
|
||||
const parser = new XMLParser({
|
||||
ignoreAttributes: false,
|
||||
attributeNamePrefix: '@_',
|
||||
isArray: (name) => ['entry', 'ref', 'field', 'record', 'option', 'citizenship', 'language'].includes(name),
|
||||
trimValues: true
|
||||
});
|
||||
|
||||
function extractRefs(container: any): string[] {
|
||||
if (!container?.ref) return [];
|
||||
return container.ref.map((r: any) => r['@_id']);
|
||||
}
|
||||
|
||||
export function parseSpecies(xml: string): SpeciesData {
|
||||
const root = parser.parse(xml).species;
|
||||
const subspecies = root.subspecies?.entry ?? [];
|
||||
|
||||
return {
|
||||
id: root['@_id'],
|
||||
name: root['@_name'],
|
||||
description: root.description?.trim(),
|
||||
subspeciesLabel: root['@_subspeciesLabel'],
|
||||
languages: extractRefs(root.languages),
|
||||
citizenships: extractRefs(root.citizenships),
|
||||
subspecies: subspecies.map((e: any) => ({
|
||||
id: e['@_id'],
|
||||
name: e['@_name'],
|
||||
description: e.description?.trim()
|
||||
}))
|
||||
};
|
||||
}
|
||||
|
||||
export function parseCitizenships(xml: string): CitizenshipData[] {
|
||||
const root = parser.parse(xml).citizenships;
|
||||
return root.citizenship.map((c: any) => ({
|
||||
id: c['@_id'],
|
||||
name: c['@_name'],
|
||||
description: c.description?.trim()
|
||||
}));
|
||||
}
|
||||
|
||||
export function parseLanguages(xml: string): LanguageData[] {
|
||||
const root = parser.parse(xml).languages;
|
||||
return root.language.map((l: any) => ({
|
||||
id: l['@_id'],
|
||||
name: l['@_name'],
|
||||
description: l.description?.trim()
|
||||
}));
|
||||
}
|
||||
|
||||
function parseOptions(field: any): SelectOption[] {
|
||||
if (!field.option) return [];
|
||||
return field.option.map((o: any) => ({
|
||||
value: o['@_value'],
|
||||
label: o['@_label']
|
||||
}));
|
||||
}
|
||||
|
||||
function parseField(raw: any): FieldDef {
|
||||
const base = {
|
||||
label: raw['@_label'],
|
||||
...(raw['@_required'] === 'true' && { required: true }),
|
||||
...(raw['@_from'] && { from: raw['@_from'] })
|
||||
};
|
||||
const type = raw['@_type'];
|
||||
|
||||
switch (type) {
|
||||
case 'name':
|
||||
case 'text':
|
||||
case 'textarea':
|
||||
case 'date':
|
||||
return { ...base, type, placeholder: raw['@_placeholder'] };
|
||||
case 'list':
|
||||
case 'height':
|
||||
case 'weight':
|
||||
case 'species':
|
||||
case 'subspecies':
|
||||
case 'citizenship':
|
||||
case 'languages':
|
||||
return { ...base, type };
|
||||
case 'separator':
|
||||
return { type: 'separator', label: raw['@_label'] ?? '' };
|
||||
case 'number':
|
||||
return {
|
||||
...base,
|
||||
type,
|
||||
min: raw['@_min'] != null ? Number(raw['@_min']) : undefined,
|
||||
max: raw['@_max'] != null ? Number(raw['@_max']) : undefined,
|
||||
unit: raw['@_unit']
|
||||
};
|
||||
case 'select':
|
||||
case 'multi-select':
|
||||
case 'checkbox':
|
||||
return { ...base, type, options: parseOptions(raw) };
|
||||
default:
|
||||
return { ...base, type: 'text' };
|
||||
}
|
||||
}
|
||||
|
||||
export function parseTemplate(xml: string, id: string): Template {
|
||||
const root = parser.parse(xml).template;
|
||||
|
||||
const records: RecordDef[] = root.record.map((r: any) => ({
|
||||
type: r['@_type'],
|
||||
preamble: r.preamble?.trim(),
|
||||
note: r.note?.trim(),
|
||||
fields: r.field.map(parseField)
|
||||
}));
|
||||
|
||||
const speciesAttr = root['@_species'];
|
||||
return {
|
||||
id,
|
||||
name: root['@_name'],
|
||||
description: root.description ?? '',
|
||||
schemaVersion: Number(root['@_schemaVersion'] ?? 1),
|
||||
...(speciesAttr && { species: speciesAttr.split(',').map((s: string) => s.trim()) }),
|
||||
records
|
||||
};
|
||||
}
|
||||
21
src/lib/data/types.ts
Normal file
21
src/lib/data/types.ts
Normal file
|
|
@ -0,0 +1,21 @@
|
|||
export interface SpeciesData {
|
||||
id: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
subspeciesLabel: string;
|
||||
subspecies: { id: string; name: string; description?: string }[];
|
||||
languages: string[];
|
||||
citizenships: string[];
|
||||
}
|
||||
|
||||
export interface CitizenshipData {
|
||||
id: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
}
|
||||
|
||||
export interface LanguageData {
|
||||
id: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
}
|
||||
110
src/lib/file.test.ts
Normal file
110
src/lib/file.test.ts
Normal file
|
|
@ -0,0 +1,110 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import { exportCharacter, parseCharacterFile } from './file';
|
||||
import { presets } from './presets';
|
||||
import type { Character } from './types';
|
||||
|
||||
const standardPreset = presets.find((p) => p.id === 'preset:standard')!;
|
||||
|
||||
const testCharacter: Character = {
|
||||
id: 'abc-123',
|
||||
template: standardPreset,
|
||||
data: {
|
||||
name: 'Yury Zakharov',
|
||||
species: 'human',
|
||||
'employment-history': 'Shaft Miner'
|
||||
}
|
||||
};
|
||||
|
||||
describe('exportCharacter', () => {
|
||||
it('returns valid JSON with version, templateId, template, and data', () => {
|
||||
const json = exportCharacter(testCharacter);
|
||||
const parsed = JSON.parse(json);
|
||||
expect(parsed.version).toBe(1);
|
||||
expect(parsed.templateId).toBe('preset:standard');
|
||||
expect(parsed.template).toBeDefined();
|
||||
expect(parsed.template.name).toBe('General');
|
||||
expect(parsed.data).toEqual({
|
||||
name: 'Yury Zakharov',
|
||||
species: 'human',
|
||||
'employment-history': 'Shaft Miner'
|
||||
});
|
||||
});
|
||||
|
||||
it('strips template id from embedded template', () => {
|
||||
const json = exportCharacter(testCharacter);
|
||||
const parsed = JSON.parse(json);
|
||||
expect(parsed.template).not.toHaveProperty('id');
|
||||
});
|
||||
|
||||
it('prunes empty values from data', () => {
|
||||
const char: Character = {
|
||||
...testCharacter,
|
||||
data: { name: 'Yury Zakharov', species: '', 'hair-color': '' }
|
||||
};
|
||||
const json = exportCharacter(char);
|
||||
const parsed = JSON.parse(json);
|
||||
expect(parsed.data).toEqual({ name: 'Yury Zakharov' });
|
||||
});
|
||||
|
||||
it('omits templateId for non-preset templates', () => {
|
||||
const char: Character = {
|
||||
...testCharacter,
|
||||
template: {
|
||||
id: 'custom:test',
|
||||
name: 'Custom',
|
||||
description: 'Test',
|
||||
schemaVersion: 1,
|
||||
records: []
|
||||
}
|
||||
};
|
||||
const json = exportCharacter(char);
|
||||
const parsed = JSON.parse(json);
|
||||
expect(parsed.templateId).toBeUndefined();
|
||||
expect(parsed.template.name).toBe('Custom');
|
||||
});
|
||||
});
|
||||
|
||||
describe('parseCharacterFile', () => {
|
||||
it('resolves preset template by templateId', () => {
|
||||
const json = exportCharacter(testCharacter);
|
||||
const result = parseCharacterFile(json);
|
||||
expect(result.template).toHaveProperty('id', 'preset:standard');
|
||||
expect(result.data.name).toBe('Yury Zakharov');
|
||||
});
|
||||
|
||||
it('falls back to embedded template for unknown preset', () => {
|
||||
const payload = {
|
||||
version: 1,
|
||||
templateId: 'preset:nonexistent',
|
||||
template: { name: 'Fallback', description: '', schemaVersion: 1, records: [] },
|
||||
data: { name: 'Test' }
|
||||
};
|
||||
const result = parseCharacterFile(JSON.stringify(payload));
|
||||
expect(result.template.name).toBe('Fallback');
|
||||
expect(result.template).not.toHaveProperty('id');
|
||||
});
|
||||
|
||||
it('uses embedded template when no templateId', () => {
|
||||
const payload = {
|
||||
version: 1,
|
||||
template: { name: 'Custom', description: '', schemaVersion: 1, records: [] },
|
||||
data: { name: 'Test' }
|
||||
};
|
||||
const result = parseCharacterFile(JSON.stringify(payload));
|
||||
expect(result.template.name).toBe('Custom');
|
||||
});
|
||||
|
||||
it('throws on invalid JSON', () => {
|
||||
expect(() => parseCharacterFile('not json')).toThrow();
|
||||
});
|
||||
|
||||
it('throws on missing data field', () => {
|
||||
const payload = { version: 1, template: { name: 'X', description: '', schemaVersion: 1, records: [] } };
|
||||
expect(() => parseCharacterFile(JSON.stringify(payload))).toThrow();
|
||||
});
|
||||
|
||||
it('throws on missing template and templateId', () => {
|
||||
const payload = { version: 1, data: { name: 'Test' } };
|
||||
expect(() => parseCharacterFile(JSON.stringify(payload))).toThrow();
|
||||
});
|
||||
});
|
||||
53
src/lib/file.ts
Normal file
53
src/lib/file.ts
Normal file
|
|
@ -0,0 +1,53 @@
|
|||
import type { Character, Template } from './types';
|
||||
import { pruneEmpty } from './sharing';
|
||||
import { presets } from './presets';
|
||||
import { slugify } from './utils/slugify';
|
||||
|
||||
interface CharacterFilePayload {
|
||||
version: number;
|
||||
templateId?: string;
|
||||
template: Omit<Template, 'id'>;
|
||||
data: Record<string, unknown>;
|
||||
}
|
||||
|
||||
export function exportCharacter(char: Character): string {
|
||||
const isPreset = char.template.id.startsWith('preset:');
|
||||
const { id, ...templateWithoutId } = char.template;
|
||||
const payload: CharacterFilePayload = {
|
||||
version: 1,
|
||||
template: templateWithoutId,
|
||||
data: pruneEmpty(char.data)
|
||||
};
|
||||
if (isPreset) {
|
||||
payload.templateId = char.template.id;
|
||||
}
|
||||
return JSON.stringify(payload, null, 2);
|
||||
}
|
||||
|
||||
export function parseCharacterFile(json: string): { template: Template | Omit<Template, 'id'>; data: Record<string, unknown> } {
|
||||
const payload = JSON.parse(json);
|
||||
if (!payload.data || typeof payload.data !== 'object') {
|
||||
throw new Error('Invalid character file: missing data');
|
||||
}
|
||||
if (!payload.template && !payload.templateId) {
|
||||
throw new Error('Invalid character file: missing template');
|
||||
}
|
||||
if (payload.templateId) {
|
||||
const preset = presets.find((p) => p.id === payload.templateId);
|
||||
if (preset) {
|
||||
return { template: preset, data: payload.data };
|
||||
}
|
||||
}
|
||||
if (payload.template) {
|
||||
return { template: payload.template, data: payload.data };
|
||||
}
|
||||
throw new Error('Invalid character file: could not resolve template');
|
||||
}
|
||||
|
||||
export function characterFileName(char: Character): string {
|
||||
const nameField = char.template.records.flatMap((r) => r.fields).find((f) => f.type === 'name');
|
||||
const key = nameField ? slugify(nameField.label) : 'name';
|
||||
const name = char.data[key] as string | undefined;
|
||||
if (!name || !name.trim()) return 'character.json';
|
||||
return name.trim().replace(/[^a-zA-Z0-9'-]/g, '-').replace(/-+/g, '-') + '.json';
|
||||
}
|
||||
272
src/lib/output.test.ts
Normal file
272
src/lib/output.test.ts
Normal file
|
|
@ -0,0 +1,272 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import { formatFieldOutput, generateRecord } from './output';
|
||||
import type { FieldDef, Template } from './types';
|
||||
import type { SpeciesData } from './data/types';
|
||||
|
||||
const stubSpecies: SpeciesData[] = [
|
||||
{
|
||||
id: 'human',
|
||||
name: 'Human',
|
||||
description: '',
|
||||
subspeciesLabel: 'Variant',
|
||||
subspecies: [{ id: 'offworlder', name: 'Offworlder', description: '' }],
|
||||
languages: ['tau-ceti-basic'],
|
||||
citizenships: ['biesel']
|
||||
},
|
||||
{
|
||||
id: 'tajara',
|
||||
name: 'Tajara',
|
||||
description: '',
|
||||
subspeciesLabel: 'Ethnicity',
|
||||
subspecies: [
|
||||
{ id: 'hharar', name: 'Hharar', description: '' },
|
||||
{ id: 'zhan-khazan', name: 'Zhan-Khazan', description: '' }
|
||||
],
|
||||
languages: ['siik-maas'],
|
||||
citizenships: ['pra']
|
||||
}
|
||||
];
|
||||
|
||||
describe('formatFieldOutput', () => {
|
||||
it('formats text fields', () => {
|
||||
const field: FieldDef = { label: 'Pronouns', type: 'text' };
|
||||
expect(formatFieldOutput(field, 'she/her')).toBe('Pronouns: she/her');
|
||||
});
|
||||
|
||||
it('returns null for empty text', () => {
|
||||
const field: FieldDef = { label: 'Pronouns', type: 'text' };
|
||||
expect(formatFieldOutput(field, '')).toBeNull();
|
||||
expect(formatFieldOutput(field, undefined)).toBeNull();
|
||||
});
|
||||
|
||||
it('formats textarea with header', () => {
|
||||
const field: FieldDef = { label: 'Distinguishing Features', type: 'textarea' };
|
||||
expect(formatFieldOutput(field, 'Scar across left eye')).toBe(
|
||||
'Distinguishing Features:\nScar across left eye'
|
||||
);
|
||||
});
|
||||
|
||||
it('returns null for empty textarea', () => {
|
||||
const field: FieldDef = { label: 'Distinguishing Features', type: 'textarea' };
|
||||
expect(formatFieldOutput(field, '')).toBeNull();
|
||||
});
|
||||
|
||||
it('formats list as bullet points', () => {
|
||||
const field: FieldDef = { label: 'Employment History', type: 'list' };
|
||||
expect(formatFieldOutput(field, 'Shaft Miner')).toBe(
|
||||
'Employment History:\n - Shaft Miner'
|
||||
);
|
||||
});
|
||||
|
||||
it('returns null for empty list', () => {
|
||||
const field: FieldDef = { label: 'Employment History', type: 'list' };
|
||||
expect(formatFieldOutput(field, '')).toBeNull();
|
||||
});
|
||||
|
||||
it('filters blank lines from list', () => {
|
||||
const field: FieldDef = { label: 'Employment History', type: 'list' };
|
||||
expect(formatFieldOutput(field, 'Line 1\n\nLine 2\n')).toBe(
|
||||
'Employment History:\n - Line 1\n - Line 2'
|
||||
);
|
||||
});
|
||||
|
||||
it('formats height with conversion', () => {
|
||||
const field: FieldDef = { label: 'Height', type: 'height' };
|
||||
expect(formatFieldOutput(field, 180)).toBe('Height: 180 cm (5\'11")');
|
||||
});
|
||||
|
||||
it('returns null for zero/undefined height', () => {
|
||||
const field: FieldDef = { label: 'Height', type: 'height' };
|
||||
expect(formatFieldOutput(field, 0)).toBeNull();
|
||||
expect(formatFieldOutput(field, undefined)).toBeNull();
|
||||
});
|
||||
|
||||
it('formats weight with conversion', () => {
|
||||
const field: FieldDef = { label: 'Weight', type: 'weight' };
|
||||
expect(formatFieldOutput(field, 75)).toBe('Weight: 75 kg (165 lb)');
|
||||
});
|
||||
|
||||
it('returns null for zero/undefined weight', () => {
|
||||
const field: FieldDef = { label: 'Weight', type: 'weight' };
|
||||
expect(formatFieldOutput(field, 0)).toBeNull();
|
||||
});
|
||||
|
||||
it('formats species with display name', () => {
|
||||
const field: FieldDef = { label: 'Species', type: 'species' };
|
||||
expect(formatFieldOutput(field, 'tajara', stubSpecies)).toBe('Species: Tajara');
|
||||
});
|
||||
|
||||
it('formats subspecies with dynamic label', () => {
|
||||
const field: FieldDef = { label: 'Subspecies', type: 'subspecies' };
|
||||
expect(formatFieldOutput(field, 'hharar', stubSpecies, 'tajara')).toBe('Ethnicity: Hharar');
|
||||
});
|
||||
|
||||
it('returns null for empty subspecies', () => {
|
||||
const field: FieldDef = { label: 'Subspecies', type: 'subspecies' };
|
||||
expect(formatFieldOutput(field, '', stubSpecies, 'tajara')).toBeNull();
|
||||
});
|
||||
|
||||
it('formats languages as comma list', () => {
|
||||
const field: FieldDef = { label: 'Spoken Languages', type: 'languages' };
|
||||
expect(formatFieldOutput(field, ['Tau Ceti Basic', 'Siik\'maas'])).toBe(
|
||||
'Spoken Languages: Tau Ceti Basic, Siik\'maas'
|
||||
);
|
||||
});
|
||||
|
||||
it('returns null for empty languages', () => {
|
||||
const field: FieldDef = { label: 'Spoken Languages', type: 'languages' };
|
||||
expect(formatFieldOutput(field, [])).toBeNull();
|
||||
});
|
||||
|
||||
it('formats checkbox as bullet list of selected', () => {
|
||||
const field: FieldDef = {
|
||||
label: 'Opt-Outs',
|
||||
type: 'checkbox',
|
||||
options: [
|
||||
{ value: 'no-borg', label: 'Do Not Borgify' },
|
||||
{ value: 'no-revive', label: 'Do Not Revive' },
|
||||
{ value: 'no-prosthetic', label: 'Do Not Prostheticize' }
|
||||
]
|
||||
};
|
||||
expect(formatFieldOutput(field, ['no-borg', 'no-revive'])).toBe(
|
||||
'Opt-Outs:\n - Do Not Borgify\n - Do Not Revive'
|
||||
);
|
||||
});
|
||||
|
||||
it('returns null for empty checkbox', () => {
|
||||
const field: FieldDef = {
|
||||
label: 'Opt-Outs',
|
||||
type: 'checkbox',
|
||||
options: [{ value: 'no-borg', label: 'Do Not Borgify' }]
|
||||
};
|
||||
expect(formatFieldOutput(field, [])).toBeNull();
|
||||
});
|
||||
|
||||
it('formats select fields', () => {
|
||||
const field: FieldDef = {
|
||||
label: 'Citizenship',
|
||||
type: 'select',
|
||||
options: [{ value: 'biesel', label: 'Republic of Biesel' }]
|
||||
};
|
||||
expect(formatFieldOutput(field, 'biesel')).toBe('Citizenship: Republic of Biesel');
|
||||
});
|
||||
|
||||
it('formats date fields', () => {
|
||||
const field: FieldDef = { label: 'Date of Birth', type: 'date' };
|
||||
expect(formatFieldOutput(field, 'March 15th, 2438')).toBe('Date of Birth: March 15th, 2438');
|
||||
});
|
||||
|
||||
it('formats number fields', () => {
|
||||
const field: FieldDef = { label: 'Age', type: 'number' };
|
||||
expect(formatFieldOutput(field, 30)).toBe('Age: 30');
|
||||
});
|
||||
|
||||
it('formats citizenship type', () => {
|
||||
const field: FieldDef = { label: 'Citizenship', type: 'citizenship' };
|
||||
expect(formatFieldOutput(field, 'Republic of Biesel')).toBe('Citizenship: Republic of Biesel');
|
||||
});
|
||||
|
||||
it('formats multi-select as comma list', () => {
|
||||
const field: FieldDef = {
|
||||
label: 'Other Skills',
|
||||
type: 'multi-select',
|
||||
options: [
|
||||
{ value: 'engineering', label: 'Engineering' },
|
||||
{ value: 'medical', label: 'Medical' }
|
||||
]
|
||||
};
|
||||
expect(formatFieldOutput(field, ['engineering', 'medical'])).toBe(
|
||||
'Other Skills: Engineering, Medical'
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
const testTemplate: Template = {
|
||||
id: 'test',
|
||||
name: 'Test Template',
|
||||
description: '',
|
||||
schemaVersion: 1,
|
||||
records: [
|
||||
{
|
||||
type: 'public',
|
||||
fields: [
|
||||
{ label: 'Name', type: 'text' },
|
||||
{ label: 'Species', type: 'species' },
|
||||
{ label: 'Pronouns', type: 'text' }
|
||||
]
|
||||
},
|
||||
{
|
||||
type: 'employment',
|
||||
preamble: 'This information has been verified by employment agents.',
|
||||
fields: [
|
||||
{ label: 'Employment History', type: 'list' },
|
||||
{ label: 'Formal Education', type: 'list' }
|
||||
]
|
||||
},
|
||||
{
|
||||
type: 'medical',
|
||||
preamble: 'Protected by doctor-patient confidentiality.',
|
||||
fields: [
|
||||
{
|
||||
label: 'Opt-Outs',
|
||||
type: 'checkbox',
|
||||
options: [{ value: 'no-borg', label: 'Do Not Borgify' }]
|
||||
},
|
||||
{ label: 'Allergies', type: 'list' }
|
||||
]
|
||||
},
|
||||
{
|
||||
type: 'security',
|
||||
preamble: 'This information has been verified by employment agents.',
|
||||
fields: [
|
||||
{ label: 'Attitude Towards SCC', type: 'textarea' },
|
||||
{ label: 'Arrest History', type: 'list' }
|
||||
]
|
||||
}
|
||||
]
|
||||
};
|
||||
|
||||
describe('generateRecord', () => {
|
||||
it('includes public header and employment body', () => {
|
||||
const data = {
|
||||
name: 'Yury Zakharov',
|
||||
species: 'tajara',
|
||||
'employment-history': 'Janitor'
|
||||
};
|
||||
const out = generateRecord(testTemplate, data, 'employment', stubSpecies);
|
||||
expect(out).toContain('/// PUBLIC RECORD ///');
|
||||
expect(out).toContain('Name: Yury Zakharov');
|
||||
expect(out).toContain('Species: Tajara');
|
||||
expect(out).toContain('/// EMPLOYMENT RECORD ///');
|
||||
expect(out).toContain('This information has been verified by employment agents.');
|
||||
expect(out).toContain(' - Janitor');
|
||||
expect(out).toContain('LAST UPDATED:');
|
||||
});
|
||||
|
||||
it('shows NO RECORD FOUND when body is empty', () => {
|
||||
const data = { name: 'Yury Zakharov' };
|
||||
const out = generateRecord(testTemplate, data, 'medical', stubSpecies);
|
||||
expect(out).toContain('/// NO MEDICAL RECORD FOUND ///');
|
||||
});
|
||||
|
||||
it('includes preamble in medical record', () => {
|
||||
const data = {
|
||||
name: 'Yury Zakharov',
|
||||
allergies: 'Peanuts'
|
||||
};
|
||||
const out = generateRecord(testTemplate, data, 'medical', stubSpecies);
|
||||
expect(out).toContain('Protected by doctor-patient confidentiality.');
|
||||
expect(out).toContain(' - Peanuts');
|
||||
});
|
||||
|
||||
it('includes preamble in security record', () => {
|
||||
const data = {
|
||||
name: 'Yury Zakharov',
|
||||
'attitude-towards-scc': 'Loyal employee'
|
||||
};
|
||||
const out = generateRecord(testTemplate, data, 'security', stubSpecies);
|
||||
expect(out).toContain('/// SECURITY RECORD ///');
|
||||
expect(out).toContain('This information has been verified by employment agents.');
|
||||
expect(out).toContain('Loyal employee');
|
||||
});
|
||||
});
|
||||
164
src/lib/output.ts
Normal file
164
src/lib/output.ts
Normal file
|
|
@ -0,0 +1,164 @@
|
|||
import type { FieldDef, Template } from './types';
|
||||
import type { SpeciesData } from './data/types';
|
||||
import { cmToFeetInches, kgToLb } from './utils/conversions';
|
||||
import { formatICDate } from './utils/dates';
|
||||
import { slugify } from './utils/slugify';
|
||||
|
||||
export function formatFieldOutput(
|
||||
field: FieldDef,
|
||||
value: unknown,
|
||||
speciesData?: SpeciesData[],
|
||||
currentSpecies?: string
|
||||
): string | null {
|
||||
switch (field.type) {
|
||||
case 'name':
|
||||
case 'text':
|
||||
case 'date':
|
||||
case 'citizenship':
|
||||
return value ? `${field.label}: ${value}` : null;
|
||||
|
||||
case 'textarea':
|
||||
return value ? `${field.label}:\n${value}` : null;
|
||||
|
||||
case 'list': {
|
||||
const lines = splitLines(value as string);
|
||||
return lines.length ? `${field.label}:\n${formatBullets(lines)}` : null;
|
||||
}
|
||||
|
||||
case 'number':
|
||||
return value != null && value !== 0 ? `${field.label}: ${value}` : null;
|
||||
|
||||
case 'height':
|
||||
return value ? `${field.label}: ${value} cm (${cmToFeetInches(value as number)})` : null;
|
||||
|
||||
case 'weight': {
|
||||
if (!value) return null;
|
||||
const lb = Math.round(kgToLb(value as number));
|
||||
return `${field.label}: ${value} kg (${lb} lb)`;
|
||||
}
|
||||
|
||||
case 'species': {
|
||||
if (!value || !speciesData) return null;
|
||||
const sp = speciesData.find((s) => s.id === value);
|
||||
return sp ? `${field.label}: ${sp.name}` : `${field.label}: ${value}`;
|
||||
}
|
||||
|
||||
case 'subspecies': {
|
||||
if (!value || !speciesData || !currentSpecies) return null;
|
||||
const sp = speciesData.find((s) => s.id === currentSpecies);
|
||||
if (!sp) return null;
|
||||
const sub = sp.subspecies.find((s) => s.id === value);
|
||||
return sub ? `${sp.subspeciesLabel}: ${sub.name}` : null;
|
||||
}
|
||||
|
||||
case 'languages': {
|
||||
const arr = value as string[] | undefined;
|
||||
return arr?.length ? `${field.label}: ${arr.join(', ')}` : null;
|
||||
}
|
||||
|
||||
case 'checkbox': {
|
||||
const selected = value as string[] | undefined;
|
||||
if (!selected?.length) return null;
|
||||
const labels = selected
|
||||
.map((v) => field.options.find((o) => o.value === v)?.label ?? v)
|
||||
return `${field.label}:\n${formatBullets(labels)}`;
|
||||
}
|
||||
|
||||
case 'select': {
|
||||
if (!value) return null;
|
||||
const opt = field.options.find((o) => o.value === value);
|
||||
return `${field.label}: ${opt?.label ?? value}`;
|
||||
}
|
||||
|
||||
case 'multi-select': {
|
||||
const vals = value as string[] | undefined;
|
||||
if (!vals?.length) return null;
|
||||
const labels = vals.map((v) => field.options.find((o) => o.value === v)?.label ?? v);
|
||||
return `${field.label}: ${labels.join(', ')}`;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export function generateRecord(
|
||||
template: Template,
|
||||
data: Record<string, unknown>,
|
||||
recordType: string,
|
||||
speciesData?: SpeciesData[]
|
||||
): string {
|
||||
const publicRecord = template.records.find((r) => r.type === 'public');
|
||||
const targetRecord = template.records.find((r) => r.type === recordType);
|
||||
const currentSpecies = data['species'] as string | undefined;
|
||||
|
||||
const parts: string[] = [];
|
||||
|
||||
// Public section
|
||||
if (publicRecord) {
|
||||
const publicLines = renderFields(publicRecord.fields, data, speciesData, currentSpecies);
|
||||
if (publicLines.length) {
|
||||
parts.push('/// PUBLIC RECORD ///');
|
||||
parts.push(publicLines.join('\n'));
|
||||
}
|
||||
}
|
||||
|
||||
// Target record section
|
||||
if (targetRecord) {
|
||||
const bodyLines = renderFields(targetRecord.fields, data, speciesData, currentSpecies);
|
||||
const typeLabel = recordType.toUpperCase();
|
||||
|
||||
if (!bodyLines.length) {
|
||||
parts.push(`/// NO ${typeLabel} RECORD FOUND ///`);
|
||||
} else {
|
||||
parts.push(`/// ${typeLabel} RECORD ///`);
|
||||
if (targetRecord.preamble) {
|
||||
parts.push(targetRecord.preamble);
|
||||
}
|
||||
parts.push(bodyLines.join('\n\n'));
|
||||
}
|
||||
}
|
||||
|
||||
parts.push(`LAST UPDATED: ${formatICDate(new Date())}`);
|
||||
return parts.join('\n\n');
|
||||
}
|
||||
|
||||
function renderFields(
|
||||
fields: FieldDef[],
|
||||
data: Record<string, unknown>,
|
||||
speciesData?: SpeciesData[],
|
||||
currentSpecies?: string
|
||||
): string[] {
|
||||
// Split fields into groups by separator boundaries
|
||||
const groups: FieldDef[][] = [[]];
|
||||
for (const field of fields) {
|
||||
if (field.type === 'separator') {
|
||||
groups.push([]);
|
||||
} else {
|
||||
groups[groups.length - 1].push(field);
|
||||
}
|
||||
}
|
||||
|
||||
const rendered = groups.map((group) => {
|
||||
const lines: string[] = [];
|
||||
for (const field of group) {
|
||||
const out = formatFieldOutput(field, data[slugify(field.label)], speciesData, currentSpecies);
|
||||
if (out) lines.push(out);
|
||||
}
|
||||
return lines;
|
||||
});
|
||||
|
||||
const result: string[] = [];
|
||||
for (const lines of rendered) {
|
||||
if (!lines.length) continue;
|
||||
if (result.length) result.push('');
|
||||
result.push(...lines);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
function splitLines(text: string | undefined): string[] {
|
||||
if (!text) return [];
|
||||
return text.split('\n').map((l) => l.trim()).filter(Boolean);
|
||||
}
|
||||
|
||||
function formatBullets(items: string[]): string {
|
||||
return items.map((item) => ` - ${item}`).join('\n');
|
||||
}
|
||||
12
src/lib/presets.ts
Normal file
12
src/lib/presets.ts
Normal file
|
|
@ -0,0 +1,12 @@
|
|||
import { parseTemplate } from './data/parse';
|
||||
|
||||
const templateModules = import.meta.glob('../../data/templates/*.xml', {
|
||||
query: '?raw',
|
||||
import: 'default',
|
||||
eager: true
|
||||
});
|
||||
|
||||
export const presets = Object.entries(templateModules).map(([path, xml]) => {
|
||||
const filename = path.split('/').pop()!.replace('.xml', '');
|
||||
return parseTemplate(xml as string, `preset:${filename}`);
|
||||
});
|
||||
198
src/lib/schema.test.ts
Normal file
198
src/lib/schema.test.ts
Normal file
|
|
@ -0,0 +1,198 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import { buildCharacterSchema } from './schema';
|
||||
import type { Template } from './types';
|
||||
|
||||
function makeTemplate(fields: any[]): Template {
|
||||
return {
|
||||
id: 'test',
|
||||
name: 'Test',
|
||||
description: '',
|
||||
schemaVersion: 1,
|
||||
records: [{ type: 'public', fields }]
|
||||
};
|
||||
}
|
||||
|
||||
describe('buildCharacterSchema', () => {
|
||||
it('validates text fields as optional strings', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Pronouns', type: 'text' }])
|
||||
);
|
||||
expect(schema.parse({})).toEqual({});
|
||||
expect(schema.parse({ pronouns: 'She/her' })).toEqual({ pronouns: 'She/her' });
|
||||
});
|
||||
|
||||
it('validates textarea fields as optional strings', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{ label: 'Distinguishing Features', type: 'textarea' }
|
||||
])
|
||||
);
|
||||
expect(schema.parse({ 'distinguishing-features': 'Scar across left eye' })).toEqual({
|
||||
'distinguishing-features': 'Scar across left eye'
|
||||
});
|
||||
expect(schema.parse({})).toEqual({});
|
||||
});
|
||||
|
||||
it('validates list fields as optional strings', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{ label: 'Employment History', type: 'list' }
|
||||
])
|
||||
);
|
||||
expect(schema.parse({ 'employment-history': 'NanoTrasen Intern\nShaft Miner' })).toEqual({
|
||||
'employment-history': 'NanoTrasen Intern\nShaft Miner'
|
||||
});
|
||||
expect(schema.parse({})).toEqual({});
|
||||
});
|
||||
|
||||
it('validates date fields as optional strings', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{ label: 'Date of Birth', type: 'date', placeholder: 'March 15th, 2438' }
|
||||
])
|
||||
);
|
||||
expect(schema.parse({ 'date-of-birth': 'March 15th, 2438' })).toEqual({
|
||||
'date-of-birth': 'March 15th, 2438'
|
||||
});
|
||||
});
|
||||
|
||||
it('validates select fields as optional strings', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{
|
||||
label: 'Citizenship',
|
||||
type: 'select',
|
||||
options: [{ value: 'biesel', label: 'Republic of Biesel' }]
|
||||
}
|
||||
])
|
||||
);
|
||||
expect(schema.parse({ citizenship: 'biesel' })).toEqual({ citizenship: 'biesel' });
|
||||
});
|
||||
|
||||
it('validates number fields as optional numbers', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Age', type: 'number', min: 0, max: 999 }])
|
||||
);
|
||||
expect(schema.parse({ age: 30 })).toEqual({ age: 30 });
|
||||
expect(schema.parse({})).toEqual({});
|
||||
});
|
||||
|
||||
it('validates height as optional number', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Height', type: 'height' }])
|
||||
);
|
||||
expect(schema.parse({ height: 180 })).toEqual({ height: 180 });
|
||||
});
|
||||
|
||||
it('validates weight as optional number', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Weight', type: 'weight' }])
|
||||
);
|
||||
expect(schema.parse({ weight: 75 })).toEqual({ weight: 75 });
|
||||
});
|
||||
|
||||
it('validates name as optional string', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Name', type: 'text' }])
|
||||
);
|
||||
expect(schema.parse({ name: 'Ka\'Akaix\'Lak Zo\'ra' })).toEqual({
|
||||
name: 'Ka\'Akaix\'Lak Zo\'ra'
|
||||
});
|
||||
expect(schema.parse({})).toEqual({});
|
||||
});
|
||||
|
||||
it('validates multi-select as optional string array', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{
|
||||
label: 'Spoken Languages',
|
||||
type: 'multi-select',
|
||||
options: [
|
||||
{ value: 'tau-ceti-basic', label: 'Tau Ceti Basic' },
|
||||
{ value: 'sol-common', label: 'Sol Common' }
|
||||
]
|
||||
}
|
||||
])
|
||||
);
|
||||
expect(schema.parse({ 'spoken-languages': ['tau-ceti-basic', 'sol-common'] })).toEqual({
|
||||
'spoken-languages': ['tau-ceti-basic', 'sol-common']
|
||||
});
|
||||
expect(schema.parse({})).toEqual({});
|
||||
});
|
||||
|
||||
it('validates checkbox as optional string array', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{
|
||||
label: 'Opt-Outs',
|
||||
type: 'checkbox',
|
||||
options: [
|
||||
{ value: 'no-borg', label: 'Do Not Borgify' },
|
||||
{ value: 'no-revive', label: 'Do Not Revive' },
|
||||
{ value: 'no-prosthetic', label: 'Do Not Prostheticize' }
|
||||
]
|
||||
}
|
||||
])
|
||||
);
|
||||
expect(schema.parse({ 'opt-outs': ['no-borg', 'no-revive'] })).toEqual({
|
||||
'opt-outs': ['no-borg', 'no-revive']
|
||||
});
|
||||
});
|
||||
|
||||
it('validates languages as optional string array', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{ label: 'Spoken Languages', type: 'languages' }
|
||||
])
|
||||
);
|
||||
expect(
|
||||
schema.parse({ 'spoken-languages': ['Tau Ceti Basic', 'Siik\'maas'] })
|
||||
).toEqual({
|
||||
'spoken-languages': ['Tau Ceti Basic', 'Siik\'maas']
|
||||
});
|
||||
});
|
||||
|
||||
it('validates species as optional string', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Species', type: 'species' }])
|
||||
);
|
||||
expect(schema.parse({ species: 'tajara' })).toEqual({ species: 'tajara' });
|
||||
});
|
||||
|
||||
it('validates subspecies as optional string', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Subspecies', type: 'subspecies' }])
|
||||
);
|
||||
expect(schema.parse({ subspecies: 'zhan-khazan' })).toEqual({
|
||||
subspecies: 'zhan-khazan'
|
||||
});
|
||||
});
|
||||
|
||||
it('validates citizenship type as optional string', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Citizenship', type: 'citizenship' }])
|
||||
);
|
||||
expect(schema.parse({ citizenship: 'sol-alliance' })).toEqual({
|
||||
citizenship: 'sol-alliance'
|
||||
});
|
||||
});
|
||||
|
||||
it('allows all fields to be missing', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([
|
||||
{ label: 'Name', type: 'text' },
|
||||
{ label: 'Height', type: 'height' },
|
||||
{ label: 'Spoken Languages', type: 'languages' },
|
||||
{ label: 'Skin Color', type: 'text' }
|
||||
])
|
||||
);
|
||||
expect(schema.parse({})).toEqual({});
|
||||
});
|
||||
|
||||
it('rejects wrong types', () => {
|
||||
const schema = buildCharacterSchema(
|
||||
makeTemplate([{ label: 'Height', type: 'height' }])
|
||||
);
|
||||
expect(() => schema.parse({ height: 'tall' })).toThrow();
|
||||
});
|
||||
});
|
||||
39
src/lib/schema.ts
Normal file
39
src/lib/schema.ts
Normal file
|
|
@ -0,0 +1,39 @@
|
|||
import { z } from 'zod';
|
||||
import type { FieldDef, Template } from './types';
|
||||
import { slugify } from './utils/slugify';
|
||||
|
||||
function zodForField(field: FieldDef): z.ZodTypeAny {
|
||||
switch (field.type) {
|
||||
case 'name':
|
||||
case 'text':
|
||||
case 'textarea':
|
||||
case 'list':
|
||||
case 'date':
|
||||
case 'select':
|
||||
case 'species':
|
||||
case 'subspecies':
|
||||
case 'citizenship':
|
||||
return z.string().optional();
|
||||
|
||||
case 'number':
|
||||
case 'height':
|
||||
case 'weight':
|
||||
return z.number().optional();
|
||||
|
||||
case 'multi-select':
|
||||
case 'checkbox':
|
||||
case 'languages':
|
||||
return z.array(z.string()).optional();
|
||||
}
|
||||
}
|
||||
|
||||
export function buildCharacterSchema(template: Template): z.ZodObject<Record<string, z.ZodTypeAny>> {
|
||||
const shape: Record<string, z.ZodTypeAny> = {};
|
||||
for (const record of template.records) {
|
||||
for (const field of record.fields) {
|
||||
if (field.type === 'separator') continue;
|
||||
shape[slugify(field.label)] = zodForField(field);
|
||||
}
|
||||
}
|
||||
return z.object(shape).partial();
|
||||
}
|
||||
119
src/lib/sharing.test.ts
Normal file
119
src/lib/sharing.test.ts
Normal file
|
|
@ -0,0 +1,119 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import {
|
||||
encodeCharacterURL,
|
||||
decodeCharacterURL,
|
||||
encodeTemplateURL,
|
||||
decodeTemplateURL
|
||||
} from './sharing';
|
||||
import { presets } from './presets';
|
||||
import type { Character, Template } from './types';
|
||||
|
||||
const standardPreset = presets.find((p) => p.id === 'preset:standard')!;
|
||||
|
||||
const testCharacter: Character = {
|
||||
id: 'abc-123',
|
||||
template: standardPreset,
|
||||
data: {
|
||||
name: 'Yury Zakharov',
|
||||
species: 'human',
|
||||
'employment-history': 'Shaft Miner'
|
||||
}
|
||||
};
|
||||
|
||||
const customTemplate: Template = {
|
||||
id: 'custom:test',
|
||||
name: 'Custom',
|
||||
description: 'A custom template.',
|
||||
schemaVersion: 1,
|
||||
records: [
|
||||
{
|
||||
type: 'public',
|
||||
fields: [
|
||||
{ label: 'Name', type: 'text' },
|
||||
{ label: 'Species', type: 'species' }
|
||||
]
|
||||
}
|
||||
]
|
||||
};
|
||||
|
||||
describe('character URL encoding', () => {
|
||||
it('round-trips preset character data', () => {
|
||||
const encoded = encodeCharacterURL(testCharacter);
|
||||
const decoded = decodeCharacterURL(encoded);
|
||||
expect(decoded.data).toEqual(testCharacter.data);
|
||||
expect(decoded.template.name).toBe('General');
|
||||
});
|
||||
|
||||
it('uses short encoding for preset templates', () => {
|
||||
const encoded = encodeCharacterURL(testCharacter);
|
||||
const customChar = { ...testCharacter, template: customTemplate };
|
||||
const customEncoded = encodeCharacterURL(customChar);
|
||||
expect(encoded.length).toBeLessThan(customEncoded.length);
|
||||
});
|
||||
|
||||
it('round-trips custom template character', () => {
|
||||
const char: Character = { ...testCharacter, template: customTemplate };
|
||||
const encoded = encodeCharacterURL(char);
|
||||
const decoded = decodeCharacterURL(encoded);
|
||||
expect(decoded.data).toEqual(testCharacter.data);
|
||||
expect(decoded.template.name).toBe('Custom');
|
||||
});
|
||||
|
||||
it('starts with c1. prefix', () => {
|
||||
const encoded = encodeCharacterURL(testCharacter);
|
||||
expect(encoded.startsWith('c1.')).toBe(true);
|
||||
});
|
||||
|
||||
it('prunes empty values from data', () => {
|
||||
const char: Character = {
|
||||
...testCharacter,
|
||||
data: { name: 'Yury Zakharov', species: '', 'hair-color': '' }
|
||||
};
|
||||
const encoded = encodeCharacterURL(char);
|
||||
const decoded = decodeCharacterURL(encoded);
|
||||
expect(decoded.data).toEqual({ name: 'Yury Zakharov' });
|
||||
});
|
||||
});
|
||||
|
||||
describe('template URL encoding', () => {
|
||||
it('round-trips template structure', () => {
|
||||
const encoded = encodeTemplateURL(customTemplate);
|
||||
const decoded = decodeTemplateURL(encoded);
|
||||
expect(decoded.name).toBe('Custom');
|
||||
expect(decoded.records).toEqual(customTemplate.records);
|
||||
});
|
||||
|
||||
it('starts with t1. prefix', () => {
|
||||
const encoded = encodeTemplateURL(customTemplate);
|
||||
expect(encoded.startsWith('t1.')).toBe(true);
|
||||
});
|
||||
|
||||
it('strips id', () => {
|
||||
const encoded = encodeTemplateURL(customTemplate);
|
||||
const decoded = decodeTemplateURL(encoded);
|
||||
expect(decoded).not.toHaveProperty('id');
|
||||
});
|
||||
});
|
||||
|
||||
describe('unicode support', () => {
|
||||
it('round-trips unicode content', () => {
|
||||
const char: Character = {
|
||||
...testCharacter,
|
||||
data: { name: "Ka'Akaix'Lak Zo'ra", species: 'vaurca' }
|
||||
};
|
||||
const encoded = encodeCharacterURL(char);
|
||||
const decoded = decodeCharacterURL(encoded);
|
||||
expect(decoded.data.name).toBe("Ka'Akaix'Lak Zo'ra");
|
||||
});
|
||||
});
|
||||
|
||||
describe('error handling', () => {
|
||||
it('throws on invalid character input', () => {
|
||||
expect(() => decodeCharacterURL('c1.invaliddata!!!')).toThrow();
|
||||
});
|
||||
|
||||
it('throws on wrong prefix', () => {
|
||||
const encoded = encodeCharacterURL(testCharacter);
|
||||
expect(() => decodeTemplateURL(encoded)).toThrow();
|
||||
});
|
||||
});
|
||||
75
src/lib/sharing.ts
Normal file
75
src/lib/sharing.ts
Normal file
|
|
@ -0,0 +1,75 @@
|
|||
import pako from 'pako';
|
||||
import type { Character, Template } from './types';
|
||||
import { presets } from './presets';
|
||||
|
||||
function toBase64url(bytes: Uint8Array): string {
|
||||
let binary = '';
|
||||
for (const b of bytes) binary += String.fromCharCode(b);
|
||||
return btoa(binary).replace(/\+/g, '-').replace(/\//g, '_').replace(/=+$/, '');
|
||||
}
|
||||
|
||||
function fromBase64url(str: string): Uint8Array {
|
||||
const padded = str.replace(/-/g, '+').replace(/_/g, '/');
|
||||
const binary = atob(padded);
|
||||
const bytes = new Uint8Array(binary.length);
|
||||
for (let i = 0; i < binary.length; i++) bytes[i] = binary.charCodeAt(i);
|
||||
return bytes;
|
||||
}
|
||||
|
||||
export function pruneEmpty(data: Record<string, unknown>): Record<string, unknown> {
|
||||
const out: Record<string, unknown> = {};
|
||||
for (const [k, v] of Object.entries(data)) {
|
||||
if (v === '' || v === undefined || v === null) continue;
|
||||
if (Array.isArray(v) && v.length === 0) continue;
|
||||
out[k] = v;
|
||||
}
|
||||
return out;
|
||||
}
|
||||
|
||||
export function encodeCharacterURL(char: Character): string {
|
||||
const isPreset = char.template.id.startsWith('preset:');
|
||||
const payload: any = {
|
||||
data: pruneEmpty(char.data)
|
||||
};
|
||||
if (isPreset) {
|
||||
payload.templateId = char.template.id;
|
||||
} else {
|
||||
payload.template = stripId(char.template);
|
||||
}
|
||||
const json = JSON.stringify(payload);
|
||||
const compressed = pako.deflate(new TextEncoder().encode(json));
|
||||
return 'c1.' + toBase64url(compressed);
|
||||
}
|
||||
|
||||
export function decodeCharacterURL(encoded: string): { template: Template | Omit<Template, 'id'>; data: Record<string, unknown> } {
|
||||
if (!encoded.startsWith('c1.')) throw new Error('Invalid character URL prefix');
|
||||
const bytes = fromBase64url(encoded.slice(3));
|
||||
const json = new TextDecoder().decode(pako.inflate(bytes));
|
||||
const payload = JSON.parse(json);
|
||||
|
||||
if (payload.templateId) {
|
||||
const preset = presets.find((p) => p.id === payload.templateId);
|
||||
if (!preset) throw new Error(`Unknown template: ${payload.templateId}`);
|
||||
return { template: preset, data: payload.data };
|
||||
}
|
||||
return { template: payload.template, data: payload.data };
|
||||
}
|
||||
|
||||
export function encodeTemplateURL(template: Template): string {
|
||||
const payload = stripId(template);
|
||||
const json = JSON.stringify(payload);
|
||||
const compressed = pako.deflate(new TextEncoder().encode(json));
|
||||
return 't1.' + toBase64url(compressed);
|
||||
}
|
||||
|
||||
export function decodeTemplateURL(encoded: string): Omit<Template, 'id'> {
|
||||
if (!encoded.startsWith('t1.')) throw new Error('Invalid template URL prefix');
|
||||
const bytes = fromBase64url(encoded.slice(3));
|
||||
const json = new TextDecoder().decode(pako.inflate(bytes));
|
||||
return JSON.parse(json);
|
||||
}
|
||||
|
||||
function stripId(obj: Record<string, any>): Record<string, any> {
|
||||
const { id, ...rest } = obj;
|
||||
return rest;
|
||||
}
|
||||
147
src/lib/state.svelte.ts
Normal file
147
src/lib/state.svelte.ts
Normal file
|
|
@ -0,0 +1,147 @@
|
|||
import { getAllCharacters, saveCharacter, deleteCharacter } from './storage';
|
||||
import { isBlankCharacter } from './utils/blank';
|
||||
import { slugify } from './utils/slugify';
|
||||
import type { Character, Template } from './types';
|
||||
|
||||
let characters = $state<Character[]>([]);
|
||||
let activeId = $state<string | null>(null);
|
||||
let saveStatus = $state<'idle' | 'saving' | 'saved'>('idle');
|
||||
let saveTimer: ReturnType<typeof setTimeout> | null = null;
|
||||
let statusTimer: ReturnType<typeof setTimeout> | null = null;
|
||||
|
||||
const SINGLETON_TYPES = new Set([
|
||||
'name', 'species', 'subspecies', 'citizenship', 'languages', 'height', 'weight'
|
||||
]);
|
||||
|
||||
function allFields(template: Template) {
|
||||
return template.records.flatMap((r) => r.fields).filter((f) => f.type !== 'separator');
|
||||
}
|
||||
|
||||
function migrateData(char: Character, preset: Template) {
|
||||
for (const record of preset.records) {
|
||||
for (const field of record.fields) {
|
||||
if (!field.from) continue;
|
||||
const newKey = slugify(field.label);
|
||||
if (char.data[newKey] !== undefined) continue;
|
||||
const oldNames = field.from.split(',').map((s) => s.trim());
|
||||
for (const oldName of oldNames) {
|
||||
const oldKey = slugify(oldName);
|
||||
if (char.data[oldKey] !== undefined) {
|
||||
char.data[newKey] = char.data[oldKey];
|
||||
delete char.data[oldKey];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const oldByType = new Map<string, string>();
|
||||
for (const f of allFields(char.template)) {
|
||||
if (SINGLETON_TYPES.has(f.type)) {
|
||||
oldByType.set(f.type, slugify(f.label));
|
||||
}
|
||||
}
|
||||
for (const f of allFields(preset)) {
|
||||
if (!SINGLETON_TYPES.has(f.type)) continue;
|
||||
const newKey = slugify(f.label);
|
||||
if (char.data[newKey] !== undefined) continue;
|
||||
const oldKey = oldByType.get(f.type);
|
||||
if (oldKey && oldKey !== newKey && char.data[oldKey] !== undefined) {
|
||||
char.data[newKey] = char.data[oldKey];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export const roster = {
|
||||
get characters() { return characters; },
|
||||
get active() { return characters.find((c) => c.id === activeId) ?? null; },
|
||||
get saveStatus() { return saveStatus; },
|
||||
|
||||
async migrateToPreset(char: Character, preset: Template) {
|
||||
migrateData(char, preset);
|
||||
if (preset.species?.length === 1) {
|
||||
char.data[slugify('Species')] = preset.species[0];
|
||||
}
|
||||
char.template = $state.snapshot(preset);
|
||||
await saveCharacter($state.snapshot(char));
|
||||
},
|
||||
|
||||
async load() {
|
||||
const all = await getAllCharacters();
|
||||
const kept: Character[] = [];
|
||||
|
||||
for (const char of all) {
|
||||
if (isBlankCharacter(char)) {
|
||||
await deleteCharacter(char.id);
|
||||
} else {
|
||||
kept.push(char);
|
||||
}
|
||||
}
|
||||
|
||||
characters = kept;
|
||||
|
||||
const stored = localStorage.getItem('activeCharacterId');
|
||||
if (stored && characters.some((c) => c.id === stored)) {
|
||||
activeId = stored;
|
||||
} else if (characters.length) {
|
||||
activeId = characters[0].id;
|
||||
}
|
||||
},
|
||||
|
||||
async create(template: Template, data: Record<string, unknown> = {}) {
|
||||
const initial: Record<string, unknown> = { ...data };
|
||||
if (template.species?.length === 1) {
|
||||
initial[slugify('Species')] ??= template.species[0];
|
||||
}
|
||||
const char: Character = {
|
||||
id: crypto.randomUUID(),
|
||||
template: $state.snapshot(template),
|
||||
data: initial
|
||||
};
|
||||
characters.push(char);
|
||||
activeId = char.id;
|
||||
localStorage.setItem('activeCharacterId', char.id);
|
||||
await saveCharacter($state.snapshot(char));
|
||||
return char;
|
||||
},
|
||||
|
||||
async remove(id: string) {
|
||||
characters = characters.filter((c) => c.id !== id);
|
||||
await deleteCharacter(id);
|
||||
if (activeId === id) {
|
||||
activeId = characters[0]?.id ?? null;
|
||||
if (activeId) localStorage.setItem('activeCharacterId', activeId);
|
||||
else localStorage.removeItem('activeCharacterId');
|
||||
}
|
||||
},
|
||||
|
||||
async duplicate(id: string) {
|
||||
const source = characters.find((c) => c.id === id);
|
||||
if (!source) return;
|
||||
const copy: Character = {
|
||||
id: crypto.randomUUID(),
|
||||
template: $state.snapshot(source.template),
|
||||
data: $state.snapshot(source.data)
|
||||
};
|
||||
characters.push(copy);
|
||||
activeId = copy.id;
|
||||
localStorage.setItem('activeCharacterId', copy.id);
|
||||
await saveCharacter($state.snapshot(copy));
|
||||
},
|
||||
|
||||
setActive(id: string) {
|
||||
activeId = id;
|
||||
localStorage.setItem('activeCharacterId', id);
|
||||
},
|
||||
|
||||
scheduleSave(char: Character) {
|
||||
if (saveTimer) clearTimeout(saveTimer);
|
||||
saveTimer = setTimeout(async () => {
|
||||
saveStatus = 'saving';
|
||||
await saveCharacter($state.snapshot(char));
|
||||
saveStatus = 'saved';
|
||||
if (statusTimer) clearTimeout(statusTimer);
|
||||
statusTimer = setTimeout(() => { saveStatus = 'idle'; }, 1500);
|
||||
}, 300);
|
||||
}
|
||||
};
|
||||
46
src/lib/storage.ts
Normal file
46
src/lib/storage.ts
Normal file
|
|
@ -0,0 +1,46 @@
|
|||
import { openDB, type DBSchema } from 'idb';
|
||||
import type { Character, Template } from './types';
|
||||
|
||||
interface RecordsDB extends DBSchema {
|
||||
characters: { key: string; value: Character };
|
||||
templates: { key: string; value: Template };
|
||||
}
|
||||
|
||||
const dbPromise = openDB<RecordsDB>('aurora-records', 1, {
|
||||
upgrade(db) {
|
||||
db.createObjectStore('characters', { keyPath: 'id' });
|
||||
db.createObjectStore('templates', { keyPath: 'id' });
|
||||
}
|
||||
});
|
||||
|
||||
export async function getAllCharacters() {
|
||||
return (await dbPromise).getAll('characters');
|
||||
}
|
||||
|
||||
export async function getCharacter(id: string) {
|
||||
return (await dbPromise).get('characters', id);
|
||||
}
|
||||
|
||||
export async function saveCharacter(char: Character) {
|
||||
await (await dbPromise).put('characters', char);
|
||||
}
|
||||
|
||||
export async function deleteCharacter(id: string) {
|
||||
await (await dbPromise).delete('characters', id);
|
||||
}
|
||||
|
||||
export async function getAllTemplates() {
|
||||
return (await dbPromise).getAll('templates');
|
||||
}
|
||||
|
||||
export async function getTemplate(id: string) {
|
||||
return (await dbPromise).get('templates', id);
|
||||
}
|
||||
|
||||
export async function saveTemplate(tmpl: Template) {
|
||||
await (await dbPromise).put('templates', tmpl);
|
||||
}
|
||||
|
||||
export async function deleteTemplate(id: string) {
|
||||
await (await dbPromise).delete('templates', id);
|
||||
}
|
||||
25
src/lib/theme.svelte.ts
Normal file
25
src/lib/theme.svelte.ts
Normal file
|
|
@ -0,0 +1,25 @@
|
|||
let dark = $state(false);
|
||||
|
||||
export const theme = {
|
||||
get dark() { return dark; },
|
||||
|
||||
init() {
|
||||
const stored = localStorage.getItem('theme');
|
||||
if (stored) {
|
||||
dark = stored === 'dark';
|
||||
} else {
|
||||
dark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
||||
}
|
||||
apply();
|
||||
},
|
||||
|
||||
toggle() {
|
||||
dark = !dark;
|
||||
localStorage.setItem('theme', dark ? 'dark' : 'light');
|
||||
apply();
|
||||
}
|
||||
};
|
||||
|
||||
function apply() {
|
||||
document.documentElement.classList.toggle('dark', dark);
|
||||
}
|
||||
125
src/lib/types.ts
Normal file
125
src/lib/types.ts
Normal file
|
|
@ -0,0 +1,125 @@
|
|||
export interface SelectOption {
|
||||
value: string;
|
||||
label: string;
|
||||
}
|
||||
|
||||
export interface BaseFieldDef {
|
||||
label: string;
|
||||
required?: boolean;
|
||||
from?: string;
|
||||
}
|
||||
|
||||
export interface NameField extends BaseFieldDef {
|
||||
type: 'name';
|
||||
placeholder?: string;
|
||||
}
|
||||
|
||||
export interface TextField extends BaseFieldDef {
|
||||
type: 'text';
|
||||
placeholder?: string;
|
||||
}
|
||||
|
||||
export interface TextareaField extends BaseFieldDef {
|
||||
type: 'textarea';
|
||||
placeholder?: string;
|
||||
}
|
||||
|
||||
export interface ListField extends BaseFieldDef {
|
||||
type: 'list';
|
||||
}
|
||||
|
||||
export interface NumberField extends BaseFieldDef {
|
||||
type: 'number';
|
||||
min?: number;
|
||||
max?: number;
|
||||
unit?: string;
|
||||
}
|
||||
|
||||
export interface SelectField extends BaseFieldDef {
|
||||
type: 'select';
|
||||
options: SelectOption[];
|
||||
}
|
||||
|
||||
export interface MultiSelectField extends BaseFieldDef {
|
||||
type: 'multi-select';
|
||||
options: SelectOption[];
|
||||
}
|
||||
|
||||
export interface CheckboxField extends BaseFieldDef {
|
||||
type: 'checkbox';
|
||||
options: SelectOption[];
|
||||
}
|
||||
|
||||
export interface DateField extends BaseFieldDef {
|
||||
type: 'date';
|
||||
placeholder?: string;
|
||||
}
|
||||
|
||||
export interface HeightField extends BaseFieldDef {
|
||||
type: 'height';
|
||||
}
|
||||
|
||||
export interface WeightField extends BaseFieldDef {
|
||||
type: 'weight';
|
||||
}
|
||||
|
||||
export interface SpeciesField extends BaseFieldDef {
|
||||
type: 'species';
|
||||
}
|
||||
|
||||
export interface SubspeciesField extends BaseFieldDef {
|
||||
type: 'subspecies';
|
||||
}
|
||||
|
||||
export interface CitizenshipField extends BaseFieldDef {
|
||||
type: 'citizenship';
|
||||
}
|
||||
|
||||
export interface LanguagesField extends BaseFieldDef {
|
||||
type: 'languages';
|
||||
}
|
||||
|
||||
export interface SeparatorField {
|
||||
type: 'separator';
|
||||
label: string;
|
||||
}
|
||||
|
||||
export type FieldDef =
|
||||
| NameField
|
||||
| TextField
|
||||
| TextareaField
|
||||
| ListField
|
||||
| NumberField
|
||||
| SelectField
|
||||
| MultiSelectField
|
||||
| CheckboxField
|
||||
| DateField
|
||||
| HeightField
|
||||
| WeightField
|
||||
| SpeciesField
|
||||
| SubspeciesField
|
||||
| CitizenshipField
|
||||
| LanguagesField
|
||||
| SeparatorField;
|
||||
|
||||
export interface RecordDef {
|
||||
type: string;
|
||||
preamble?: string;
|
||||
note?: string;
|
||||
fields: FieldDef[];
|
||||
}
|
||||
|
||||
export interface Template {
|
||||
id: string;
|
||||
name: string;
|
||||
description: string;
|
||||
schemaVersion: number;
|
||||
species?: string[];
|
||||
records: RecordDef[];
|
||||
}
|
||||
|
||||
export interface Character {
|
||||
id: string;
|
||||
template: Template;
|
||||
data: Record<string, unknown>;
|
||||
}
|
||||
58
src/lib/utils/blank.test.ts
Normal file
58
src/lib/utils/blank.test.ts
Normal file
|
|
@ -0,0 +1,58 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import { isBlankCharacter } from './blank';
|
||||
import type { Character } from '../types';
|
||||
|
||||
const template = {
|
||||
id: 'preset:standard',
|
||||
name: 'Standard',
|
||||
description: '',
|
||||
schemaVersion: 1,
|
||||
records: [
|
||||
{
|
||||
type: 'public',
|
||||
fields: [
|
||||
{ label: 'Name', type: 'text' as const },
|
||||
{ label: 'Species', type: 'species' as const },
|
||||
{ label: 'Spoken Languages', type: 'languages' as const }
|
||||
]
|
||||
}
|
||||
]
|
||||
};
|
||||
|
||||
function makeChar(data: Record<string, unknown>): Character {
|
||||
return { id: 'test', template, data };
|
||||
}
|
||||
|
||||
describe('isBlankCharacter', () => {
|
||||
it('returns true for empty data', () => {
|
||||
expect(isBlankCharacter(makeChar({}))).toBe(true);
|
||||
});
|
||||
|
||||
it('returns true when all values are empty strings', () => {
|
||||
expect(isBlankCharacter(makeChar({ name: '', species: '' }))).toBe(true);
|
||||
});
|
||||
|
||||
it('returns false when any string has a value', () => {
|
||||
expect(isBlankCharacter(makeChar({ name: 'Yury Zakharov' }))).toBe(false);
|
||||
});
|
||||
|
||||
it('returns true for empty arrays', () => {
|
||||
expect(isBlankCharacter(makeChar({ 'spoken-languages': [] }))).toBe(true);
|
||||
});
|
||||
|
||||
it('returns false when languages has any value', () => {
|
||||
expect(isBlankCharacter(makeChar({ 'spoken-languages': ['Tau Ceti Basic'] }))).toBe(false);
|
||||
});
|
||||
|
||||
it('returns false when languages has custom values', () => {
|
||||
expect(isBlankCharacter(makeChar({ 'spoken-languages': ['Tau Ceti Basic', "Siik'maas"] }))).toBe(false);
|
||||
});
|
||||
|
||||
it('returns true for zero numbers', () => {
|
||||
expect(isBlankCharacter(makeChar({ height: 0, weight: 0 }))).toBe(true);
|
||||
});
|
||||
|
||||
it('returns false for non-zero numbers', () => {
|
||||
expect(isBlankCharacter(makeChar({ height: 180 }))).toBe(false);
|
||||
});
|
||||
});
|
||||
14
src/lib/utils/blank.ts
Normal file
14
src/lib/utils/blank.ts
Normal file
|
|
@ -0,0 +1,14 @@
|
|||
import type { Character } from '../types';
|
||||
|
||||
export function isBlankCharacter(char: Character): boolean {
|
||||
if (!char.data) return true;
|
||||
for (const value of Object.values(char.data)) {
|
||||
if (value === '' || value === undefined || value === null || value === 0) continue;
|
||||
if (Array.isArray(value)) {
|
||||
if (value.length === 0) continue;
|
||||
return false;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
50
src/lib/utils/conversions.test.ts
Normal file
50
src/lib/utils/conversions.test.ts
Normal file
|
|
@ -0,0 +1,50 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import { cmToFeetInches, feetInchesToCm, kgToLb, lbToKg } from './conversions';
|
||||
|
||||
describe('cmToFeetInches', () => {
|
||||
it('converts 180 cm', () => {
|
||||
expect(cmToFeetInches(180)).toBe('5\'11"');
|
||||
});
|
||||
|
||||
it('converts 152 cm', () => {
|
||||
expect(cmToFeetInches(152)).toBe('5\'0"');
|
||||
});
|
||||
|
||||
it('converts 0 cm', () => {
|
||||
expect(cmToFeetInches(0)).toBe('0\'0"');
|
||||
});
|
||||
|
||||
it('converts 30 cm (just inches)', () => {
|
||||
expect(cmToFeetInches(30)).toBe('1\'0"');
|
||||
});
|
||||
});
|
||||
|
||||
describe('feetInchesToCm', () => {
|
||||
it('converts 5\'11" back', () => {
|
||||
expect(feetInchesToCm(5, 11)).toBeCloseTo(180.34, 0);
|
||||
});
|
||||
|
||||
it('converts 0\'0"', () => {
|
||||
expect(feetInchesToCm(0, 0)).toBe(0);
|
||||
});
|
||||
});
|
||||
|
||||
describe('kgToLb', () => {
|
||||
it('converts 75 kg', () => {
|
||||
expect(kgToLb(75)).toBeCloseTo(165.35, 0);
|
||||
});
|
||||
|
||||
it('converts 0 kg', () => {
|
||||
expect(kgToLb(0)).toBe(0);
|
||||
});
|
||||
});
|
||||
|
||||
describe('lbToKg', () => {
|
||||
it('converts 165 lb', () => {
|
||||
expect(lbToKg(165)).toBeCloseTo(74.84, 0);
|
||||
});
|
||||
|
||||
it('converts 0 lb', () => {
|
||||
expect(lbToKg(0)).toBe(0);
|
||||
});
|
||||
});
|
||||
22
src/lib/utils/conversions.ts
Normal file
22
src/lib/utils/conversions.ts
Normal file
|
|
@ -0,0 +1,22 @@
|
|||
const CM_PER_INCH = 2.54;
|
||||
const INCHES_PER_FOOT = 12;
|
||||
const LB_PER_KG = 2.20462;
|
||||
|
||||
export function cmToFeetInches(cm: number): string {
|
||||
const totalInches = Math.round(cm / CM_PER_INCH);
|
||||
const feet = Math.floor(totalInches / INCHES_PER_FOOT);
|
||||
const inches = totalInches % INCHES_PER_FOOT;
|
||||
return `${feet}'${inches}"`;
|
||||
}
|
||||
|
||||
export function feetInchesToCm(feet: number, inches: number): number {
|
||||
return (feet * INCHES_PER_FOOT + inches) * CM_PER_INCH;
|
||||
}
|
||||
|
||||
export function kgToLb(kg: number): number {
|
||||
return kg * LB_PER_KG;
|
||||
}
|
||||
|
||||
export function lbToKg(lb: number): number {
|
||||
return lb / LB_PER_KG;
|
||||
}
|
||||
33
src/lib/utils/dates.test.ts
Normal file
33
src/lib/utils/dates.test.ts
Normal file
|
|
@ -0,0 +1,33 @@
|
|||
import { describe, it, expect } from 'vitest';
|
||||
import { icYear, formatICDate } from './dates';
|
||||
|
||||
describe('icYear', () => {
|
||||
it('adds 442 to the real year', () => {
|
||||
expect(icYear(2026)).toBe(2468);
|
||||
});
|
||||
|
||||
it('works for other years', () => {
|
||||
expect(icYear(2000)).toBe(2442);
|
||||
});
|
||||
});
|
||||
|
||||
describe('formatICDate', () => {
|
||||
it('formats with ordinal suffixes', () => {
|
||||
expect(formatICDate(new Date(2026, 2, 1))).toBe('March 1st, 2468');
|
||||
expect(formatICDate(new Date(2026, 2, 2))).toBe('March 2nd, 2468');
|
||||
expect(formatICDate(new Date(2026, 2, 3))).toBe('March 3rd, 2468');
|
||||
expect(formatICDate(new Date(2026, 2, 4))).toBe('March 4th, 2468');
|
||||
});
|
||||
|
||||
it('handles 11th, 12th, 13th', () => {
|
||||
expect(formatICDate(new Date(2026, 0, 11))).toBe('January 11th, 2468');
|
||||
expect(formatICDate(new Date(2026, 0, 12))).toBe('January 12th, 2468');
|
||||
expect(formatICDate(new Date(2026, 0, 13))).toBe('January 13th, 2468');
|
||||
});
|
||||
|
||||
it('handles 21st, 22nd, 23rd', () => {
|
||||
expect(formatICDate(new Date(2026, 5, 21))).toBe('June 21st, 2468');
|
||||
expect(formatICDate(new Date(2026, 5, 22))).toBe('June 22nd, 2468');
|
||||
expect(formatICDate(new Date(2026, 5, 23))).toBe('June 23rd, 2468');
|
||||
});
|
||||
});
|
||||
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue