From adeda638f33130cb1fb454def0067b329ced17fe Mon Sep 17 00:00:00 2001
From: scarf
Date: Wed, 6 Sep 2023 07:57:59 +0900
Subject: [PATCH] style: simplify formatting with deno, part 1: `doc/` (#3108)
* chore: simplify deno config
used flat config from v1.33.0
see: https://deno.com/blog/v1.33#flatter-denojson-configuration
* docs(README): convert invalid table to badges
* docs: JSON and Markdown formatting guidelines
* style: format all markdown files with deno
* ci(autofix.ci): format markdown files with deno
---
.github/pull_request_template.md | 2 +-
.github/workflows/autofix.yml | 15 +-
CODE_OF_CONDUCT.md | 28 +-
ISSUES.md | 44 +-
README.ko.md | 71 +-
README.md | 72 +-
data/json/LOADING_ORDER.md | 37 +-
deno.jsonc | 19 +-
doc/ACCESSIBILITY.md | 23 +-
doc/BASECAMP.md | 342 ++-
doc/CHANGELOG_GUIDELINES.md | 45 +-
doc/CODE_STYLE.md | 30 +-
doc/COLOR.md | 88 +-
doc/COMPILING/COMPILER_SUPPORT.md | 34 +-
doc/COMPILING/COMPILING-CMAKE.md | 483 ++-
doc/COMPILING/COMPILING-CYGWIN.md | 49 +-
doc/COMPILING/COMPILING-MSYS.md | 57 +-
doc/COMPILING/COMPILING-VS-VCPKG.md | 107 +-
doc/COMPILING/COMPILING.md | 381 ++-
doc/CONTRIBUTING.ko.md | 141 +-
doc/CONTRIBUTING.md | 147 +-
doc/DEVELOPER_FAQ.md | 111 +-
doc/DEVELOPER_TOOLING.md | 321 +-
doc/EFFECTS_JSON.md | 530 ++--
doc/FACTIONS.md | 136 +-
doc/GAME_BALANCE.md | 453 ++-
doc/GUIDE_COMESTIBLES.md | 34 +-
doc/IN_REPO_MODS.md | 83 +-
doc/ITEM_SPAWN.md | 170 +-
.../Guide for beginning mapgen.md | 901 +++---
doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md | 254 +-
doc/JSON_FLAGS.md | 2624 +++++++++--------
doc/JSON_INFO.md | 2206 +++++++-------
doc/JSON_INHERITANCE.md | 75 +-
doc/JSON_STYLE.md | 71 +-
doc/LUA_SUPPORT.md | 462 +--
doc/MAGIC.md | 427 +--
doc/MANUAL_OF_STYLE.md | 11 +-
doc/MAPGEN.md | 936 +++---
doc/MARTIALART_JSON.md | 32 +-
doc/MISSIONS_JSON.md | 332 ++-
doc/MODDING.md | 244 +-
doc/MONSTERS.md | 430 ++-
doc/NPCs.md | 710 +++--
doc/OVERMAP.md | 285 +-
doc/PLAYER_ACTIVITY.md | 154 +-
doc/POINTS_COORDINATES.md | 223 +-
doc/POSTAPOC_PRICE_GUIDE.md | 103 +-
doc/REGION_SETTINGS.md | 529 ++--
doc/RELICS.md | 46 +-
doc/SOUNDPACKS.md | 438 +--
doc/TER_FURN_TRANSFORM.md | 20 +-
doc/TESTING.md | 163 +-
doc/TILESET.md | 394 ++-
doc/TRANSLATING.md | 391 ++-
doc/TRANSLATING_MODS.md | 155 +-
doc/USER_INTERFACE.md | 13 +-
doc/VEHICLES_JSON.md | 40 +-
doc/VITAMIN.md | 42 +-
doc/WEATHER_TYPE.md | 148 +-
.../MELEE_BALANCE_SPREADSHEET.md | 159 +-
61 files changed, 9671 insertions(+), 7400 deletions(-)
diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md
index f9f74d30f6aa..e8423a5704ce 100644
--- a/.github/pull_request_template.md
+++ b/.github/pull_request_template.md
@@ -5,7 +5,7 @@ NOTE: Please grant permission for repository maintainers to edit your PR. It is
CODE STYLE: please follow below guide.
JSON: https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/JSON_STYLE.md
-C++: https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/CODE_STYLE.md
+C++ and Markdown: https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/CODE_STYLE.md
!!!!!!!!!! WARNING !!!!!!!!!!
diff --git a/.github/workflows/autofix.yml b/.github/workflows/autofix.yml
index 80f2a2aabceb..5446dbae4a90 100644
--- a/.github/workflows/autofix.yml
+++ b/.github/workflows/autofix.yml
@@ -16,10 +16,17 @@ jobs:
steps:
- uses: actions/checkout@v3
- - name: astyle
- run: |
- sudo apt-get install astyle
- make astyle
+ - run: sudo apt-get install astyle
+
+ - uses: denoland/setup-deno@v1
+ with:
+ deno-version: v1.x
+
+ - name: format C++ files
+ run: make astyle
+
+ - name: format markdown files
+ run: deno fmt
- name: json formatting
run: make style-all-json-parallel RELEASE=1
diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md
index a2a234c9910b..d13c0b815ef3 100644
--- a/CODE_OF_CONDUCT.md
+++ b/CODE_OF_CONDUCT.md
@@ -1,24 +1,34 @@
## Our Pledge
-In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.
+In the interest of fostering an open and welcoming environment, we as contributors and maintainers
+pledge to making participation in our project and our community a harassment-free experience for
+everyone, regardless of age, body size, disability, ethnicity, sex characteristics, gender identity
+and expression, level of experience, education, socio-economic status, nationality, personal
+appearance, race, religion, or sexual identity and orientation.
## Our Standards
1. Assume other participants are posting in good faith, even if you disagree with what they say.
2. Make an effort to treat other participants with respect.
- 1. Do not take a harsh tone towards other participants, and especially don't make personal attacks against them.
- 2. Recognize that criticism of your statements is not a personal attack on you.
- 3. Avoid statements about the presumed typical desires, capabilities or actions of some demographic group.
+ 1. Do not take a harsh tone towards other participants, and especially don't make personal
+ attacks against them.
+ 2. Recognize that criticism of your statements is not a personal attack on you.
+ 3. Avoid statements about the presumed typical desires, capabilities or actions of some
+ demographic group.
3. Be especially kind to other contributors when saying they made a mistake.
-4. Don't argue unceasingly for your preferred course of action when a decision for some other course has already been made.
-5. If other participants complain about the way you express your ideas, please make an effort to cater to them.
+4. Don't argue unceasingly for your preferred course of action when a decision for some other course
+ has already been made.
+5. If other participants complain about the way you express your ideas, please make an effort to
+ cater to them.
6. Don't raise unrelated political issues.
-7. If you feel these standards are being violated, please alert the project's "ombudsman" Jakob at lamandus@hotmail.com
+7. If you feel these standards are being violated, please alert the project's "ombudsman" Jakob at
+ lamandus@hotmail.com
## Attribution
-This Code of Conduct is adapted from the [Contributor Covenant][CoC homepage], version 1.4, available at [http://contributor-covenant.org/version/1/4][CoC version] and the [GNU Kind Communications Guidelines][GKCG homepage].
-
+This Code of Conduct is adapted from the [Contributor Covenant][CoC homepage], version 1.4,
+available at [http://contributor-covenant.org/version/1/4][CoC version] and the
+[GNU Kind Communications Guidelines][GKCG homepage].
[CoC homepage]: http://contributor-covenant.org
[CoC version]: http://contributor-covenant.org/version/1/4/
diff --git a/ISSUES.md b/ISSUES.md
index dd800f72d7a6..5fa8ba6ce18f 100644
--- a/ISSUES.md
+++ b/ISSUES.md
@@ -2,24 +2,28 @@
## How to create new issues properly
-GitHub issues are used for everything from bug reporting to suggesting long-term ideas. You can make everything run much smoother by following some simple rules.
+GitHub issues are used for everything from bug reporting to suggesting long-term ideas. You can make
+everything run much smoother by following some simple rules.
### Rule zero
Always give your issue a meaningful title as this is the first thing anyone will see.
-Note: `[CR]` and `[WIP]` "tags" are meaningful only for PRs. All open issues by definition are request for comments and work in progress.
+Note: `[CR]` and `[WIP]` "tags" are meaningful only for PRs. All open issues by definition are
+request for comments and work in progress.
### Bug reports
-Before you submit a bug always search the current list of issues to see if it has been reported already.
+Before you submit a bug always search the current list of issues to see if it has been reported
+already.
Your bug report has to include:
- On what OS did you experience the problem (Windows, Linux, OS X etc.)
- What version were you playing:
- - Tiles or Curses (text-based)
- - Version string (preferably full version e.g. "0.C-4547-g3f1c109", or Jenkins build number e.g. 3245)
+- Tiles or Curses (text-based)
+- Version string (preferably full version e.g. "0.C-4547-g3f1c109", or Jenkins build number
+ e.g. 3245)
- Description of the problem written in a way that enables anyone to try and recreate it
Your bug report may include:
@@ -32,24 +36,42 @@ Bonus points for:
- Checking if the bug exists under latest experimental build
- Checking if it is OS specific
-The OS and BN version are very important - with the pace of changes here it is possible the bug you have encountered has already been fixed. After that reproducibility is the key, so write your report with all the necessary details.
+The OS and BN version are very important - with the pace of changes here it is possible the bug you
+have encountered has already been fixed. After that reproducibility is the key, so write your report
+with all the necessary details.
### Enhancements and addition ideas
-We have hundreds of issues open - most of them are ideas and suggestions. If you have a general idea or anything that can't be easily described in terms of *current* code changes you'd be better off suggesting it in the appropriate section of [the forum](https://discourse.cataclysmdda.org/). You'll also get much broader exposure for your idea there. After developing a polished idea on the forum, it should be easy to make a GitHub issue for it.
+We have hundreds of issues open - most of them are ideas and suggestions. If you have a general idea
+or anything that can't be easily described in terms of _current_ code changes you'd be better off
+suggesting it in the appropriate section of [the forum](https://discourse.cataclysmdda.org/). You'll
+also get much broader exposure for your idea there. After developing a polished idea on the forum,
+it should be easy to make a GitHub issue for it.
-Please first search if something like what you have on mind has been already proposed. If so, feel free to join the discussion! If your idea is related but sufficiently different, open a new issue and refer to the older discussion (use GitHub's `#issue_number` reference system).
+Please first search if something like what you have on mind has been already proposed. If so, feel
+free to join the discussion! If your idea is related but sufficiently different, open a new issue
+and refer to the older discussion (use GitHub's `#issue_number` reference system).
Remember to take part in the discussion of your suggestions.
### Questions
-You should direct your questions to the forum or ask on IRC. You should also read the included documentation and additional text files, e.g. [COMPILING.md](doc/COMPILING/COMPILING.md) if you have problems building.
+You should direct your questions to the forum or ask on IRC. You should also read the included
+documentation and additional text files, e.g. [COMPILING.md](doc/COMPILING/COMPILING.md) if you have
+problems building.
## Bounties
-Placing a bounty *does not* necessarily mean that change will be incorporated into the main game. Please coordinate in the issue you intend to place a bounty on to determine if it is a change the project will accept, and keep in mind that placing a bounty will not confer special status on the issue. A good way of thinking about bounties is as encouragement for contributors to work on a particular issue, and certainly not as "paying for features".
+Placing a bounty _does not_ necessarily mean that change will be incorporated into the main game.
+Please coordinate in the issue you intend to place a bounty on to determine if it is a change the
+project will accept, and keep in mind that placing a bounty will not confer special status on the
+issue. A good way of thinking about bounties is as encouragement for contributors to work on a
+particular issue, and certainly not as "paying for features".
## Issue resolution
-We do not assign people to issues. If you plan to work on a bug fix or a validated idea feel free to just comment about that. Actual PRs are of much greater value than any assignments. In general the first correct PR about something will be the PR that will get merged, but remember: we are using Git - you can collaborate with someone else easily by sending them patches or PRs against their PR branch.
+We do not assign people to issues. If you plan to work on a bug fix or a validated idea feel free to
+just comment about that. Actual PRs are of much greater value than any assignments. In general the
+first correct PR about something will be the PR that will get merged, but remember: we are using
+Git - you can collaborate with someone else easily by sending them patches or PRs against their PR
+branch.
diff --git a/README.ko.md b/README.ko.md
index 22db839ce565..fd39ffe1f0c4 100644
--- a/README.ko.md
+++ b/README.ko.md
@@ -3,8 +3,7 @@
- [![en][icon-en]][en]
- [![ko][icon-ko]][ko]
+[![en][icon-en]][en] [![ko][icon-ko]][ko]
@@ -13,36 +12,48 @@
[ko]: ./README.ko.md
[icon-ko]: https://img.shields.io/badge/lang-ko-orange?style=flat-square
-
-
카타클리즘: 밝은 밤은 포스트 아포칼립스 세계에서 벌어지는 공상과학 로그라이크입니다.
-"좀비 게임"이라 불리기도 하지만, 그뿐만이 아닙니다. 험난하고, 절차적으로 생성되는 세계에서 살아남아야 합니다. 이제는 죽어버린 문명의 잔해를 뒤져 음식, 장비, 운이 좋다면 연료가 가득 찬 차량을 찾아 탈출하세요.
+"좀비 게임"이라 불리기도 하지만, 그뿐만이 아닙니다. 험난하고, 절차적으로 생성되는 세계에서
+살아남아야 합니다. 이제는 죽어버린 문명의 잔해를 뒤져 음식, 장비, 운이 좋다면 연료가 가득 찬 차량을
+찾아 탈출하세요.
-좀비, 거대 곤충, 킬러 로봇, 더 이상하고 치명적인 것들, 그리고 당신이 가진 것을 노리는 이들까지, 강력한 괴물들과 싸우거나 도망치세요.
+좀비, 거대 곤충, 킬러 로봇, 더 이상하고 치명적인 것들, 그리고 당신이 가진 것을 노리는 이들까지,
+강력한 괴물들과 싸우거나 도망치세요.
카타클리즘을 멈출 방법을 찾거나.... 가장 강력한 괴물 중 하나가 되세요.
-> 카타클리즘: 밝은 밤(Cataclysm: Bright Nights)은 카타클리즘: 어두운 나날(Cataclysm: Dark Days Ahead)의 포크입니다. [위키](https://github.com/cataclysmbnteam/cataclysm-BN/wiki/Changes-so-far)에서 어떤 차이가 있는지 확인하세요.
+> 카타클리즘: 밝은 밤(Cataclysm: Bright Nights)은 카타클리즘: 어두운 나날(Cataclysm: Dark Days
+> Ahead)의 포크입니다.
+> [위키](https://github.com/cataclysmbnteam/cataclysm-BN/wiki/Changes-so-far)에서 어떤 차이가 있는지
+> 확인하세요.
## 다운로드
-[![안정판][stable-releases-badge]][stable-releases] [![최신 릴리즈][all-releases-badge]][all-releases]
+### 실행 파일
+
+[![Stable][stable-releases-badge]][stable-releases] [![Recent][all-releases-badge]][all-releases]
-[stable-releases]: https://github.com/cataclysmbnteam/cataclysm-BN/releases/tag/cbn-0.3
-[stable-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/cataclysm-BN?style=for-the-badge&color=success&label=stable
-[all-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases?q=prerelease%3Atrue&expanded=true
-[all-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/cataclysm-BN?style=for-the-badge&color=important&label=Latest%20Release&include_prereleases&sort=date
+### 소스 코드
-| [소스 코드][source-zip-archive] | [저장소에서 클론하기][clone] |
-| :-----------------------------: | :--------------------------: |
+[![Source Code][source-badge]][source] [![Zip Archive][clone-badge]][clone]
-[source-zip-archive]: https://github.com/cataclysmbnteam/cataclysm-BN/archive/master.zip "소스 코드를 .zip 아카이브로 다운로드할 수 있습니다"
-[clone]: https://github.com/cataclysmbnteam/cataclysm-BN/ "GitHub 저장소에서 클론할 수 있습니다"
+[stable-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases/latest "안정판 실행 파일 내려받기"
+[stable-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/Cataclysm-BN?style=for-the-badge&color=success&label=안정판
+[all-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases?q=prerelease%3Atrue&expanded=true "실험판 실행 파일 내려받기"
+[all-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/Cataclysm-BN?style=for-the-badge&color=important&label=최신%20실험판&include_prereleases&sort=date
+[source]: https://github.com/cataclysmbnteam/Cataclysm-BN/archive/master.zip "소스 코드를 .zip 아카이브로 다운로드할 수 있습니다"
+[source-badge]: https://img.shields.io/badge/ZIP%20아카이브로%20내려받기-black?style=for-the-badge&logo=github
+[clone]: https://github.com/cataclysmbnteam/Cataclysm-BN/ "GitHub 저장소에서 클론할 수 있습니다"
+[clone-badge]: https://img.shields.io/badge/저장소에서%20클론하기-black?style=for-the-badge&logo=github
## 빌드하기
-[COMPILING.md](doc/COMPILING/COMPILING.md)를 참고하세요 - 리눅스, OS X, 윈도우즈와 BSD에서 빌드하기 위한 일반적인 정보부터 보다 자세한 레시피가 담겨있습니다. [COMPILER_SUPPORT.md](doc/COMPILING/COMPILER_SUPPORT.md)에서 지원하는 컴파일러를 확인할 수 있습니다. 더 자세한 정보는 [doc/](https://github.com/cataclysmbnteam/cataclysm-BN/tree/upload/doc)에서도 찾아볼 수 있습니다.
+[COMPILING.md](doc/COMPILING/COMPILING.md)를 참고하세요 - 리눅스, OS X, 윈도우즈와 BSD에서 빌드하기
+위한 일반적인 정보부터 보다 자세한 레시피가 담겨있습니다.
+[COMPILER_SUPPORT.md](doc/COMPILING/COMPILER_SUPPORT.md)에서 지원하는 컴파일러를 확인할 수 있습니다.
+더 자세한 정보는 [doc/](https://github.com/cataclysmbnteam/cataclysm-BN/tree/upload/doc)에서도
+찾아볼 수 있습니다.
또한 다음 빌드 가이드도 있습니다.
@@ -52,17 +63,20 @@
## 기여하기
-> 카타클리즘: 밝은 밤은 크리에이티브 커먼즈 저작자표시-동일조건변경허락 3.0 라이선스로 개발되고 있습니다. 소스 코드와 게임 내용물은 어떠한 목적으로든 사용, 수정, 재배포가 가능합니다. 자세한 내용은 를 참고하세요.
-> 일부 코드는 프로젝트와 함께 배포되지만, 다른 소프트웨어 라이선스에 따라 배포됩니다. 다른 소프트웨어 라이선스에 따라 배포되는 파일들은 각 파일에 라이선스 공지가 포함되어 있습니다.
+> 카타클리즘: 밝은 밤은 크리에이티브 커먼즈 저작자표시-동일조건변경허락 3.0 라이선스로 개발되고
+> 있습니다. 소스 코드와 게임 내용물은 어떠한 목적으로든 사용, 수정, 재배포가 가능합니다. 자세한
+> 내용은 를 참고하세요. 일부 코드는 프로젝트와 함께
+> 배포되지만, 다른 소프트웨어 라이선스에 따라 배포됩니다. 다른 소프트웨어 라이선스에 따라 배포되는
+> 파일들은 각 파일에 라이선스 공지가 포함되어 있습니다.
[CONTRIBUTING.ko.md](./doc/CONTRIBUTING.ko.md)에서 자세한 내용을 확인할 수 있습니다.
## 커뮤니티
-| [저장소][repo] | [토론][discussion] | [공식 저장소][discord] | [BN 채널@모딩 커뮤니티][modding] |
-| :------------: | :----------------: | :--------------------: | :------------------------------: |
+[![Discussions](https://img.shields.io/badge/포럼에서%20토론하기-black?style=for-the-badge&logo=github)][discussion]
+[![Discord](https://img.shields.io/discord/830879262763909202?style=for-the-badge&logo=discord&label=공식%20디스코드%20서버)][discord]
+[![Discussions](https://img.shields.io/badge/CDDA%20모딩%20커뮤니티-green?style=for-the-badge&logo=discord)][modding]
-[repo]: https://github.com/cataclysmbnteam/cataclysm-BN
[discussion]: https://github.com/cataclysmbnteam/cataclysm-BN/discussions
[discord]: https://discord.gg/XW7XhXuZ89
[modding]: https://discord.gg/B5q4XCa "비공식 DDA모딩 커뮤니티에도 BN 채널이 있습니다."
@@ -71,11 +85,13 @@
#### 튜토리얼이 있나요?
-메인 메뉴의 **Special** 메뉴에서 찾을 수 있습니다. (코드 변경으로 인해 튜토리얼이 작동하지 않을 수 있습니다.) 게임 내에서도 `?` 키를 눌러 도움 메뉴에 접근할 수 있습니다.
+메인 메뉴의 **Special** 메뉴에서 찾을 수 있습니다. (코드 변경으로 인해 튜토리얼이 작동하지 않을 수
+있습니다.) 게임 내에서도 `?` 키를 눌러 도움 메뉴에 접근할 수 있습니다.
#### 단축키를 어떻게 바꾸나요?
-`?` 키를 누르고 `1` 키를 눌러 전체 단축키 목록을 확인할 수 있습니다. `+` 키를 눌러 단축키를 추가할 수 있고, `a-w` 키를 눌러 해당 행동을 선택한 후, 할당할 키를 누르면 됩니다.
+`?` 키를 누르고 `1` 키를 눌러 전체 단축키 목록을 확인할 수 있습니다. `+` 키를 눌러 단축키를 추가할
+수 있고, `a-w` 키를 눌러 해당 행동을 선택한 후, 할당할 키를 누르면 됩니다.
#### 새 세계를 어떻게 시작하나요?
@@ -83,7 +99,9 @@
#### 버그를 발견했어요. 어떻게 해야 하나요?
-디버그 메뉴에서 [버그 리포트](https://github.com/cataclysmbnteam/cataclysm-BN/issues/new?template=bug_report.yml)를 제출할 수 있습니다.
+디버그 메뉴에서
+[버그 리포트](https://github.com/cataclysmbnteam/cataclysm-BN/issues/new?template=bug_report.yml)를
+제출할 수 있습니다.
게임 내에서 `Submit a bug report on github`를 실행하여 이슈를 제출할 수 있습니다.
@@ -97,4 +115,5 @@
#### 제안을 하고 싶어요. 어떻게 해야 하나요?
-[토론 페이지](https://github.com/cataclysmbnteam/cataclysm-BN/discussions/categories/ideas)에서 아이디어를 제안할 수 있습니다. 새 기능, 포팅 요청, 모드 아이디어, 무엇이든지요!
+[토론 페이지](https://github.com/cataclysmbnteam/cataclysm-BN/discussions/categories/ideas)에서
+아이디어를 제안할 수 있습니다. 새 기능, 포팅 요청, 모드 아이디어, 무엇이든지요!
diff --git a/README.md b/README.md
index 083b1770cc60..e03b4a2754b5 100644
--- a/README.md
+++ b/README.md
@@ -3,8 +3,7 @@
- [![en][icon-en]][en]
- [![ko][icon-ko]][ko]
+[![en][icon-en]][en] [![ko][icon-ko]][ko]
@@ -13,12 +12,16 @@
[ko]: ./README.ko.md
[icon-ko]: https://img.shields.io/badge/lang-ko-orange?style=flat-square
-
Cataclysm: Bright Nights is a roguelike with sci-fi elements set in a post-apocalyptic world.
-While some have described it as a "zombie game", there is far more to Cataclysm than that. Struggle to survive in a harsh, persistent, procedurally generated world. Scavenge the remnants of a dead civilization for food, equipment, or, if you are lucky, a vehicle with a full tank of gas to get you the hell out of there.
+While some have described it as a "zombie game", there is far more to Cataclysm than that. Struggle
+to survive in a harsh, persistent, procedurally generated world. Scavenge the remnants of a dead
+civilization for food, equipment, or, if you are lucky, a vehicle with a full tank of gas to get you
+the hell out of there.
-Fight to defeat or escape from a wide variety of powerful monstrosities, from zombies to giant insects to killer robots and things far stranger and deadlier, and against the others like yourself, who want what you have.
+Fight to defeat or escape from a wide variety of powerful monstrosities, from zombies to giant
+insects to killer robots and things far stranger and deadlier, and against the others like yourself,
+who want what you have.
Find a way to stop the Cataclysm ... or become one of its strongest monsters.
@@ -27,42 +30,54 @@ Find a way to stop the Cataclysm ... or become one of its strongest monsters.
## Downloads
+### Executables
+
[![Stable][stable-releases-badge]][stable-releases] [![Recent][all-releases-badge]][all-releases]
-[stable-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases/tag/cbn-0.3
-[stable-releases-badge]:
-[all-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases?q=prerelease%3Atrue&expanded=true
-[all-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/Cataclysm-BN?style=for-the-badge&color=important&label=Latest%20Release&include_prereleases&sort=date
+### Source Code
-| [Source Code][source-zip-archive] | [Clone From Repo][clone] |
-| :-------------------------------: | :----------------------: |
+[![Source Code][source-badge]][source] [![Zip Archive][clone-badge]][clone]
-[source-zip-archive]: https://github.com/cataclysmbnteam/Cataclysm-BN/archive/master.zip "The source can be downloaded as a .zip archive"
+[stable-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases/latest "Download stable executable"
+[stable-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/Cataclysm-BN?style=for-the-badge&color=success&label=stable "Download experimental executable"
+[all-releases]: https://github.com/cataclysmbnteam/Cataclysm-BN/releases?q=prerelease%3Atrue&expanded=true
+[all-releases-badge]: https://img.shields.io/github/v/release/cataclysmbnteam/Cataclysm-BN?style=for-the-badge&color=important&label=Latest%20Release&include_prereleases&sort=date
+[source]: https://github.com/cataclysmbnteam/Cataclysm-BN/archive/master.zip "The source can be downloaded as a .zip archive"
+[source-badge]: https://img.shields.io/badge/Zip%20Archive-black?style=for-the-badge&logo=github
[clone]: https://github.com/cataclysmbnteam/Cataclysm-BN/ "clone from our GitHub repo"
+[clone-badge]: https://img.shields.io/badge/Clone%20From%20Repo-black?style=for-the-badge&logo=github
## Compile
-Please read [COMPILING.md](doc/COMPILING/COMPILING.md) - it covers general information and more specific recipes for Linux, OS X, Windows and BSD. See [COMPILER_SUPPORT.md](doc/COMPILING/COMPILER_SUPPORT.md) for details on which compilers we support. And you can always dig for more information in [doc/](https://github.com/cataclysmbnteam/Cataclysm-BN/tree/upload/doc).
+Please read [COMPILING.md](doc/COMPILING/COMPILING.md) - it covers general information and more
+specific recipes for Linux, OS X, Windows and BSD. See
+[COMPILER_SUPPORT.md](doc/COMPILING/COMPILER_SUPPORT.md) for details on which compilers we support.
+And you can always dig for more information in
+[doc/](https://github.com/cataclysmbnteam/Cataclysm-BN/tree/upload/doc).
We also have the following build guides:
- Building on Windows with `MSYS2` at [COMPILING-MSYS.md](doc/COMPILING/COMPILING-MSYS.md)
- Building on Windows with `vcpkg` at [COMPILING-VS-VCPKG.md](doc/COMPILING/COMPILING-VS-VCPKG.md)
-- Building with `cmake` at [COMPILING-CMAKE.md](doc/COMPILING/COMPILING-CMAKE.md) (_unofficial guide_)
+- Building with `cmake` at [COMPILING-CMAKE.md](doc/COMPILING/COMPILING-CMAKE.md) (_unofficial
+ guide_)
## Contribute
-> Cataclysm: Bright Nights developed under Creative Commons Attribution ShareAlike 3.0 license. The code and content of the game is free to use, modify, and redistribute for any purpose whatsoever. See http://creativecommons.org/licenses/by-sa/3.0/ for details.
-> Some code distributed with the project is not part of the project and is released under different software licenses, the files covered by different software licenses have their own license notices.
+> Cataclysm: Bright Nights developed under Creative Commons Attribution ShareAlike 3.0 license. The
+> code and content of the game is free to use, modify, and redistribute for any purpose whatsoever.
+> See http://creativecommons.org/licenses/by-sa/3.0/ for details. Some code distributed with the
+> project is not part of the project and is released under different software licenses, the files
+> covered by different software licenses have their own license notices.
Please see [CONTRIBUTING.md](./doc/CONTRIBUTING.md) for details.
## Community
-| [Repository][repo] | [Discussions][discussion] | [Official Discord][discord] | [BN Channel@Modding Community][modding] |
-| :----------------: | :-----------------------: | :-------------------------: | :-------------------------------------: |
+[![Discussions](https://img.shields.io/badge/Discussions-black?style=for-the-badge&logo=github)][discussion]
+[![Discord](https://img.shields.io/discord/830879262763909202?style=for-the-badge&logo=discord)][discord]
+[![Discussions](https://img.shields.io/badge/CDDA%20Modding-green?style=for-the-badge&logo=discord)][modding]
-[repo]: https://github.com/cataclysmbnteam/Cataclysm-BN
[discussion]: https://github.com/cataclysmbnteam/Cataclysm-BN/discussions
[discord]: https://discord.gg/XW7XhXuZ89
[modding]: https://discord.gg/B5q4XCa "Unofficial DDA modding community discord has a BN channel"
@@ -71,11 +86,15 @@ Please see [CONTRIBUTING.md](./doc/CONTRIBUTING.md) for details.
#### Is there a tutorial?
-Yes, you can find the tutorial in the **Special** menu at the main menu (be aware that due to many code changes the tutorial may not function). You can also access documentation in-game via the `?` key.
+Yes, you can find the tutorial in the **Special** menu at the main menu (be aware that due to many
+code changes the tutorial may not function). You can also access documentation in-game via the `?`
+key.
#### How can I change the key bindings?
-Press the `?` key, followed by the `1` key to see the full list of key commands. Press the `+` key to add a key binding, select which action with the corresponding letter key `a-w`, and then the key you wish to assign to that action.
+Press the `?` key, followed by the `1` key to see the full list of key commands. Press the `+` key
+to add a key binding, select which action with the corresponding letter key `a-w`, and then the key
+you wish to assign to that action.
#### How can I start a new world?
@@ -83,7 +102,8 @@ Press the `?` key, followed by the `1` key to see the full list of key commands.
#### I've found a bug. What should I do?
-[Bug report](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/new?template=bug_report.yml) can be submitted via debug menu.
+[Bug report](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/new?template=bug_report.yml) can
+be submitted via debug menu.
Run `Submit a bug report on github` inside the game to submit an issue.
@@ -97,5 +117,9 @@ It will open a bug report on browser with `Version and configuration` filled in.
#### I would like to make a suggestion. What should I do?
-- For simple ideas: please visit [our Discussions page](https://github.com/cataclysmbnteam/Cataclysm-BN/discussions/categories/ideas). It could be a new feature, a port request, a mod idea, or anything else.
-- Please submit an issue on [our GitHub page](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/) using [feature request form](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/new?template=feature_request.yml).
+- For simple ideas: please visit
+ [our Discussions page](https://github.com/cataclysmbnteam/Cataclysm-BN/discussions/categories/ideas).
+ It could be a new feature, a port request, a mod idea, or anything else.
+- Please submit an issue on
+ [our GitHub page](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/) using
+ [feature request form](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/new?template=feature_request.yml).
diff --git a/data/json/LOADING_ORDER.md b/data/json/LOADING_ORDER.md
index 5459b38c5d31..9f48a405678e 100644
--- a/data/json/LOADING_ORDER.md
+++ b/data/json/LOADING_ORDER.md
@@ -1,18 +1,16 @@
-# JSON Loading Order #
+# JSON Loading Order
-All files here in data/json are read eventually, but the order in which they're
-read can be important for objects with dependencies on other kinds of objects
-(e.g. recipes depend on skills). Ensuring the proper loading order will prevent
-surprises that, most often, manifest as crash-to-desktop with segfault (a very
-bad thing).
+All files here in data/json are read eventually, but the order in which they're read can be
+important for objects with dependencies on other kinds of objects (e.g. recipes depend on skills).
+Ensuring the proper loading order will prevent surprises that, most often, manifest as
+crash-to-desktop with segfault (a very bad thing).
-The way Cataclysm finds and loads json files is by running a breadth-first
-search in the tree data/json/. This means `data/json/whatever.json` will
-**always** be read before `data/json/subdir/whatever.json`. This tells us how to
-ensure dependency loading order.
+The way Cataclysm finds and loads json files is by running a breadth-first search in the tree
+data/json/. This means `data/json/whatever.json` will **always** be read before
+`data/json/subdir/whatever.json`. This tells us how to ensure dependency loading order.
-For instance, if you have scenarios that depend on professions that depend on
-skills, you'll want a directory structure such as the following:
+For instance, if you have scenarios that depend on professions that depend on skills, you'll want a
+directory structure such as the following:
```
data/json/
@@ -23,13 +21,12 @@ data/json/
scenarios.json
```
-Which results in a loading order of: `skills.json` then `professions.json` and
-then `scenarios.json`.
+Which results in a loading order of: `skills.json` then `professions.json` and then
+`scenarios.json`.
-## Same-depth loading order ##
+## Same-depth loading order
-Note that, when files (or directories) are at the same depth
-(i.e. all in `data/json/`), they will be read in lexical order, which is
-more or less equivalent to alphabetical order for file names that use only
-ascii characters. For UTF-8 or otherwise non-ascii file names, the names will be
-ordered by codepoint.
+Note that, when files (or directories) are at the same depth (i.e. all in `data/json/`), they will
+be read in lexical order, which is more or less equivalent to alphabetical order for file names that
+use only ascii characters. For UTF-8 or otherwise non-ascii file names, the names will be ordered by
+codepoint.
diff --git a/deno.jsonc b/deno.jsonc
index 6f78ace93c48..af42dc86124e 100644
--- a/deno.jsonc
+++ b/deno.jsonc
@@ -1,11 +1,12 @@
{
- "tasks": {
- "tools": "deno run --watch -A"
- },
- "test": { "files": { "include": ["./tools"] } },
- "lint": { "files": { "include": ["./tools"] } },
- "fmt": {
- "files": { "include": ["./tools"] },
- "options": { "semiColons": false, "lineWidth": 100 }
- }
+ "tasks": {
+ "tools": "deno run --watch -A"
+ },
+ "test": { "include": ["./tools"] },
+ "lint": { "include": ["./tools"] },
+ "fmt": {
+ "include": ["./tools", "./doc", "*.md"],
+ "semiColons": false,
+ "lineWidth": 100
+ }
}
diff --git a/doc/ACCESSIBILITY.md b/doc/ACCESSIBILITY.md
index 5a4c511464d8..a3d75e415445 100644
--- a/doc/ACCESSIBILITY.md
+++ b/doc/ACCESSIBILITY.md
@@ -1,17 +1,15 @@
### Compatibility with screen readers
-There are people who uses screen readers to play Cataclysm DDA. In order for screen
-readers to announce the most important information in a UI, the terminal cursor has
-to be placed at the correct location. This information may be text such as selected
-item names in a list, etc, and the cursor has to be placed exactly at the beginning
-of the text for screen readers to announce it.
+There are people who uses screen readers to play Cataclysm DDA. In order for screen readers to
+announce the most important information in a UI, the terminal cursor has to be placed at the correct
+location. This information may be text such as selected item names in a list, etc, and the cursor
+has to be placed exactly at the beginning of the text for screen readers to announce it.
-`wmove` in `output.h|cpp` is the function to move the cursor to a specific location.
-After calling `wmove` with the target `catacurses::window` and cursor position,
-`wrefresh` needs to be called immediately afterwards for `wmove` to take effect.
+`wmove` in `output.h|cpp` is the function to move the cursor to a specific location. After calling
+`wmove` with the target `catacurses::window` and cursor position, `wrefresh` needs to be called
+immediately afterwards for `wmove` to take effect.
-Here is an example of placing the cursor explicitly at the beginning of a piece
-of text:
+Here is an example of placing the cursor explicitly at the beginning of a piece of text:
```cpp
catacurses::window win = ...; // target window
@@ -34,6 +32,5 @@ wrefresh( win );
// no output code should follow as they might change the cursor position
```
-As shown in the above example, it is preferable to record the intended cursor
-position in a variable when the text is printed, and move the cursor later using
-the variable to ensure consisitency.
+As shown in the above example, it is preferable to record the intended cursor position in a variable
+when the text is printed, and move the cursor later using the variable to ensure consisitency.
diff --git a/doc/BASECAMP.md b/doc/BASECAMP.md
index a0fececdc6e2..2b3cbe31c26a 100644
--- a/doc/BASECAMP.md
+++ b/doc/BASECAMP.md
@@ -1,195 +1,257 @@
# Recommended reading
-Basecamps leverage many existing aspects of JSON data such as recipes and mapgen. It's recommended to be familiar with those:
-* [JSON info](JSON_INFO.md) has information on common fields for recipes
-* [mapgen](MAPGEN.md), see section 3 about `update_mapgen`
+Basecamps leverage many existing aspects of JSON data such as recipes and mapgen. It's recommended
+to be familiar with those:
+
+- [JSON info](JSON_INFO.md) has information on common fields for recipes
+- [mapgen](MAPGEN.md), see section 3 about `update_mapgen`
# Adding alternate basecamp upgrade paths
-A basecamp upgrade path is a series of basecamp upgrade missions that upgrade the camp. Upgrade missions are generally performed sequentially, but there is an option to have them branch. Branched missions optionally can have further missions that require missions from other branches.
+A basecamp upgrade path is a series of basecamp upgrade missions that upgrade the camp. Upgrade
+missions are generally performed sequentially, but there is an option to have them branch. Branched
+missions optionally can have further missions that require missions from other branches.
Bascamp upgrade paths are defined by several related files:
-* The recipe JSONs that define what the material, tool, and skill requirements to perform an upgrade mission and the blueprint mapgen, blueprint requirements, blueprint provides, and blueprint resources associated with each upgrade mission.
-* The mapgen_update JSONs that define how the map will change when the upgrade mission is complete. These may include shared instances of nested mapgen, such a standard room or tent.
-* The recipe_group JSONs that define what recipes can be crafted after completing the upgrade mission and what camps and expansions are available.
+
+- The recipe JSONs that define what the material, tool, and skill requirements to perform an upgrade
+ mission and the blueprint mapgen, blueprint requirements, blueprint provides, and blueprint
+ resources associated with each upgrade mission.
+- The mapgen_update JSONs that define how the map will change when the upgrade mission is complete.
+ These may include shared instances of nested mapgen, such a standard room or tent.
+- The recipe_group JSONs that define what recipes can be crafted after completing the upgrade
+ mission and what camps and expansions are available.
## recipe JSONs
+
The recipe JSONs are standard recipe JSONs, with the addition of a few fields.
-New field | Description
--- | --
-`"construction_blueprint"` | string, the `"update_mapgen_id"` of the mapgen_update JSON that will be performed when the upgrade mission is complete.
-`"construction_name"` | string, a short description/name of the upgrade mission that is displayed in the base camp mission selector. The recipe's `"description"` field has the extended description that explains why a player might want to have an NPC perform this upgrade mission.
-`"blueprint_provides"` | array of objects, with each object having an `"id"` string and an optional `"amount"` integer. These are the camp features that are available when the upgrade mission is complete. Every upgrade mission provides its recipe `"result"` with an amount of 1 automatically and that string does not need to be listed in `"blueprint_provides"`.
-`"blueprint_requires"` | array of objects, with each object having an `"id"` string and an optional `"amount"` integer. These are the camp features that are required before the upgrade mission can be attempted.
-`"blueprint_excludes"` | array of objects, with each object having an `"id"` string and an optional `"amount"` integer. These are the camp features that prevent the upgrade mission from being attempted if they exist.
-`"blueprint_resources"` | array of `"itype_id"`s. Items with those ids will be added to the camp inventory after the upgrade mission is completed and can be used for crafting or additional upgrade missions.
+| New field | Description |
+| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
+| `"construction_blueprint"` | string, the `"update_mapgen_id"` of the mapgen_update JSON that will be performed when the upgrade mission is complete. |
+| `"construction_name"` | string, a short description/name of the upgrade mission that is displayed in the base camp mission selector. The recipe's `"description"` field has the extended description that explains why a player might want to have an NPC perform this upgrade mission. |
+| `"blueprint_provides"` | array of objects, with each object having an `"id"` string and an optional `"amount"` integer. These are the camp features that are available when the upgrade mission is complete. Every upgrade mission provides its recipe `"result"` with an amount of 1 automatically and that string does not need to be listed in `"blueprint_provides"`. |
+| `"blueprint_requires"` | array of objects, with each object having an `"id"` string and an optional `"amount"` integer. These are the camp features that are required before the upgrade mission can be attempted. |
+| `"blueprint_excludes"` | array of objects, with each object having an `"id"` string and an optional `"amount"` integer. These are the camp features that prevent the upgrade mission from being attempted if they exist. |
+| `"blueprint_resources"` | array of `"itype_id"`s. Items with those ids will be added to the camp inventory after the upgrade mission is completed and can be used for crafting or additional upgrade missions. |
### blueprint requires, provides, and excludes
-blueprint requires, blueprint provides, and blueprint exlcudes are abstract concepts or flags that an upgrade mission requires to start, or that are provided by a previous upgrade mission to satisfy the blueprint requirements of a current upgrade mission, or that prevent an upgrade mission from being available. Each one has an `"id"` and an `"amount"`. Multiple requires, provides, or excludes with the same `"id"` sum their `"amount"` if they're on the same basecamp expansion.
-
-Every upgrade mission has its recipe `"result"` as a blueprint_provides and a blueprint_excludes, so upgrade missions will automatically prevent themselves from being repeatable.
-
-These are arbitrary strings and can be used to control the branching of the upgrade paths. However, some strings have meaning within the basecamp code:
-
-provides `"id"` | meaning
--- | --
-`"bed"` | every 2 `"amount"`' of `"bed"` allows another expansion in the camp, to a maximum of 8, not include the camp center.
-`"tool_storage"` | after this upgrade mission is complete, the Store Tools mission will be available
-.
-`"radio"` | after this upgrade mission is complete, two way radios communicating to the camp have extended range.
-`"pantry"` | after this upgrade mission is complete, the Distribute Food mission is more efficient when dealing with short term spoilage items.
-`"gathering"` | after this upgrade mission is complete, the Gather Materials, Distribute Food, and Reset Sort Points basecamp missions will be available.
-`"firewood"` | after this upgrade mission is complete, the Gather Firewood basecamp mission will be available.
-`"sorting"` | after this upgrade mission is complete, the Menial Labor basecamp mission will be available.
-`"logging"` | after this upgrade mission is complete, the Cut Logs and Clear a Forest basecamp missions will be available.
-`"relaying"` | after this upgrade mission is complete, the Setup Hide Site and Relay Hide Site basecamp missions will be available.
-`"foraging"` | after this upgrade mission is complete, the Forage for Plants basecamp mission will be available.
-`"trapping"` | after this upgrade mission is complete, the Trap Small Game basecamp mission will be available.
-`"hunting"` | after this upgrade mission is complete, the Hunt Large Animals basecamp mission will be available.
-`"walls"` | after this upgrade mission is complete, the Construct Map Fortifications basecamp mission will be available.
-`"recruiting"` | after this upgrade mission is complete, the Recruit Companions basecamp mission will be available.
-`"scouting"` | after this upgrade mission is complete, the Scout Mission basecamp mission will be available.
-`"patrolling"` | after this upgrade mission is complete, the Combat Patrol basecamp mission will be available.
-`"dismantling"` | after this upgrade mission is complete, the Chop Shop basecamp mission will be available.
-`"farming"` | after this upgrade mission is complete, the Plow Fields, Plant Fields, Fertilize Fields, and Harvest Fields basecamp missions will be available.
-
-`blueprint_provides` can also be used to name objects from `recipe_group.json`. The recipes will be craftable by NPCs at that expansion, allowing the creation of custom recipes that can be performed exclusively at faction camps.
+
+blueprint requires, blueprint provides, and blueprint exlcudes are abstract concepts or flags that
+an upgrade mission requires to start, or that are provided by a previous upgrade mission to satisfy
+the blueprint requirements of a current upgrade mission, or that prevent an upgrade mission from
+being available. Each one has an `"id"` and an `"amount"`. Multiple requires, provides, or excludes
+with the same `"id"` sum their `"amount"` if they're on the same basecamp expansion.
+
+Every upgrade mission has its recipe `"result"` as a blueprint_provides and a blueprint_excludes, so
+upgrade missions will automatically prevent themselves from being repeatable.
+
+These are arbitrary strings and can be used to control the branching of the upgrade paths. However,
+some strings have meaning within the basecamp code:
+
+| provides `"id"` | meaning |
+| ---------------- | ------------------------------------------------------------------------------------------------------------------------------------------------ |
+| `"bed"` | every 2 `"amount"`' of `"bed"` allows another expansion in the camp, to a maximum of 8, not include the camp center. |
+| `"tool_storage"` | after this upgrade mission is complete, the Store Tools mission will be available |
+| . | |
+| `"radio"` | after this upgrade mission is complete, two way radios communicating to the camp have extended range. |
+| `"pantry"` | after this upgrade mission is complete, the Distribute Food mission is more efficient when dealing with short term spoilage items. |
+| `"gathering"` | after this upgrade mission is complete, the Gather Materials, Distribute Food, and Reset Sort Points basecamp missions will be available. |
+| `"firewood"` | after this upgrade mission is complete, the Gather Firewood basecamp mission will be available. |
+| `"sorting"` | after this upgrade mission is complete, the Menial Labor basecamp mission will be available. |
+| `"logging"` | after this upgrade mission is complete, the Cut Logs and Clear a Forest basecamp missions will be available. |
+| `"relaying"` | after this upgrade mission is complete, the Setup Hide Site and Relay Hide Site basecamp missions will be available. |
+| `"foraging"` | after this upgrade mission is complete, the Forage for Plants basecamp mission will be available. |
+| `"trapping"` | after this upgrade mission is complete, the Trap Small Game basecamp mission will be available. |
+| `"hunting"` | after this upgrade mission is complete, the Hunt Large Animals basecamp mission will be available. |
+| `"walls"` | after this upgrade mission is complete, the Construct Map Fortifications basecamp mission will be available. |
+| `"recruiting"` | after this upgrade mission is complete, the Recruit Companions basecamp mission will be available. |
+| `"scouting"` | after this upgrade mission is complete, the Scout Mission basecamp mission will be available. |
+| `"patrolling"` | after this upgrade mission is complete, the Combat Patrol basecamp mission will be available. |
+| `"dismantling"` | after this upgrade mission is complete, the Chop Shop basecamp mission will be available. |
+| `"farming"` | after this upgrade mission is complete, the Plow Fields, Plant Fields, Fertilize Fields, and Harvest Fields basecamp missions will be available. |
+
+`blueprint_provides` can also be used to name objects from `recipe_group.json`. The recipes will be
+craftable by NPCs at that expansion, allowing the creation of custom recipes that can be performed
+exclusively at faction camps.
### Sample recipe JSON
+
```JSON
- {
- "type": "recipe",
- "result": "faction_base_camp_8",
- "description": "We need to expand our base to include basic dining facilities.",
- "category": "CC_BUILDING",
- "subcategory": "CSC_BUILDING_BASES",
- "skill_used": "fabrication",
- "difficulty": 5,
- "autolearn": false,
- "never_learn": true,
- "comment": "2hrs*4wall + .5 hr*2tables + .5hr*4benches+ 1hrs pits = 12 hrs (12hrs on/off) 1 days total",
- "time": "1440 m",
- "construction_blueprint": "faction_base_field_camp_8",
- "blueprint_name": "basic central kitchen",
- "blueprint_resources": [ "fake_stove" ],
- "blueprint_provides": [ { "id": "trapping", "amount": 1 }, { "id": "hunting", "amount": 1 }, { "id": "walls", "amount": 1 }, { "id": "recruiting", "amount": 1 }
- ],
- "blueprint_requires": [ { "id": "faction_base_camp_6", "amount": 1 } ],
- "qualities": [ [ { "id": "DIG", "level": 1 } ], [ { "id": "SAW_M", "level": 1 } ], [ { "id": "HAMMER", "level": 2 } ] ],
- "components": [ [ [ "2x4", 28 ] ], [ [ "log", 16 ] ], [ [ "nail", 56 ] ], [ [ "stick", 24 ] ], [ [ "metal_tank", 1 ] ], [ [ "pipe", 1 ] ]
- ]
- },
+{
+ "type": "recipe",
+ "result": "faction_base_camp_8",
+ "description": "We need to expand our base to include basic dining facilities.",
+ "category": "CC_BUILDING",
+ "subcategory": "CSC_BUILDING_BASES",
+ "skill_used": "fabrication",
+ "difficulty": 5,
+ "autolearn": false,
+ "never_learn": true,
+ "comment": "2hrs*4wall + .5 hr*2tables + .5hr*4benches+ 1hrs pits = 12 hrs (12hrs on/off) 1 days total",
+ "time": "1440 m",
+ "construction_blueprint": "faction_base_field_camp_8",
+ "blueprint_name": "basic central kitchen",
+ "blueprint_resources": [ "fake_stove" ],
+ "blueprint_provides": [ { "id": "trapping", "amount": 1 }, { "id": "hunting", "amount": 1 }, { "id": "walls", "amount": 1 }, { "id": "recruiting", "amount": 1 }
+ ],
+ "blueprint_requires": [ { "id": "faction_base_camp_6", "amount": 1 } ],
+ "qualities": [ [ { "id": "DIG", "level": 1 } ], [ { "id": "SAW_M", "level": 1 } ], [ { "id": "HAMMER", "level": 2 } ] ],
+ "components": [ [ [ "2x4", 28 ] ], [ [ "log", 16 ] ], [ [ "nail", 56 ] ], [ [ "stick", 24 ] ], [ [ "metal_tank", 1 ] ], [ [ "pipe", 1 ] ]
+ ]
+},
```
-The `"faction_base_camp_8"` upgrade mission can be performed after `"faction_base_camp_6"` and enables the trapping, hunting, fortification, and recruiting basecamp missions. It adds a camp stove to the camp's inventory to represent the iron stove.
+The `"faction_base_camp_8"` upgrade mission can be performed after `"faction_base_camp_6"` and
+enables the trapping, hunting, fortification, and recruiting basecamp missions. It adds a camp stove
+to the camp's inventory to represent the iron stove.
-`"blueprint_autocalc": true` could be used instead of `components` to let the cost be calculated automatically from the mapgen\_update data.
+`"blueprint_autocalc": true` could be used instead of `components` to let the cost be calculated
+automatically from the mapgen\_update data.
## mapgen_update JSON
-These are standard mapgen_update JSON; see doc/MAPGEN.md for more details. Each one should change a localized portion of the camp map and should, as much as possible, be independent of any other pieces of mapgen_update for the basecamp upgrade path. Obviously, some bits are going to expand other bits, in which case their `"blueprint_requires"` should include the `"blueprint_provides"` of the previous upgrade missions.
+These are standard mapgen_update JSON; see doc/MAPGEN.md for more details. Each one should change a
+localized portion of the camp map and should, as much as possible, be independent of any other
+pieces of mapgen_update for the basecamp upgrade path. Obviously, some bits are going to expand
+other bits, in which case their `"blueprint_requires"` should include the `"blueprint_provides"` of
+the previous upgrade missions.
### Sample mapgen_update JSON
```json
- {
- "type": "mapgen",
- "update_mapgen_id": "faction_base_field_camp_7",
- "method": "json",
- "object": { "place_nested": [ { "chunks": [ "basecamp_large_tent_east" ], "x": 2, "y": 10 } ] }
- },
- {
- "type": "mapgen",
- "method": "json",
- "nested_mapgen_id": "basecamp_large_tent_east",
- "object": {
- "mapgensize": [ 5, 5 ],
- "rows": [
- "WWWWW",
- "Wbb$W",
- "Wr;;D",
- "Wbb$W",
- "WWWWW"
- ],
- "palettes": [ "acidia_camp_palette" ]
- }
- },
+{
+ "type": "mapgen",
+ "update_mapgen_id": "faction_base_field_camp_7",
+ "method": "json",
+ "object": { "place_nested": [ { "chunks": [ "basecamp_large_tent_east" ], "x": 2, "y": 10 } ] }
+},
+{
+ "type": "mapgen",
+ "method": "json",
+ "nested_mapgen_id": "basecamp_large_tent_east",
+ "object": {
+ "mapgensize": [ 5, 5 ],
+ "rows": [
+ "WWWWW",
+ "Wbb$W",
+ "Wr;;D",
+ "Wbb$W",
+ "WWWWW"
+ ],
+ "palettes": [ "acidia_camp_palette" ]
+ }
+},
```
-This mapgen_update places a large tent in the west central portion of the camp. The `"place_nested"` references a standard map used in the primitive field camp.
+This mapgen_update places a large tent in the west central portion of the camp. The `"place_nested"`
+references a standard map used in the primitive field camp.
## Recipe groups
-Recipe groups serve two purposes: they indicate what recipes can produced by the camp after an upgrade mission is completed, and they indicate what upgrade paths are available and where camps can be placed.
+
+Recipe groups serve two purposes: they indicate what recipes can produced by the camp after an
+upgrade mission is completed, and they indicate what upgrade paths are available and where camps can
+be placed.
### Upgrade Paths and Expansions
-There are two special recipe groups, `"all_faction_base_types"` and `"all_faction_base_expansions"`. They both look like this:
+
+There are two special recipe groups, `"all_faction_base_types"` and `"all_faction_base_expansions"`.
+They both look like this:
+
```json
- {
- "type": "recipe_group",
- "name": "all_faction_base_expansions",
- "building_type": "NONE",
- "recipes": [
- { "id": "faction_base_farm_0", "description": "Farm", "om_terrains": [ "field" ] },
- { "id": "faction_base_garage_0", "description": "Garage", "om_terrains": [ "field" ] },
- { "id": "faction_base_kitchen_0", "description": "Kitchen", "om_terrains": [ "field" ] },
- { "id": "faction_base_blacksmith_0", "description": "Blacksmith Shop", "om_terrains": [ "field" ] }
- ]
- },
+{
+ "type": "recipe_group",
+ "name": "all_faction_base_expansions",
+ "building_type": "NONE",
+ "recipes": [
+ { "id": "faction_base_farm_0", "description": "Farm", "om_terrains": [ "field" ] },
+ { "id": "faction_base_garage_0", "description": "Garage", "om_terrains": [ "field" ] },
+ { "id": "faction_base_kitchen_0", "description": "Kitchen", "om_terrains": [ "field" ] },
+ { "id": "faction_base_blacksmith_0", "description": "Blacksmith Shop", "om_terrains": [ "field" ] }
+ ]
+},
```
-Each entry in the `"recipes"` array must be a dictionary with the `"id"`, `"description"`, and `"om_terrains"` fields. `"id"` is the recipe `"id"` of the recipe that starts that basecamp or basecamp expansion upgrade path. `"description"` is a short name of the basecamp or basecamp expansion. `"om_terrains"` is a list of overmap terrain ids which can be used as the basis for the basecamp or basecamp expansion.
+Each entry in the `"recipes"` array must be a dictionary with the `"id"`, `"description"`, and
+`"om_terrains"` fields. `"id"` is the recipe `"id"` of the recipe that starts that basecamp or
+basecamp expansion upgrade path. `"description"` is a short name of the basecamp or basecamp
+expansion. `"om_terrains"` is a list of overmap terrain ids which can be used as the basis for the
+basecamp or basecamp expansion.
-All recipes that start an upgrade path or expansion should have a blueprint requirement that can never be met, such as "not_an_upgrade", to prevent them from showing up as available upgrades.
+All recipes that start an upgrade path or expansion should have a blueprint requirement that can
+never be met, such as "not_an_upgrade", to prevent them from showing up as available upgrades.
-If the player attempts to start a basecamp on an overmap terrain that has two or more valid basecamp expansion paths, she will allowed to choose which path to start.
+If the player attempts to start a basecamp on an overmap terrain that has two or more valid basecamp
+expansion paths, she will allowed to choose which path to start.
## Sample basecamp upgrade path
The primitive field camp has the following upgrade path:
+
1. `"faction_base_camp_0"` - the initial camp survey and the bulletin board.
2. `"faction_base_camp_1"` - setting up the northeast tent
3. `"faction_base_camp_2"` - digging the fire pit and constructing a bed
4. `"faction_base_camp_3"` - adding the bookshelves to store stuff
5. `"faction_base_camp_4"` - adding the second bed to the tent
-6. any of:
-`"faction_base_camp_5"`, `"faction_base_camp_7"`, `"faction_base_camp_9"`, `"faction_base_camp_15"`, `"faction_base_camp_18"` - add additional tents with beds in any order.
-or `"faction_base_camp_6"` - build the central kitchen. `"faction_base_camp_8"`, `"faction_base_camp_10"`, `"faction_base_camp_11"`, `"faction_base_camp_16"` must be built in that sequence afterwards, though interleaved with the other optional upgrade missions after `"faction_base_camp_4"`.
-or `"faction_base_camp_12"` - the central camp well.
-or `"faction_base_camp_13"` - building the camp's outer wall, which can be followed by `"faction_base_camp_14"` to complete the wall.
-or `"faction_base_camp_19"` - building the camp's radio tower, which can be followed by `"faction_base_camp_20"` to build the radio tower console and make it operational.
-7. `"faction_base_camp_17"` - adding better doors to camp wall and the central building must be built after `"faction_base_camp_14"` and `"faction_base_camp_16"`.
-
-After setting up the first tent, the player has a lot of options: they can build additional tents to enable expansions, they can build the well for water, they can build as much or as little of the central kitchen as they desire, or add the wall to give the camp defenses, or the radio tower to improve the range of their two-way radios.
+6. any of: `"faction_base_camp_5"`, `"faction_base_camp_7"`, `"faction_base_camp_9"`,
+ `"faction_base_camp_15"`, `"faction_base_camp_18"` - add additional tents with beds in any order.
+ or `"faction_base_camp_6"` - build the central kitchen. `"faction_base_camp_8"`,
+ `"faction_base_camp_10"`, `"faction_base_camp_11"`, `"faction_base_camp_16"` must be built in
+ that sequence afterwards, though interleaved with the other optional upgrade missions after
+ `"faction_base_camp_4"`. or `"faction_base_camp_12"` - the central camp well. or
+ `"faction_base_camp_13"` - building the camp's outer wall, which can be followed by
+ `"faction_base_camp_14"` to complete the wall. or `"faction_base_camp_19"` - building the camp's
+ radio tower, which can be followed by `"faction_base_camp_20"` to build the radio tower console
+ and make it operational.
+7. `"faction_base_camp_17"` - adding better doors to camp wall and the central building must be
+ built after `"faction_base_camp_14"` and `"faction_base_camp_16"`.
+
+After setting up the first tent, the player has a lot of options: they can build additional tents to
+enable expansions, they can build the well for water, they can build as much or as little of the
+central kitchen as they desire, or add the wall to give the camp defenses, or the radio tower to
+improve the range of their two-way radios.
## Modular Basecamp conventions
-The modular basecamp is a structure for designing basecamp upgrade paths. You don't have to use it, but elements of the design might get more code support in the future.
+
+The modular basecamp is a structure for designing basecamp upgrade paths. You don't have to use it,
+but elements of the design might get more code support in the future.
### Layout
-A modular camp is laid out on a 24x24 overmap tile. The outer 3 map squares on each side are reserved for fortifications and movement corridors, and the inner 18x18 map squares are divided into a 3x3 grid of 6x6 areas.
-
-Area | upper left position | lower right position
--- | -- | --
-northwest | 3, 3 | 8, 8
-north | 9, 3 | 14, 8
-northeast | 15, 3 | 20, 8
-west | 3, 9 | 8, 14
-center | 9, 9 | 14, 14
-east | 15, 9, | 20, 14
-southwest | 3, 15 | 8, 20
-south | 9, 15 | 14, 20
-southeast | 15, 15 | 20, 20
+
+A modular camp is laid out on a 24x24 overmap tile. The outer 3 map squares on each side are
+reserved for fortifications and movement corridors, and the inner 18x18 map squares are divided into
+a 3x3 grid of 6x6 areas.
+
+| Area | upper left position | lower right position |
+| --------- | ------------------- | -------------------- |
+| northwest | 3, 3 | 8, 8 |
+| north | 9, 3 | 14, 8 |
+| northeast | 15, 3 | 20, 8 |
+| west | 3, 9 | 8, 14 |
+| center | 9, 9 | 14, 14 |
+| east | 15, 9, | 20, 14 |
+| southwest | 3, 15 | 8, 20 |
+| south | 9, 15 | 14, 20 |
+| southeast | 15, 15 | 20, 20 |
Ideally, nested mapgen chunks should fit entirely into a 6x6 area.
### Naming scheme
-Modular bases use the following naming scheme for recipes. Each element is separated by an `_`
-
-* `"faction_base"` <-- encouraged for all basecamp recipes, _and_
-* `"modular"` <-- indicates a modular construction, _and_
-* overmap terrain id <-- the base terrain that this modular base is built upon, _and_
-* DESCRIPTOR <-- arbitrary string that describes what is being built. For buildings, "room" is used to mean a construction that is intended to be part of a larger building and might share walls with other parts of the building in adjacent areas; "shack" means a free standing building. Numbers indicate stages of construction, with 4 usually meaning a complete structure, with walls and roof, _and_
-* MATERIAL <-- the type of material used in construction, such as metal, wood, brick, or wad (short for wattle-and-daub), _and_
-* AREA <-- the area in the 3x3 grid of the modular camp layout.
-blueprint keywords follow a similar scheme, but `"faction_base_modular"` is collapsed into `"fbm"` and the overmap terrain id is collapsed into a short identifier. ie, `"fbmf"` is the keyword identifier for elements of the modular field base.
+Modular bases use the following naming scheme for recipes. Each element is separated by an `_`
+
+- `"faction_base"` <-- encouraged for all basecamp recipes, _and_
+- `"modular"` <-- indicates a modular construction, _and_
+- overmap terrain id <-- the base terrain that this modular base is built upon, _and_
+- DESCRIPTOR <-- arbitrary string that describes what is being built. For buildings, "room" is used
+ to mean a construction that is intended to be part of a larger building and might share walls with
+ other parts of the building in adjacent areas; "shack" means a free standing building. Numbers
+ indicate stages of construction, with 4 usually meaning a complete structure, with walls and roof,
+ _and_
+- MATERIAL <-- the type of material used in construction, such as metal, wood, brick, or wad (short
+ for wattle-and-daub), _and_
+- AREA <-- the area in the 3x3 grid of the modular camp layout.
+
+blueprint keywords follow a similar scheme, but `"faction_base_modular"` is collapsed into `"fbm"`
+and the overmap terrain id is collapsed into a short identifier. ie, `"fbmf"` is the keyword
+identifier for elements of the modular field base.
diff --git a/doc/CHANGELOG_GUIDELINES.md b/doc/CHANGELOG_GUIDELINES.md
index a88828ee2624..153791c55438 100644
--- a/doc/CHANGELOG_GUIDELINES.md
+++ b/doc/CHANGELOG_GUIDELINES.md
@@ -1,38 +1,39 @@
# Changelog Guidelines
-- categories used by `Pull Request Summary` in the [PR Template](../.github/pull_request_template.md).
-- these are only guidelines, not rules. choose the best category for your PR freely.
+- categories used by `Pull Request Summary` in the
+ [PR Template](../.github/pull_request_template.md).
+- these are only guidelines, not rules. choose the best category for your PR freely.
## Features
Changes related to player:
-- player can do something new (e.g: mutations, skills)
-- something new can happen to the player (e.g: new disease)
+- player can do something new (e.g: mutations, skills)
+- something new can happen to the player (e.g: new disease)
## Content
Added new contents like:
-- new monsters
-- new map areas
-- new items
-- new vehicles
-- new doohickeys
+- new monsters
+- new map areas
+- new items
+- new vehicles
+- new doohickeys
## Interface
UI/UX changes like:
-- adding/adjusting menus
-- change shortcuts
-- streamlining workflows
-- quality of life improvements
+- adding/adjusting menus
+- change shortcuts
+- streamlining workflows
+- quality of life improvements
## Mods
-- changes contained within a mod
-- extends what is capable within a mod
+- changes contained within a mod
+- extends what is capable within a mod
## Balance
@@ -50,18 +51,18 @@ Improvements to game performance.
Make development easier:
-- `C++` refactorings and overhaul
-- `Json` reorganizations
-- `docs/`, `.github/` and repository changes
-- other development tools
+- `C++` refactorings and overhaul
+- `Json` reorganizations
+- `docs/`, `.github/` and repository changes
+- other development tools
## Build
Improve build process:
-- more robust
-- easier to use
-- faster compile time
+- more robust
+- easier to use
+- faster compile time
## I18N
diff --git a/doc/CODE_STYLE.md b/doc/CODE_STYLE.md
index a9678abf60b6..b871a759a1d9 100644
--- a/doc/CODE_STYLE.md
+++ b/doc/CODE_STYLE.md
@@ -1,14 +1,17 @@
# Code Style Guide
-All of the C++ code in the project is styled, you should run any changes you make through astyle before pushing a pull request.
+All of the C++ code in the project is styled, you should run any changes you make through astyle
+before pushing a pull request.
-We are using astyle version 3.0.1. Version 3.1 should also work, though there are a few cases where they disagree and require annotation.
+We are using astyle version 3.0.1. Version 3.1 should also work, though there are a few cases where
+they disagree and require annotation.
Blocks of code can be passed through astyle to ensure that their formatting is correct:
astyle --style=1tbs --attach-inlines --indent=spaces=4 --align-pointer=name --max-code-length=100 --break-after-logical --indent-classes --indent-preprocessor --indent-switches --indent-col1-comments --min-conditional-indent=0 --pad-oper --unpad-paren --pad-paren-in --add-brackets --convert-tabs
-These options are mirrored in `.astylerc`, `doc/CODE_STYLE.txt` and `msvc-full-features/AStyleExtension-Cataclysm-BN.cfg`
+These options are mirrored in `.astylerc`, `doc/CODE_STYLE.txt` and
+`msvc-full-features/AStyleExtension-Cataclysm-BN.cfg`
For example, from `vi`, set marks a and b around the block, then:
@@ -20,7 +23,7 @@ See [DEVELOPER_TOOLING.md](DEVELOPER_TOOLING.md) for other environments.
Here's an example that illustrates the most common points of style:
-````c++
+```c++
int foo( int arg1, int *arg2 )
{
if( arg1 < 5 ) {
@@ -44,19 +47,22 @@ int foo( int arg1, int *arg2 )
}
return 0;
}
-````
+```
## Code Guidelines
These are less generic guidelines and more pain points we've stumbled across over time.
1. Use int for everything.
- 1. Long in particular is problematic since it is *not* a larger type than int on platforms we care about.
- 1. If you need an integral value larger than 32 bits, you don't. But if you do, use int64_t.
- 2. Uint is also a problem, it has poor behavior when overflowing and should be avoided for general purpose programming.
- 1. If you need binary data, unsigned int or unsigned char are fine, but you should probably use a std::bitset instead.
- 3. Float is to be avoided, but has valid uses.
+ 1. Long in particular is problematic since it is _not_ a larger type than int on platforms we
+ care about.
+ 1. If you need an integral value larger than 32 bits, you don't. But if you do, use int64_t.
+ 2. Uint is also a problem, it has poor behavior when overflowing and should be avoided for
+ general purpose programming.
+ 1. If you need binary data, unsigned int or unsigned char are fine, but you should probably
+ use a std::bitset instead.
+ 3. Float is to be avoided, but has valid uses.
2. Auto Almost Nothing. Auto has two uses, others should be avoided.
- 1. Aliasing for extremely long iterator or functional declarations.
- 2. Generic code support (but decltype is better).
+ 1. Aliasing for extremely long iterator or functional declarations.
+ 2. Generic code support (but decltype is better).
3. Avoid using declaration for standard namespaces.
diff --git a/doc/COLOR.md b/doc/COLOR.md
index 4babaf72cd25..442414a9f2ee 100644
--- a/doc/COLOR.md
+++ b/doc/COLOR.md
@@ -2,40 +2,48 @@
DDA is colorful game. You can use several foreground and background colors in various places:
-* map data (terrain and furniture);
-* item data;
-* text data;
-* etc.
+- map data (terrain and furniture);
+- item data;
+- text data;
+- etc.
-**Note:** Map data objects can only have one color-related node defined (either `color` or `bgcolor`).
+**Note:** Map data objects can only have one color-related node defined (either `color` or
+`bgcolor`).
## Color string format
-Whenever color is defined in JSON it should be defined in following format: `Prefix_Foreground_Background`.
+Whenever color is defined in JSON it should be defined in following format:
+`Prefix_Foreground_Background`.
`Prefix` can take one of following values:
-* `c_` - default color prefix (can be omitted);
-* `i_` - optional prefix which indicates that foreground color should be inverted (special rules will be applied to foreground and background colors);
-* `h_` - optional prefix which indicates that foreground color should be highlighted (special rules will be applied to foreground and background colors).
+- `c_` - default color prefix (can be omitted);
+- `i_` - optional prefix which indicates that foreground color should be inverted (special rules
+ will be applied to foreground and background colors);
+- `h_` - optional prefix which indicates that foreground color should be highlighted (special rules
+ will be applied to foreground and background colors).
`Foreground` - defines mandatory color of foreground/ink/font.
`Background` - defines optional color of background/paper.
-**Note:** Not all foreground + background pairs are defined by their full name. Use in-game Color manager to see all color names.
+**Note:** Not all foreground + background pairs are defined by their full name. Use in-game Color
+manager to see all color names.
-**Note:** If color was not found by its name, then `c_unset` is used for `Foreground` and `i_white` for `Background`.
+**Note:** If color was not found by its name, then `c_unset` is used for `Foreground` and `i_white`
+for `Background`.
## Examples of color strings
- `c_white` - `white` color (with default prefix `c_`);
-- `black` - `black` color (default prefix `c_` is omitted);
+- `black` - `black` color (default prefix `c_` is omitted);
- `i_red` - inverted `red` color;
- `dark_gray_white` - `dark_gray` foreground color with `white` background color;
- `light_gray_light_red` - `light_gray` foreground color with `light_red` background color;
-- `dkgray_red` - `dark_gray` foreground color with `red` background color (deprecated prefix `dk` instead of `dark_`);
-- `ltblue_red` - `light_blue` foreground color with `red` background color (deprecated prefix `lt` instead of `light_`).
+- `dkgray_red` - `dark_gray` foreground color with `red` background color (deprecated prefix `dk`
+ instead of `dark_`);
+- `ltblue_red` - `light_blue` foreground color with `red` background color (deprecated prefix `lt`
+ instead of `light_`).
## Color code
@@ -43,33 +51,35 @@ Color code is short string which defines color and can be used, for example, in
## Possible colors
-| Color (image) | Color name (dda) | Color name (curses) | Default R,G,B values | Color code | Notes |
-|:--------------------------------------------------------:|:------------------:|:-------------------:|:--------------------:|:----------:|:------------------------------------------------------:|
-| ![#000000](https://placehold.it/20/000000/000000?text=+) | `black` | `BLACK` | `0,0,0` | | |
-| ![#ff0000](https://placehold.it/20/ff0000/000000?text=+) | `red` | `RED` | `255,0,0` | `R` | |
-| ![#006e00](https://placehold.it/20/006e00/000000?text=+) | `green` | `GREEN` | `0,110,0` | `G` | |
-| ![#5c3317](https://placehold.it/20/5c3317/000000?text=+) | `brown` | `BROWN` | `92,51,23` | `br` | |
-| ![#0000c8](https://placehold.it/20/0000c8/000000?text=+) | `blue` | `BLUE` | `0,0,200` | `B` | |
-| ![#8b3a62](https://placehold.it/20/8b3a62/000000?text=+) | `magenta` or `pink`| `MAGENTA` | `139,58,98` | `P` | |
-| ![#0096b4](https://placehold.it/20/0096b4/000000?text=+) | `cyan` | `CYAN` | `0,150,180` | `C` | |
-| ![#969696](https://placehold.it/20/969696/000000?text=+) | `light_gray` | `GRAY` | `150,150,150` | `lg` | deprecated `lt` prefix can be used instead of `light_` |
-| ![#636363](https://placehold.it/20/636363/000000?text=+) | `dark_gray` | `DGRAY` | `99,99,99` | `dg` | deprecated `dk` prefix can be used instead of `dark_` |
-| ![#ff9696](https://placehold.it/20/ff9696/000000?text=+) | `light_red` | `LRED` | `255,150,150` | | deprecated `lt` prefix can be used instead of `light_` |
-| ![#00ff00](https://placehold.it/20/00ff00/000000?text=+) | `light_green` | `LGREEN` | `0,255,0` | `g` | deprecated `lt` prefix can be used instead of `light_` |
-| ![#ffff00](https://placehold.it/20/ffff00/000000?text=+) | `light_yellow` | `YELLOW` | `255,255,0` | | deprecated `lt` prefix can be used instead of `light_` |
-| ![#6464ff](https://placehold.it/20/6464ff/000000?text=+) | `light_blue` | `LBLUE` | `100,100,255` | `b` | deprecated `lt` prefix can be used instead of `light_` |
-| ![#fe00fe](https://placehold.it/20/fe00fe/000000?text=+) | `light_magenta` | `LMAGENTA` | `254,0,254` | `lm` | deprecated `lt` prefix can be used instead of `light_` |
-| ![#00f0ff](https://placehold.it/20/00f0ff/000000?text=+) | `light_cyan` | `LCYAN` | `0,240,255` | `c` | deprecated `lt` prefix can be used instead of `light_` |
-| ![#ffffff](https://placehold.it/20/ffffff/000000?text=+) | `white` | `WHITE` | `255,255,255` | `W` | |
-
-**Note:** Default RGB values are taken from file `\data\raw\colors.json`.
-**Note:** RGB values can be redefined in file `\config\base_colors.json`.
+| Color (image) | Color name (dda) | Color name (curses) | Default R,G,B values | Color code | Notes |
+| :------------------------------------------------------: | :-----------------: | :-----------------: | :------------------: | :--------: | :----------------------------------------------------: |
+| ![#000000](https://placehold.it/20/000000/000000?text=+) | `black` | `BLACK` | `0,0,0` | | |
+| ![#ff0000](https://placehold.it/20/ff0000/000000?text=+) | `red` | `RED` | `255,0,0` | `R` | |
+| ![#006e00](https://placehold.it/20/006e00/000000?text=+) | `green` | `GREEN` | `0,110,0` | `G` | |
+| ![#5c3317](https://placehold.it/20/5c3317/000000?text=+) | `brown` | `BROWN` | `92,51,23` | `br` | |
+| ![#0000c8](https://placehold.it/20/0000c8/000000?text=+) | `blue` | `BLUE` | `0,0,200` | `B` | |
+| ![#8b3a62](https://placehold.it/20/8b3a62/000000?text=+) | `magenta` or `pink` | `MAGENTA` | `139,58,98` | `P` | |
+| ![#0096b4](https://placehold.it/20/0096b4/000000?text=+) | `cyan` | `CYAN` | `0,150,180` | `C` | |
+| ![#969696](https://placehold.it/20/969696/000000?text=+) | `light_gray` | `GRAY` | `150,150,150` | `lg` | deprecated `lt` prefix can be used instead of `light_` |
+| ![#636363](https://placehold.it/20/636363/000000?text=+) | `dark_gray` | `DGRAY` | `99,99,99` | `dg` | deprecated `dk` prefix can be used instead of `dark_` |
+| ![#ff9696](https://placehold.it/20/ff9696/000000?text=+) | `light_red` | `LRED` | `255,150,150` | | deprecated `lt` prefix can be used instead of `light_` |
+| ![#00ff00](https://placehold.it/20/00ff00/000000?text=+) | `light_green` | `LGREEN` | `0,255,0` | `g` | deprecated `lt` prefix can be used instead of `light_` |
+| ![#ffff00](https://placehold.it/20/ffff00/000000?text=+) | `light_yellow` | `YELLOW` | `255,255,0` | | deprecated `lt` prefix can be used instead of `light_` |
+| ![#6464ff](https://placehold.it/20/6464ff/000000?text=+) | `light_blue` | `LBLUE` | `100,100,255` | `b` | deprecated `lt` prefix can be used instead of `light_` |
+| ![#fe00fe](https://placehold.it/20/fe00fe/000000?text=+) | `light_magenta` | `LMAGENTA` | `254,0,254` | `lm` | deprecated `lt` prefix can be used instead of `light_` |
+| ![#00f0ff](https://placehold.it/20/00f0ff/000000?text=+) | `light_cyan` | `LCYAN` | `0,240,255` | `c` | deprecated `lt` prefix can be used instead of `light_` |
+| ![#ffffff](https://placehold.it/20/ffffff/000000?text=+) | `white` | `WHITE` | `255,255,255` | `W` | |
+
+**Note:** Default RGB values are taken from file `\data\raw\colors.json`. **Note:** RGB values can
+be redefined in file `\config\base_colors.json`.
## Color rules
-There are two types of special color transformation which can affect both foreground and background color:
+There are two types of special color transformation which can affect both foreground and background
+color:
-* inversion;
-* highlight.
+- inversion;
+- highlight.
-**Note:** Color rules can be redefined (for example, `\data\raw\color_templates\no_bright_background.json`).
+**Note:** Color rules can be redefined (for example,
+`\data\raw\color_templates\no_bright_background.json`).
diff --git a/doc/COMPILING/COMPILER_SUPPORT.md b/doc/COMPILING/COMPILER_SUPPORT.md
index 22e1f79df182..8c9d8a3bf4e8 100644
--- a/doc/COMPILING/COMPILER_SUPPORT.md
+++ b/doc/COMPILING/COMPILER_SUPPORT.md
@@ -1,20 +1,18 @@
# Compilers Supported
-Our goal with compiler support is to make it as easy as possible for new
-contributors to get started with development of the game, while also using the
-newest compilers (and thus language standards) that we can.
+Our goal with compiler support is to make it as easy as possible for new contributors to get started
+with development of the game, while also using the newest compilers (and thus language standards)
+that we can.
-To that end, we aim to support gcc and clang up to the newest stable versions
-and back to those shipping in any supported version of a popular distribution
-or relevant development environment, including Ubuntu, Debian, MSYS, and XCode.
+To that end, we aim to support gcc and clang up to the newest stable versions and back to those
+shipping in any supported version of a popular distribution or relevant development environment,
+including Ubuntu, Debian, MSYS, and XCode.
-In practice, compiler support is often determined by what is covered in our
-automated testing.
+In practice, compiler support is often determined by what is covered in our automated testing.
-At time of writing, the oldest relevant compiler is gcc 5.3, shipped in the
-Ubuntu Xenial LTS release. Ubuntu is likely to remain the limiting factor, so
-the set of supported compilers should be reviewed when Xenial ceases support in
-2021-05.
+At time of writing, the oldest relevant compiler is gcc 5.3, shipped in the Ubuntu Xenial LTS
+release. Ubuntu is likely to remain the limiting factor, so the set of supported compilers should be
+reviewed when Xenial ceases support in 2021-05.
## GCC
@@ -26,16 +24,16 @@ We support and test Clang from version 14.
## Mingw and Mingw-w64
-We use Mingw for cross-compilation of Windows versions on Linux. gcc 5.4 is
-currently used both in the tests and for the Windows release binaries.
+We use Mingw for cross-compilation of Windows versions on Linux. gcc 5.4 is currently used both in
+the tests and for the Windows release binaries.
## MSYS2
-MSYS2 is [a way to build the project](COMPILING-MSYS.md) on Windows. It
-currently offers gcc at versions 7 or higher.
+MSYS2 is [a way to build the project](COMPILING-MSYS.md) on Windows. It currently offers gcc at
+versions 7 or higher.
-MSYS also provides clang. We don't currently support using clang here, but
-work to that end is welcome.
+MSYS also provides clang. We don't currently support using clang here, but work to that end is
+welcome.
## Visual Studio
diff --git a/doc/COMPILING/COMPILING-CMAKE.md b/doc/COMPILING/COMPILING-CMAKE.md
index d62a9883e16d..cb6c9e26d08a 100644
--- a/doc/COMPILING/COMPILING-CMAKE.md
+++ b/doc/COMPILING/COMPILING-CMAKE.md
@@ -1,355 +1,354 @@
# Disclaimer
-**WARNING**: CMake build is **NOT** official and should be used for *dev purposes ONLY*.
+**WARNING**: CMake build is **NOT** official and should be used for _dev purposes ONLY_.
For official way to build CataclysmBN See:
- * [COMPILING.md](COMPILING.md)
+- [COMPILING.md](COMPILING.md)
# Contents
-- [Disclaimer](#disclaimer)
-- [Contents](#contents)
-- [Prerequisites](#prerequisites)
-- [Build Environment](#build-environment)
- - [UNIX Environment](#unix-environment)
- - [Windows Environment (MSYS2)](#windows-environment-msys2)
-- [CMake Build](#cmake-build)
- - [CMake Build for MSYS2 (MinGW)](#cmake-build-for-msys2-mingw)
- - [CMake Build for Visual Studio / MSBuild](#cmake-build-for-visual-studio--msbuild)
-- [Build Options](#build-options)
- - [CMake specific options](#cmake-specific-options)
- - [CataclysmBN specific options](#cataclysmbn-specific-options)
-
# Prerequisites
-You'll need to have these libraries and their development headers installed in
-order to build CataclysmBN:
-
- * General
- * `cmake` >= 3.0.0
- * `gcc-libs`
- * `glibc`
- * `zlib`
- * `bzip2`
- * Curses
- * `ncurses`
- * Tiles
- * `SDL` >= 2.0.0
- * `SDL_image` >= 2.0.0 (with PNG and JPEG support)
- * `SDL_mixer` >= 2.0.0 (with Ogg Vorbis support)
- * `SDL_ttf` >= 2.0.0
- * `freetype`
- * Sound
- * `vorbis`
- * `libbz2`
- * `libz`
+You'll need to have these libraries and their development headers installed in order to build
+CataclysmBN:
+
+- General
+ - `cmake` >= 3.0.0
+ - `gcc-libs`
+ - `glibc`
+ - `zlib`
+ - `bzip2`
+- Curses
+ - `ncurses`
+- Tiles
+ - `SDL` >= 2.0.0
+ - `SDL_image` >= 2.0.0 (with PNG and JPEG support)
+ - `SDL_mixer` >= 2.0.0 (with Ogg Vorbis support)
+ - `SDL_ttf` >= 2.0.0
+ - `freetype`
+- Sound
+ - `vorbis`
+ - `libbz2`
+ - `libz`
In order to compile localization files, you'll also need `gettext` package.
# Build Environment
-You can obtain the source code tarball for the latest version from [git](https://github.com/cataclysmbnteam/Cataclysm-BN).
-
+You can obtain the source code tarball for the latest version from
+[git](https://github.com/cataclysmbnteam/Cataclysm-BN).
## UNIX Environment
Obtain packages specified above with your system package manager.
-
## Windows Environment (MSYS2)
- 1. Follow steps from here: https://msys2.github.io/
- 2. Install CataclysmBN build deps:
-
- ```
- pacman -S mingw-w64-i686-toolchain msys/git \
- mingw-w64-i686-cmake \
- mingw-w64-i686-SDL2_{image,mixer,ttf} \
- ncurses-devel \
- gettext
- ```
+1. Follow steps from here: https://msys2.github.io/
+2. Install CataclysmBN build deps:
- This should get your environment set up to build console and tiles version of windows.
+```
+pacman -S mingw-w64-i686-toolchain msys/git \
+ mingw-w64-i686-cmake \
+ mingw-w64-i686-SDL2_{image,mixer,ttf} \
+ ncurses-devel \
+ gettext
+```
- **NOTE**: This is only for 32bit builds. 64bit requires the x86_64 instead of the i686
- packages listed above:
+This should get your environment set up to build console and tiles version of windows.
- ```
- pacman -S mingw-w64-x86_64-toolchain msys/git \
- mingw-w64-x86_64-cmake \
- mingw-w64-x86_64-SDL2_{image,mixer,ttf} \
- ncurses-devel \
- gettext
- ```
+**NOTE**: This is only for 32bit builds. 64bit requires the x86_64 instead of the i686 packages
+listed above:
- **NOTE**: If you're trying to test with Jetbrains CLion, point to the cmake version in the
- msys32/mingw32 path instead of using the built in. This will let cmake detect the
- installed packages.
+```
+pacman -S mingw-w64-x86_64-toolchain msys/git \
+ mingw-w64-x86_64-cmake \
+ mingw-w64-x86_64-SDL2_{image,mixer,ttf} \
+ ncurses-devel \
+ gettext
+```
+**NOTE**: If you're trying to test with Jetbrains CLion, point to the cmake version in the
+msys32/mingw32 path instead of using the built in. This will let cmake detect the installed
+packages.
# CMake Build
- CMake has separate configuration and build steps. Configuration
- is done using CMake itself, and the actual build is done using either `make`
- (for Makefiles generator) or build-system agnostic `cmake --build . ` .
-
- There are two ways to build CataclysmBN with CMake: inside the source tree or
- outside of it. Out-of-source builds have the advantage that you can have
- multiple builds with different options from one source directory.
+CMake has separate configuration and build steps. Configuration is done using CMake itself, and the
+actual build is done using either `make` (for Makefiles generator) or build-system agnostic
+`cmake --build .` .
- **WARNING**: Inside the source tree build is **NOT** supported.
+There are two ways to build CataclysmBN with CMake: inside the source tree or outside of it.
+Out-of-source builds have the advantage that you can have multiple builds with different options
+from one source directory.
- To build CataclysmBN out of source:
+**WARNING**: Inside the source tree build is **NOT** supported.
- ```
- $ mkdir build && cd build
- $ cmake .. -DCMAKE_BUILD_TYPE=Release
- $ make
- ```
+To build CataclysmBN out of source:
-The above example creates a build directory inside the source directory, but that's not required - you can just as easily create it in a completely different location.
+```
+$ mkdir build && cd build
+ $ cmake .. -DCMAKE_BUILD_TYPE=Release
+ $ make
+```
- To install CataclysmBN after building (as root using su or sudo if necessary):
+The above example creates a build directory inside the source directory, but that's not required -
+you can just as easily create it in a completely different location.
- ```
- # make install
- ```
+To install CataclysmBN after building (as root using su or sudo if necessary):
- To change build options, you can either pass the options on the command line:
+```
+# make install
+```
- ```
- $ cmake .. -DOPTION_NAME=option_value
- ```
+To change build options, you can either pass the options on the command line:
- Or use either the `ccmake` or `cmake-gui` front-ends, which display all options
- and their cached values on a console and graphical UI, respectively.
+```
+$ cmake .. -DOPTION_NAME=option_value
+```
- ```
- $ ccmake ..
- $ cmake-gui ..
- ```
+Or use either the `ccmake` or `cmake-gui` front-ends, which display all options and their cached
+values on a console and graphical UI, respectively.
+```
+$ ccmake ..
+ $ cmake-gui ..
+```
## CMake Build for MSYS2 (MinGW)
- **NOTE**: For development purposes it is preferred to use `MinGW Win64 Shell` or `MinGW Win32 Shell` instead of `MSYS2 Shell`. In other case, you will need to set `PATH` variable manually.
-
- For Mingw,MSYS,MSYS2 you should set [Makefiles generator](https://cmake.org/cmake/help/v3.0/manual/cmake-generators.7.html) to "MSYS Makefiles".
- Setting it to "MinGW Makefiles" might work as well, but might also require additional hackery.
-
- Example:
-
- ```
- $ cd
- $ mkdir build
- $ cd build
- $ cmake .. -G "MSYS Makefiles"
- $ make # or $ cmake --build .
- ```
-
- The resulting binary will be placed inside the source code directory.
-
- Shared libraries:
-
- If you got `libgcc_s_dw2-1.dll not found` error you need to copy shared libraries
- to directory with CataclysmBN executables.
-
- **NOTE**: For `-DRELEASE=OFF` development builds, You can automate copy process with:
-
- ```
- $ make install
- ```
-
- However, it likely will fail b/c you have different build environment setup :)
-
- Currently known depends (Maybe outdated use ldd.exe to correct it for Your system)
-
- * MINGW deps:
- * `libwinpthread-1.dll`
- * `libgcc_s_dw2-1.dll`
- * `libstdc++-6.dll`
-
- * TILES deps:
- * `SDL2.dll`
- * `SDL2_ttf.dll`
- * `libfreetype-6.dll`
- * `libbz2-1.dll`
- * `libharfbuzz-0.dll`
- * `SDL2_image.dll`
- * `libpng16-16.dll`
- * `libjpeg-8.dll`
- * `libtiff-5.dll`
- * `libjbig-0.dll`
- * `liblzma-5.dll`
- * `libwebp-5.dll`
- * `zlib1.dll`
- * `libglib-2.0-0.dll`
-
- * SOUND deps:
- * `SDL2_mixer.dll`
- * `libFLAC-8.dll`
- * `libogg-0.dll`
- * `libfluidsynth-1.dll`
- * `libportaudio-2.dll`
- * `libsndfile-1.dll`
- * `libvorbis-0.dll`
- * `libvorbisenc-2.dll`
- * `libmodplug-1.dll`
- * `smpeg2.dll`
- * `libvorbisfile-3.dll`
+**NOTE**: For development purposes it is preferred to use `MinGW Win64 Shell` or `MinGW Win32 Shell`
+instead of `MSYS2 Shell`. In other case, you will need to set `PATH` variable manually.
+
+For Mingw,MSYS,MSYS2 you should set
+[Makefiles generator](https://cmake.org/cmake/help/v3.0/manual/cmake-generators.7.html) to "MSYS
+Makefiles". Setting it to "MinGW Makefiles" might work as well, but might also require additional
+hackery.
+
+Example:
+
+```
+$ cd
+ $ mkdir build
+ $ cd build
+ $ cmake .. -G "MSYS Makefiles"
+ $ make # or $ cmake --build .
+```
+
+The resulting binary will be placed inside the source code directory.
+
+Shared libraries:
+
+If you got `libgcc_s_dw2-1.dll not found` error you need to copy shared libraries to directory with
+CataclysmBN executables.
+**NOTE**: For `-DRELEASE=OFF` development builds, You can automate copy process with:
+
+```
+$ make install
+```
+
+However, it likely will fail b/c you have different build environment setup :)
+
+Currently known depends (Maybe outdated use ldd.exe to correct it for Your system)
+
+- MINGW deps:
+ - `libwinpthread-1.dll`
+ - `libgcc_s_dw2-1.dll`
+ - `libstdc++-6.dll`
+
+- TILES deps:
+ - `SDL2.dll`
+ - `SDL2_ttf.dll`
+ - `libfreetype-6.dll`
+ - `libbz2-1.dll`
+ - `libharfbuzz-0.dll`
+ - `SDL2_image.dll`
+ - `libpng16-16.dll`
+ - `libjpeg-8.dll`
+ - `libtiff-5.dll`
+ - `libjbig-0.dll`
+ - `liblzma-5.dll`
+ - `libwebp-5.dll`
+ - `zlib1.dll`
+ - `libglib-2.0-0.dll`
+
+- SOUND deps:
+ - `SDL2_mixer.dll`
+ - `libFLAC-8.dll`
+ - `libogg-0.dll`
+ - `libfluidsynth-1.dll`
+ - `libportaudio-2.dll`
+ - `libsndfile-1.dll`
+ - `libvorbis-0.dll`
+ - `libvorbisenc-2.dll`
+ - `libmodplug-1.dll`
+ - `smpeg2.dll`
+ - `libvorbisfile-3.dll`
## CMake Build for Visual Studio / MSBuild
-CMake can generate `.sln` and `.vcxproj` files used either by Visual Studio itself or by MSBuild command line compiler (if you don't want
-a full fledged IDE) and have more "native" binaries than what MSYS/Cygwin can provide.
+CMake can generate `.sln` and `.vcxproj` files used either by Visual Studio itself or by MSBuild
+command line compiler (if you don't want a full fledged IDE) and have more "native" binaries than
+what MSYS/Cygwin can provide.
-At the moment only a limited combination of options is supported (tiles only, no localizations, no backtrace).
+At the moment only a limited combination of options is supported (tiles only, no localizations, no
+backtrace).
Get the tools:
- * CMake from the official site - https://cmake.org/download/.
- * Microsoft compiler - https://visualstudio.microsoft.com/downloads/?q=build+tools , choose "Build Tools for Visual Studio 2017". When installing chose "Visual C++ Build Tools" options.
- * alternatively, you can get download and install the complete Visual Studio, but that's not required.
+
+- CMake from the official site - https://cmake.org/download/.
+- Microsoft compiler - https://visualstudio.microsoft.com/downloads/?q=build+tools , choose "Build
+ Tools for Visual Studio 2017". When installing chose "Visual C++ Build Tools" options.
+ - alternatively, you can get download and install the complete Visual Studio, but that's not
+ required.
Get the required libraries:
- * `SDL2` - https://www.libsdl.org/download-2.0.php (you need the "(Visual C++ 32/64-bit)" version. Same below)
- * `SDL2_ttf` - https://www.libsdl.org/projects/SDL_ttf/
- * `SDL2_image` - https://www.libsdl.org/projects/SDL_image/
- * `SDL2_mixer` (optional, for sound support) - https://www.libsdl.org/projects/SDL_mixer/
- * Unsupported (and unused in the following instructions) optional libs:
- * `ncurses` - ???
+
+- `SDL2` - https://www.libsdl.org/download-2.0.php (you need the "(Visual C++ 32/64-bit)" version.
+ Same below)
+- `SDL2_ttf` - https://www.libsdl.org/projects/SDL_ttf/
+- `SDL2_image` - https://www.libsdl.org/projects/SDL_image/
+- `SDL2_mixer` (optional, for sound support) - https://www.libsdl.org/projects/SDL_mixer/
+- Unsupported (and unused in the following instructions) optional libs:
+ - `ncurses` - ???
Unpack the archives with the libraries.
-Open windows command line (or powershell), set the environment variables to point to the libs above as follows (adjusting the paths as appropriate):
+Open windows command line (or powershell), set the environment variables to point to the libs above
+as follows (adjusting the paths as appropriate):
+
```
- > set SDL2DIR=C:\path\to\SDL2-devel-2.0.9-VC
- > set SDL2TTFDIR=C:\path\to\SDL2_ttf-devel-2.0.15-VC
- > set SDL2IMAGEDIR=C:\path\to\SDL2_image-devel-2.0.4-VC
- > set SDL2MIXERDIR=C:\path\to\SDL2_mixer-devel-2.0.4-VC
+> set SDL2DIR=C:\path\to\SDL2-devel-2.0.9-VC
+> set SDL2TTFDIR=C:\path\to\SDL2_ttf-devel-2.0.15-VC
+> set SDL2IMAGEDIR=C:\path\to\SDL2_image-devel-2.0.4-VC
+> set SDL2MIXERDIR=C:\path\to\SDL2_mixer-devel-2.0.4-VC
```
+
(for powershell the syntax is `$env:SDL2DIR="C:\path\to\SDL2-devel-2.0.9-VC"`).
Make a build directory and run cmake configuration step
+
```
- > cd
- > mkdir build
- > cd build
- > cmake .. -DTILES=ON -DLANGUAGES=none -DBACKTRACE=OFF -DSOUND=ON
+> cd
+> mkdir build
+> cd build
+> cmake .. -DTILES=ON -DLANGUAGES=none -DBACKTRACE=OFF -DSOUND=ON
```
-
Build!
+
```
- > cmake --build . -j 2 -- /p:Configuration=Release
+> cmake --build . -j 2 -- /p:Configuration=Release
```
- The `-j 2` flag controls build parallelism - you can omit it if you wish. The `/p:Configuration=Release` flag is passed directly to MSBuild and
- controls optimizations. If you omit it, the `Debug` configuration would be built instead. For powershell you'll need to have an extra ` -- ` after the first one.
-The resulting files will be put into a `Release` directory inside your source Cataclysm-BN folder. To make them run you'd need to first move them to the
-source Cataclysm-BN directory itself (so that the binary has access to the game data), and second put the required `.dll`s into the same folder -
-you can find those inside the directories for dev libraries under `lib/x86/` or `lib/x64/` (you likely need the `x86` ones even if you're on 64-bit machine).
+The `-j 2` flag controls build parallelism - you can omit it if you wish. The
+`/p:Configuration=Release` flag is passed directly to MSBuild and controls optimizations. If you
+omit it, the `Debug` configuration would be built instead. For powershell you'll need to have an
+extra `--` after the first one.
-The copying of dlls is a one-time task, but you'd need to move the binary out of `Release/` each time it's built. To automate it a bit, you can configure cmake and set the desired binaries destination directory with `-DCMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE=` option (and similar for `CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG`).
+The resulting files will be put into a `Release` directory inside your source Cataclysm-BN folder.
+To make them run you'd need to first move them to the source Cataclysm-BN directory itself (so that
+the binary has access to the game data), and second put the required `.dll`s into the same folder -
+you can find those inside the directories for dev libraries under `lib/x86/` or `lib/x64/` (you
+likely need the `x86` ones even if you're on 64-bit machine).
-Run the game. Should work.
+The copying of dlls is a one-time task, but you'd need to move the binary out of `Release/` each
+time it's built. To automate it a bit, you can configure cmake and set the desired binaries
+destination directory with `-DCMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE=` option (and similar for
+`CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG`).
+Run the game. Should work.
# Build Options
- A full list of options supported by CMake, you may either run the `ccmake`
- or `cmake-gui` front-ends, or run `cmake` and open the generated CMakeCache.txt
- from the build directory in a text editor.
-
- ```
- $ cmake -DOPTION_NAME1=option_value1 [-DOPTION_NAME2=option_value2 [...]]
- ```
+A full list of options supported by CMake, you may either run the `ccmake` or `cmake-gui`
+front-ends, or run `cmake` and open the generated CMakeCache.txt from the build directory in a text
+editor.
+```
+$ cmake -DOPTION_NAME1=option_value1 [-DOPTION_NAME2=option_value2 [...]]
+```
## CMake specific options
- * CMAKE_BUILD_TYPE=``
+- CMAKE_BUILD_TYPE=``
- Selects a specific build configuration when compiling. `release` produces
- the default, optimized (-Os) build for regular use. `debug` produces a
- slower and larger unoptimized (-O0) build with full debug symbols, which is
- often needed for obtaining detailed backtraces when reporting bugs.
+Selects a specific build configuration when compiling. `release` produces the default, optimized
+(-Os) build for regular use. `debug` produces a slower and larger unoptimized (-O0) build with full
+debug symbols, which is often needed for obtaining detailed backtraces when reporting bugs.
- **NOTE**: By default, CMake will produce `debug` builds unless a different
- configuration option is passed in the command line.
+**NOTE**: By default, CMake will produce `debug` builds unless a different configuration option is
+passed in the command line.
+- CMAKE_INSTALL_PREFIX=``
- * CMAKE_INSTALL_PREFIX=``
-
- Installation prefix for binaries, resources, and documentation files.
-
+Installation prefix for binaries, resources, and documentation files.
## CataclysmBN specific options
- * CURSES=``
-
- Build curses version.
+- CURSES=``
+Build curses version.
- * TILES=``
+- TILES=``
- Build graphical tileset version.
+Build graphical tileset version.
+- SOUND=``
- * SOUND=``
+Support for in-game sounds & music.
- Support for in-game sounds & music.
+- USE_HOME_DIR=``
+Use user's home directory for save files.
- * USE_HOME_DIR=``
+- LANGUAGES=``
- Use user's home directory for save files.
+Compile localization files for specified languages. Example:
+```
+-DLANGUAGES="cs;de;el;es_AR;es_ES"
+```
- * LANGUAGES=``
-
- Compile localization files for specified languages. Example:
- ```
- -DLANGUAGES="cs;de;el;es_AR;es_ES"
- ```
-
- Note that language files are only compiled automatically when building the
- `RELEASE` build type. For other build types, you need to add the `translations_compile`
- target to the `make` command, for example `make all translations_compile`.
+Note that language files are only compiled automatically when building the `RELEASE` build type. For
+other build types, you need to add the `translations_compile` target to the `make` command, for
+example `make all translations_compile`.
+- DYNAMIC_LINKING=``
- * DYNAMIC_LINKING=``
+Use dynamic linking. Or use static to remove MinGW dependency instead.
- Use dynamic linking. Or use static to remove MinGW dependency instead.
+- CUSTOM LINKER=``
- * CUSTOM LINKER=``
+Choose custom linkers such as [gold], [lld] or [mold].
- Choose custom linkers such as [gold], [lld] or [mold].
+- Choose ld if you don't use libbacktrace.
+- Choose mold if use libbacktrace. It's the fastest linker, outperforming gold by 24x.
- - Choose ld if you don't use libbacktrace.
- - Choose mold if use libbacktrace. It's the fastest linker, outperforming gold by 24x.
+[gold]: https://en.wikipedia.org/wiki/Gold_(linker)
+[lld]: https://lld.llvm.org
+[mold]: https://github.com/rui314/mold
- [gold]: https://en.wikipedia.org/wiki/Gold_(linker)
- [lld]: https://lld.llvm.org
- [mold]: https://github.com/rui314/mold
+- BACKTRACE=``
- * BACKTRACE=``
+On crash, print a backtrace to the console. Defaults to `ON` for debug builds.
- On crash, print a backtrace to the console. Defaults to `ON` for debug builds.
+- LIBBACKTRACE=``
- * LIBBACKTRACE=``
+Print backtrace with [libbacktrace]. This allows lld and mold to print backtrace, and is generally
+much faster.
- Print backtrace with [libbacktrace]. This allows lld and mold to print backtrace, and is generally much faster.
+[libbacktrace]: https://github.com/ianlancetaylor/libbacktrace
- [libbacktrace]: https://github.com/ianlancetaylor/libbacktrace
+- GIT_BINARY=``
- * GIT_BINARY=``
+Override default Git binary name or path.
- Override default Git binary name or path.
+So a CMake command for building Cataclysm-BN in release mode with tiles and sound support will look
+as follows, provided it is run in build directory located in the project.
- So a CMake command for building Cataclysm-BN in release mode with tiles and sound support will look as follows, provided it is run in build directory located in the project.
-```
+```sh
cmake ../ -DCMAKE_BUILD_TYPE=Release -DTILES=ON -DSOUND=ON
```
-
diff --git a/doc/COMPILING/COMPILING-CYGWIN.md b/doc/COMPILING/COMPILING-CYGWIN.md
index 9ccf4c6bb47d..c2540409c77f 100644
--- a/doc/COMPILING/COMPILING-CYGWIN.md
+++ b/doc/COMPILING/COMPILING-CYGWIN.md
@@ -1,22 +1,32 @@
# Compilation guide for 64 bit Windows (using Cygwin)
-This guide contains instructions for compiling Cataclysm-BN on Windows under Cygwin. **PLEASE NOTE:** These instructions *are not intended* to produce a redistributable copy of CBN. Please download the official builds from the website or [cross-compile from Linux](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/master/doc/COMPILING/COMPILING.md#cross-compile-to-windows-from-linux) if that is your intention.
+This guide contains instructions for compiling Cataclysm-BN on Windows under Cygwin. **PLEASE
+NOTE:** These instructions _are not intended_ to produce a redistributable copy of CBN. Please
+download the official builds from the website or
+[cross-compile from Linux](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/master/doc/COMPILING/COMPILING.md#cross-compile-to-windows-from-linux)
+if that is your intention.
-These instructions were written using 64-bit Windows 7 and the 64-bit version of Cygwin; the steps should be the same for other versions of Windows.
+These instructions were written using 64-bit Windows 7 and the 64-bit version of Cygwin; the steps
+should be the same for other versions of Windows.
-Due to slow environment setup and execution of the resulting binary, compilation using MSYS2 is preferred.
+Due to slow environment setup and execution of the resulting binary, compilation using MSYS2 is
+preferred.
## Prerequisites:
-* 64-bit version of Windows 7, 8, 8.1, or 10
-* NTFS partition with ~10 Gb free space (~2 Gb for Cygwin installation, ~3 Gb for repository and ~5 Gb for ccache)
-* 64-bit version of Cygwin
+- 64-bit version of Windows 7, 8, 8.1, or 10
+- NTFS partition with ~10 Gb free space (~2 Gb for Cygwin installation, ~3 Gb for repository and ~5
+ Gb for ccache)
+- 64-bit version of Cygwin
## Installation:
-1. Go to the [Cygwin homepage](https://cygwin.com/) and download the 64-bit installer (e.g. [setup-x86_64.exe](https://cygwin.com/setup-x86_64.exe)).
+1. Go to the [Cygwin homepage](https://cygwin.com/) and download the 64-bit installer (e.g.
+ [setup-x86_64.exe](https://cygwin.com/setup-x86_64.exe)).
-2. Run downloaded file and install Cygwin. Select `Install from Internet` and select the desired installation directory. It is suggested that you install into a dev-specific directory (e.g. `C:\dev\cygwin64`), but it's not strictly necessary. Install for all users.
+2. Run downloaded file and install Cygwin. Select `Install from Internet` and select the desired
+ installation directory. It is suggested that you install into a dev-specific directory (e.g.
+ `C:\dev\cygwin64`), but it's not strictly necessary. Install for all users.
3. Give the `\downloads\` folder as the local package directory (e.g. `C:\dev\cygwin64\downloads`).
@@ -24,9 +34,11 @@ Due to slow environment setup and execution of the resulting binary, compilation
5. On the next screen, select any mirror you like.
-6. Enter `wget` into the search box, expand the "Web" category and select the latest version in the drop-down (1.21.1-1 as of the time this guide was written).
+6. Enter `wget` into the search box, expand the "Web" category and select the latest version in the
+ drop-down (1.21.1-1 as of the time this guide was written).
-7. Confirm that wget is shown in the following screen by scrolling to the bottom. This is the full list of packages that Cygwin will download and install.
+7. Confirm that wget is shown in the following screen by scrolling to the bottom. This is the full
+ list of packages that Cygwin will download and install.
8. Retry any packages that throw errors.
@@ -49,13 +61,16 @@ chmod 755 /bin/apt-cyg
apt-cyg install astyle ccache gcc-g++ intltool git libSDL2_image-devel libSDL2_mixer-devel libSDL2_ttf-devel make xinit
```
-You will see messages saying packages are already installed, as well as Cygwin installing packages you didn't ask for; this is the result of Cygwin's package manager automatically resolving dependencies.
+You will see messages saying packages are already installed, as well as Cygwin installing packages
+you didn't ask for; this is the result of Cygwin's package manager automatically resolving
+dependencies.
## Cloning and compilation:
1. Clone the Cataclysm-BN repository with following command:
-**Note:** This will download the entire CBN repository and all of its history (3GB). If you're just testing, you should probably add `--depth=1` (~350MB).
+**Note:** This will download the entire CBN repository and all of its history (3GB). If you're just
+testing, you should probably add `--depth=1` (~350MB).
```bash
cd /cygdrive/c/dev
@@ -69,15 +84,19 @@ cd Cataclysm-BN
make -j$((`nproc`+0)) CCACHE=1 RELEASE=1 CYGWIN=1 DYNAMIC_LINKING=1 SDL=1 TILES=1 SOUND=1 LANGUAGES=all LINTJSON=0 ASTYLE=0 BACKTRACE=0 RUNTESTS=0
```
-You will receive warnings about unterminated character constants; they do not impact the compilation as far as this writer is aware.
+You will receive warnings about unterminated character constants; they do not impact the compilation
+as far as this writer is aware.
-**Note**: This will compile release version with Sound and Tiles support and all localization languages, skipping checks and tests and using ccache for faster build. You can use other switches, but `CYGWIN=1`, `DYNAMIC_LINKING=1` and `BACKTRACE=0` are required to compile without issues.
+**Note**: This will compile release version with Sound and Tiles support and all localization
+languages, skipping checks and tests and using ccache for faster build. You can use other switches,
+but `CYGWIN=1`, `DYNAMIC_LINKING=1` and `BACKTRACE=0` are required to compile without issues.
## Running:
1. Execute the XWin Server from the Start menu.
-2. When the icons appear in the system tray, right-click the one that looks like a black C (X Applications Menu.)
+2. When the icons appear in the system tray, right-click the one that looks like a black C (X
+ Applications Menu.)
3. Point to System Tools, then click UXTerm.
diff --git a/doc/COMPILING/COMPILING-MSYS.md b/doc/COMPILING/COMPILING-MSYS.md
index 9a12ae803d41..d40984040c13 100644
--- a/doc/COMPILING/COMPILING-MSYS.md
+++ b/doc/COMPILING/COMPILING-MSYS.md
@@ -1,14 +1,20 @@
# Compilation guide for 64-bit Windows (using MSYS2)
-This guide contains instructions for compiling Cataclysm-BN on Windows under MSYS2. **PLEASE NOTE:** These instructions *are not intended* to produce a redistributable copy of CBN. Please download the official builds from the website or [cross-compile from Linux](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/master/doc/COMPILING/COMPILING.md#cross-compile-to-windows-from-linux) if that is your intention.
+This guide contains instructions for compiling Cataclysm-BN on Windows under MSYS2. **PLEASE NOTE:**
+These instructions _are not intended_ to produce a redistributable copy of CBN. Please download the
+official builds from the website or
+[cross-compile from Linux](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/master/doc/COMPILING/COMPILING.md#cross-compile-to-windows-from-linux)
+if that is your intention.
-These instructions were written using 64-bit Windows 7 and the 64-bit version of MSYS2; the steps should be the same for other versions of Windows.
+These instructions were written using 64-bit Windows 7 and the 64-bit version of MSYS2; the steps
+should be the same for other versions of Windows.
## Prerequisites:
-* Windows 7, 8, 8.1, or 10
-* NTFS partition with ~10 Gb free space (~2 Gb for MSYS2 installation, ~3 Gb for repository and ~5 Gb for ccache)
-* 64-bit version of MSYS2
+- Windows 7, 8, 8.1, or 10
+- NTFS partition with ~10 Gb free space (~2 Gb for MSYS2 installation, ~3 Gb for repository and ~5
+ Gb for ccache)
+- 64-bit version of MSYS2
**Note:** Windows XP is unsupported!
@@ -16,7 +22,8 @@ These instructions were written using 64-bit Windows 7 and the 64-bit version of
1. Go to the [MSYS2 homepage](http://www.msys2.org/) and download the installer.
-2. Run the installer. It is suggested that you install to a dev-specific folder (C:\dev\msys64\ or similar), but it's not strictly necessary.
+2. Run the installer. It is suggested that you install to a dev-specific folder (C:\dev\msys64\ or
+ similar), but it's not strictly necessary.
3. After installation, run MSYS2 64bit now.
@@ -28,7 +35,10 @@ These instructions were written using 64-bit Windows 7 and the 64-bit version of
pacman -Syyu
```
-2. MSYS may inform you of a cygheap base mismatch and inform you a forked process died unexpectedly; these errors appear to be due to the nature of `pacman`'s upgrades and *may be safely ignored.* You will be prompted to close the terminal window; do so, then re-start using the MSYS2 MinGW 64-bit menu item.
+2. MSYS may inform you of a cygheap base mismatch and inform you a forked process died unexpectedly;
+ these errors appear to be due to the nature of `pacman`'s upgrades and _may be safely ignored._
+ You will be prompted to close the terminal window; do so, then re-start using the MSYS2 MinGW
+ 64-bit menu item.
3. Update remaining packages:
@@ -44,34 +54,35 @@ pacman -S git make mingw-w64-x86_64-{astyle,ccache,gcc,libmad,libwebp,pkg-config
5. Close MSYS2.
-6. Update path variables in the system-wide profile file (e.g. `C:\dev\msys64\etc\profile`) as following:
+6. Update path variables in the system-wide profile file (e.g. `C:\dev\msys64\etc\profile`) as
+ following:
- find lines:
```
- MSYS2_PATH="/usr/local/bin:/usr/bin:/bin"
- MANPATH='/usr/local/man:/usr/share/man:/usr/man:/share/man'
- INFOPATH='/usr/local/info:/usr/share/info:/usr/info:/share/info'
+MSYS2_PATH="/usr/local/bin:/usr/bin:/bin"
+MANPATH='/usr/local/man:/usr/share/man:/usr/man:/share/man'
+INFOPATH='/usr/local/info:/usr/share/info:/usr/info:/share/info'
```
and
```
- PKG_CONFIG_PATH="/usr/lib/pkgconfig:/usr/share/pkgconfig:/lib/pkgconfig"
+PKG_CONFIG_PATH="/usr/lib/pkgconfig:/usr/share/pkgconfig:/lib/pkgconfig"
```
- and replace them with:
```
- MSYS2_PATH="/usr/local/bin:/usr/bin:/bin:/mingw64/bin"
- MANPATH='/usr/local/man:/usr/share/man:/usr/man:/share/man:/mingw64/share/man'
- INFOPATH='/usr/local/info:/usr/share/info:/usr/info:/share/info:/mingw64/share/man'
+MSYS2_PATH="/usr/local/bin:/usr/bin:/bin:/mingw64/bin"
+MANPATH='/usr/local/man:/usr/share/man:/usr/man:/share/man:/mingw64/share/man'
+INFOPATH='/usr/local/info:/usr/share/info:/usr/info:/share/info:/mingw64/share/man'
```
and
```
- PKG_CONFIG_PATH="/usr/lib/pkgconfig:/usr/share/pkgconfig:/lib/pkgconfig:/mingw64/lib/pkgconfig:/mingw64/share/pkgconfig"
+PKG_CONFIG_PATH="/usr/lib/pkgconfig:/usr/share/pkgconfig:/lib/pkgconfig:/mingw64/lib/pkgconfig:/mingw64/share/pkgconfig"
```
## Cloning and compilation:
@@ -84,7 +95,8 @@ git clone https://github.com/cataclysmbnteam/Cataclysm-BN.git
cd Cataclysm-BN
```
-**Note:** This will download the entire CBN repository and all of its history (3GB). If you're just testing, you should probably add `--depth=1` (~350MB).
+**Note:** This will download the entire CBN repository and all of its history (3GB). If you're just
+testing, you should probably add `--depth=1` (~350MB).
2. Compile with following command line:
@@ -92,9 +104,12 @@ cd Cataclysm-BN
make -j$((`nproc`+0)) CCACHE=1 RELEASE=1 MSYS2=1 DYNAMIC_LINKING=1 SDL=1 TILES=1 SOUND=1 LANGUAGES=all LINTJSON=0 ASTYLE=0 RUNTESTS=0
```
-You will receive warnings about unterminated character constants; they do not impact the compilation as far as this writer is aware.
+You will receive warnings about unterminated character constants; they do not impact the compilation
+as far as this writer is aware.
-**Note**: This will compile a release version with Sound and Tiles support and all localization languages, skipping checks and tests, and using ccache for build acceleration. You can use other switches, but `MSYS2=1` and `DYNAMIC_LINKING=1` are required to compile without issues.
+**Note**: This will compile a release version with Sound and Tiles support and all localization
+languages, skipping checks and tests, and using ccache for build acceleration. You can use other
+switches, but `MSYS2=1` and `DYNAMIC_LINKING=1` are required to compile without issues.
## Running:
@@ -104,4 +119,6 @@ You will receive warnings about unterminated character constants; they do not im
./cataclysm-tiles
```
-**Note:** If you want to run the compiled executable outside of MSYS2, you will also need to update your user or system `PATH` variable with the path to MSYS2's runtime binaries (e.g. `C:\dev\msys64\mingw64\bin`).
+**Note:** If you want to run the compiled executable outside of MSYS2, you will also need to update
+your user or system `PATH` variable with the path to MSYS2's runtime binaries (e.g.
+`C:\dev\msys64\mingw64\bin`).
diff --git a/doc/COMPILING/COMPILING-VS-VCPKG.md b/doc/COMPILING/COMPILING-VS-VCPKG.md
index d15b766f1e28..2b02b8fcd8ba 100644
--- a/doc/COMPILING/COMPILING-VS-VCPKG.md
+++ b/doc/COMPILING/COMPILING-VS-VCPKG.md
@@ -1,32 +1,45 @@
# Compilation guide for Windows (using Visual Studio and vcpkg)
-This guide contains steps required to allow compilation of Cataclysm-BN on Windows using Visual Studio and vcpkg.
+This guide contains steps required to allow compilation of Cataclysm-BN on Windows using Visual
+Studio and vcpkg.
-Steps from current guide were tested on Windows 10 and 11 (64 bit), Visual Studio 2019 and 2022 (64 bit), but should as well work with slight modifications for other versions of Windows and Visual Studio.
+Steps from current guide were tested on Windows 10 and 11 (64 bit), Visual Studio 2019 and 2022 (64
+bit), but should as well work with slight modifications for other versions of Windows and Visual
+Studio.
## Prerequisites:
-* Computer with modern Windows operating system installed (Windows 10 or 11), Windows 7 and 8.1 are not guaranteed to work;
-* NTFS partition with ~15 Gb free space (~10 Gb for Visual Studio, ~1 Gb for vcpkg installation, ~3 Gb for repository and ~1 Gb for build cache);
-* Git for Windows (installer can be downloaded from [Git homepage](https://git-scm.com/));
-* Visual Studio 2019 or 2022
- * **Note**: If you are using Visual Studio 2022, you must install the Visual Studio 2019 compilers to work around a vcpkg bug. In the Visual Studio Installer, select the 'Individual components' tab and search for / select the component that looks like 'MSVC v142 - VS 2019 C++ x64/x86 Build Tools'. See https://github.com/microsoft/vcpkg/issues/22287.
-* Latest version of vcpkg (see instructions on [vcpkg homepage](https://github.com/Microsoft/vcpkg)).
-* If you plan on contributing your changes to Bright Nights, you'll also have to install a code formatter, see [Code style](#code-style) section for more info.
+- Computer with modern Windows operating system installed (Windows 10 or 11), Windows 7 and 8.1 are
+ not guaranteed to work;
+- NTFS partition with ~15 Gb free space (~10 Gb for Visual Studio, ~1 Gb for vcpkg installation, ~3
+ Gb for repository and ~1 Gb for build cache);
+- Git for Windows (installer can be downloaded from [Git homepage](https://git-scm.com/));
+- Visual Studio 2019 or 2022
+ - **Note**: If you are using Visual Studio 2022, you must install the Visual Studio 2019 compilers
+ to work around a vcpkg bug. In the Visual Studio Installer, select the 'Individual components'
+ tab and search for / select the component that looks like 'MSVC v142 - VS 2019 C++ x64/x86 Build
+ Tools'. See https://github.com/microsoft/vcpkg/issues/22287.
+- Latest version of vcpkg (see instructions on
+ [vcpkg homepage](https://github.com/Microsoft/vcpkg)).
+- If you plan on contributing your changes to Bright Nights, you'll also have to install a code
+ formatter, see [Code style](#code-style) section for more info.
**Note:** Windows XP is unsupported!
## Installation and configuration:
-1. Install `Visual Studio` (installer can be downloaded from [Visual Studio homepage](https://visualstudio.microsoft.com/)).
+1. Install `Visual Studio` (installer can be downloaded from
+ [Visual Studio homepage](https://visualstudio.microsoft.com/)).
- Select the "Desktop development with C++" and "Game development with C++" workloads.
-2. Install `Git for Windows` (installer can be downloaded from [Git homepage](https://git-scm.com/)).
+2. Install `Git for Windows` (installer can be downloaded from
+ [Git homepage](https://git-scm.com/)).
3. Install and configure latest `vcpkg`:
-***WARNING: It is important that, wherever you decide to clone this repo, the path does not include whitespace. That is, `C:/dev/vcpkg` is acceptable, but `C:/dev test/vcpkg` is not.***
+_**WARNING: It is important that, wherever you decide to clone this repo, the path does not include
+whitespace. That is, `C:/dev/vcpkg` is acceptable, but `C:/dev test/vcpkg` is not.**_
```cmd
git clone https://github.com/Microsoft/vcpkg.git
@@ -39,36 +52,57 @@ vcpkg integrate install
1. Clone Cataclysm-BN repository with following command line:
-**Note:** This will download the entire CBN repository; 3+ GB of data. If you're just testing you should probably add `--depth=1`.
+**Note:** This will download the entire CBN repository; 3+ GB of data. If you're just testing you
+should probably add `--depth=1`.
```cmd
git clone https://github.com/cataclysmbnteam/Cataclysm-BN.git
cd Cataclysm-BN
```
-2. Open the provided solution (`msvc-full-features\Cataclysm-vcpkg-static.sln`) in `Visual Studio`, select configuration (`Release` is advised if you just want to compile, `Debug` is if you're planning on editing code) and platform (`x64` or `x86`) and build it. All necessary dependencies will be built and cached for future use by vcpkg automatically.
+2. Open the provided solution (`msvc-full-features\Cataclysm-vcpkg-static.sln`) in `Visual Studio`,
+ select configuration (`Release` is advised if you just want to compile, `Debug` is if you're
+ planning on editing code) and platform (`x64` or `x86`) and build it. All necessary dependencies
+ will be built and cached for future use by vcpkg automatically.
-3. Open the `Build > Configuration Manager` menu and adjust `Active solution configuration` and `Active solution platform` to match your intended target.
+3. Open the `Build > Configuration Manager` menu and adjust `Active solution configuration` and
+ `Active solution platform` to match your intended target.
-This will configure Visual Studio to compile the release version, with support for Sound, Tiles, and Localization (note, however, that language files themselves are not automatically compiled; this will be done later).
+This will configure Visual Studio to compile the release version, with support for Sound, Tiles, and
+Localization (note, however, that language files themselves are not automatically compiled; this
+will be done later).
-4. Start the build process by selecting either `Build > Build Solution` or `Build > Build > Cataclysm-vcpkg-static`. The process may take a long period of time, so you'd better prepare a cup of coffee and some books in front of your computer :) The first build of each architecture will also download and install dependencies through vcpkg, which can take an especially long time.
+4. Start the build process by selecting either `Build > Build Solution` or
+ `Build > Build > Cataclysm-vcpkg-static`. The process may take a long period of time, so you'd
+ better prepare a cup of coffee and some books in front of your computer :) The first build of
+ each architecture will also download and install dependencies through vcpkg, which can take an
+ especially long time.
-5. If you want to launch the game directly from Visual Studio, make sure you have specified correct working directory as explained below. Otherwise, you'll only be able to launch the game from the file explorer.
+5. If you want to launch the game directly from Visual Studio, make sure you have specified correct
+ working directory as explained below. Otherwise, you'll only be able to launch the game from the
+ file explorer.
-6. If you need localization support, execute the bash script `lang/compile_mo.sh` inside Git Bash GUI just like on a UNIX-like system. This will compile the language files that were not automatically compiled in step 2 above.
+6. If you need localization support, execute the bash script `lang/compile_mo.sh` inside Git Bash
+ GUI just like on a UNIX-like system. This will compile the language files that were not
+ automatically compiled in step 2 above.
### Running from Visual Studio and debugging
-1. Ensure that the Cataclysm game binary project (`Cataclysm-vcpkg-static`) is the selected startup project (right click on it in Solution Explorer -> `Set as Startup Project`)
+1. Ensure that the Cataclysm game binary project (`Cataclysm-vcpkg-static`) is the selected startup
+ project (right click on it in Solution Explorer -> `Set as Startup Project`)
-2. Configure the working directory in the project properties to `$(ProjectDir)..` (right click on it in Solution Explorer -> `Properties` -> select `All Configurations` and `All Platforms` at the top -> `Debugging` -> `Working Directory`)
+2. Configure the working directory in the project properties to `$(ProjectDir)..` (right click on it
+ in Solution Explorer -> `Properties` -> select `All Configurations` and `All Platforms` at the
+ top -> `Debugging` -> `Working Directory`)
-3. Press the debug button (green right-facing triangle near the top, or use the appropriate shortcut, e.g. F5)
+3. Press the debug button (green right-facing triangle near the top, or use the appropriate
+ shortcut, e.g. F5)
-If you discover that after pressing the debug button in Visual Studio, Cataclysm just exits after launch with return code 1, that is because of the wrong working directory.
+If you discover that after pressing the debug button in Visual Studio, Cataclysm just exits after
+launch with return code 1, that is because of the wrong working directory.
### Debug vs Release builds
+
`Debug` builds run significantly slower than `Release` builds, but provide additional safety checks.
If you just want to build the executable and play the game, `Release` is advised.
@@ -76,26 +110,43 @@ If you just want to build the executable and play the game, `Release` is advised
If you plan on editing the code, `Debug` is advised.
If you have enough experience with C++ to know:
+
- under-the-hood differences between `Debug` and `Release`
- how `Release` optimizations may affect the debugger
- how to avoid undefined behavior in code
-Then you might want to use `Release` build all the time to speed up dev process, and disable optimizations on a file-by-file basis by adding
+Then you might want to use `Release` build all the time to speed up dev process, and disable
+optimizations on a file-by-file basis by adding
+
```c++
#pragma optimize("", off)
```
+
line at the top of the file.
### Running unit tests
-Ensure that the Cataclysm test binary project (`Cataclysm-test-vcpkg-static`) is the selected startup project, configure the working directory in the project properties to `$(ProjectDir)..`, and then press the debug button (or use the appropriate shortcut, e.g. F5). This will run all of the unit tests. Additional command line arguments may be configured in the project's command line arguments setting, or if you are using a compatible unit test runner (e.g. Resharper) you can run or debug individual tests from the unit test sessions.
+Ensure that the Cataclysm test binary project (`Cataclysm-test-vcpkg-static`) is the selected
+startup project, configure the working directory in the project properties to `$(ProjectDir)..`, and
+then press the debug button (or use the appropriate shortcut, e.g. F5). This will run all of the
+unit tests. Additional command line arguments may be configured in the project's command line
+arguments setting, or if you are using a compatible unit test runner (e.g. Resharper) you can run or
+debug individual tests from the unit test sessions.
### Code style
-We use `Artistic Style` source code formatter to keep the style of our C++ code consistent. While it's available as pre-built Windows executables, which you could install and run or configure to automatically format the code before commit, a much more convenient option for Visual Studio users is to install a specific extension, see ["Astyle extensions for Visual Studio" in doc/DEVELOPER_TOOLING.md](../DEVELOPER_TOOLING.md#astyle-extensions-for-visual-studio) for more info.
+We use `Artistic Style` source code formatter to keep the style of our C++ code consistent. While
+it's available as pre-built Windows executables, which you could install and run or configure to
+automatically format the code before commit, a much more convenient option for Visual Studio users
+is to install a specific extension, see
+["Astyle extensions for Visual Studio" in doc/DEVELOPER_TOOLING.md](../DEVELOPER_TOOLING.md#astyle-extensions-for-visual-studio)
+for more info.
-As of October 2022, the code style check is run automatically on each PR on GitHub, so if you forgot to style your changes you'll see the corresponsing check failing.
+As of October 2022, the code style check is run automatically on each PR on GitHub, so if you forgot
+to style your changes you'll see the corresponsing check failing.
### Make a distribution
-There is a batch script in `msvc-full-features` folder `distribute.bat`. It will create a sub folder `distribution` and copy all required files(eg. `data/`, `Cataclysm.exe` and dlls) into that folder. Then you can zip it and share the archive on the Internet.
+There is a batch script in `msvc-full-features` folder `distribute.bat`. It will create a sub folder
+`distribution` and copy all required files(eg. `data/`, `Cataclysm.exe` and dlls) into that folder.
+Then you can zip it and share the archive on the Internet.
diff --git a/doc/COMPILING/COMPILING.md b/doc/COMPILING/COMPILING.md
index 54596af3d589..9a73208bf9e3 100644
--- a/doc/COMPILING/COMPILING.md
+++ b/doc/COMPILING/COMPILING.md
@@ -46,26 +46,32 @@
- [Building with CYGWIN](#building-with-cygwin)
- [Building with Clang and MinGW64](#building-with-clang-and-mingw64)
- [BSDs](#bsds)
- - [Building on FreeBSD/amd64 10.1 with the system compiler](#building-on-freebsdamd64-101-with-the-system-compiler)
- - [Building ncurses version on FreeBSD/amd64 9.3 with GCC 4.8.4 from ports](#building-ncurses-version-on-freebsdamd64-93-with-gcc-484-from-ports)
- - [Building on OpenBSD/amd64 5.8 with GCC 4.9.2 from ports/packages](#building-on-openbsdamd64-58-with-gcc-492-from-portspackages)
- - [Building on NetBSD/amd64 7.0RC1 with the system compiler](#building-on-netbsdamd64-70rc1-with-the-system-compiler)
+ - [Building on FreeBSD/amd64 10.1 with the system compiler](#building-on-freebsdamd64-101-with-the-system-compiler)
+ - [Building ncurses version on FreeBSD/amd64 9.3 with GCC 4.8.4 from ports](#building-ncurses-version-on-freebsdamd64-93-with-gcc-484-from-ports)
+ - [Building on OpenBSD/amd64 5.8 with GCC 4.9.2 from ports/packages](#building-on-openbsdamd64-58-with-gcc-492-from-portspackages)
+ - [Building on NetBSD/amd64 7.0RC1 with the system compiler](#building-on-netbsdamd64-70rc1-with-the-system-compiler)
# General Linux Guide
-To build Cataclysm from source you will need at least a C++ compiler, some basic developer tools, and necessary build dependencies. The exact package names vary greatly from distro to distro, so this part of the guide is intended to give you higher-level understanding of the process.
+To build Cataclysm from source you will need at least a C++ compiler, some basic developer tools,
+and necessary build dependencies. The exact package names vary greatly from distro to distro, so
+this part of the guide is intended to give you higher-level understanding of the process.
## Compiler
You have three major choices here: GCC, Clang and MXE.
- * GCC is almost always the default on Linux systems so it's likely you already have it
- * Clang is usually faster than GCC, so it's worth installing if you plan to keep up with the latest experimentals
- * MXE is a cross-compiler, so of any importance only if you plan to compile for Windows on your Linux machine
+- GCC is almost always the default on Linux systems so it's likely you already have it
+- Clang is usually faster than GCC, so it's worth installing if you plan to keep up with the latest
+ experimentals
+- MXE is a cross-compiler, so of any importance only if you plan to compile for Windows on your
+ Linux machine
-(Note that your distro may have separate packages e.g. `gcc` only includes the C compiler and for C++ you'll need to install `g++`.)
+(Note that your distro may have separate packages e.g. `gcc` only includes the C compiler and for
+C++ you'll need to install `g++`.)
-Cataclysm is targeting C++14 standard and that means you'll need a compiler that supports it. You can easily check if your version of `g++` supports C++14 by running:
+Cataclysm is targeting C++14 standard and that means you'll need a compiler that supports it. You
+can easily check if your version of `g++` supports C++14 by running:
$ g++ --std=c++14
g++: fatal error: no input files
@@ -81,82 +87,97 @@ The general rule is the newer the compiler the better.
## Tools
-Most distros seem to package essential build tools as either a single package (Debian and derivatives have `build-essential`) or a package group (Arch has `base-devel`). You should use the above if available. Otherwise you'll at least need `make` and figure out the missing dependencies as you go (if any).
+Most distros seem to package essential build tools as either a single package (Debian and
+derivatives have `build-essential`) or a package group (Arch has `base-devel`). You should use the
+above if available. Otherwise you'll at least need `make` and figure out the missing dependencies as
+you go (if any).
Besides the essentials you will need `git`.
-If you plan on keeping up with experimentals you should also install `ccache`, which will considerably speed-up partial builds.
+If you plan on keeping up with experimentals you should also install `ccache`, which will
+considerably speed-up partial builds.
## Dependencies
-There are some general dependencies, optional dependencies and then specific dependencies for either curses or tiles builds. The exact package names again depend on the distro you're using, and whether your distro packages libraries and their development files separately (e.g. Debian and derivatives).
+There are some general dependencies, optional dependencies and then specific dependencies for either
+curses or tiles builds. The exact package names again depend on the distro you're using, and whether
+your distro packages libraries and their development files separately (e.g. Debian and derivatives).
Rough list based on building on Arch:
- * General: `gcc-libs`, `glibc`, `zlib`, `bzip2`
- * Optional: `intltool`
- * Curses: `ncurses`
- * Tiles: `sdl2`, `sdl2_image`, `sdl2_ttf`, `sdl2_mixer`, `freetype2`
+- General: `gcc-libs`, `glibc`, `zlib`, `bzip2`
+- Optional: `intltool`
+- Curses: `ncurses`
+- Tiles: `sdl2`, `sdl2_image`, `sdl2_ttf`, `sdl2_mixer`, `freetype2`
-E.g. for curses build on Debian and derivatives you'll also need `libncurses5-dev` or `libncursesw5-dev`.
+E.g. for curses build on Debian and derivatives you'll also need `libncurses5-dev` or
+`libncursesw5-dev`.
Note on optional dependencies:
- * `intltool` - for building localization files; if you plan to only use English you can skip it
+- `intltool` - for building localization files; if you plan to only use English you can skip it
-You should be able to figure out what you are missing by reading the compilation errors and/or the output of `ldd` for compiled binaries.
+You should be able to figure out what you are missing by reading the compilation errors and/or the
+output of `ldd` for compiled binaries.
## Make flags
Given you're building from source you have a number of choices to make:
- * `NATIVE=` - you should only care about this if you're cross-compiling
- * `RELEASE=1` - without this you'll get a debug build (see note below)
- * `LTO=1` - enables link-time optimization with GCC/Clang
- * `TILES=1` - with this you'll get the tiles version, without it the curses version
- * `SOUND=1` - if you want sound; this requires `TILES=1`
- * `LANGUAGES=` - specifies localizations. See details [here](#compiling-localization-files)
- * `CLANG=1` - use Clang instead of GCC
- * `CCACHE=1` - use ccache
- * `USE_LIBCXX=1` - use libc++ instead of libstdc++ with Clang (default on OS X)
+- `NATIVE=` - you should only care about this if you're cross-compiling
+- `RELEASE=1` - without this you'll get a debug build (see note below)
+- `LTO=1` - enables link-time optimization with GCC/Clang
+- `TILES=1` - with this you'll get the tiles version, without it the curses version
+- `SOUND=1` - if you want sound; this requires `TILES=1`
+- `LANGUAGES=` - specifies localizations. See details [here](#compiling-localization-files)
+- `CLANG=1` - use Clang instead of GCC
+- `CCACHE=1` - use ccache
+- `USE_LIBCXX=1` - use libc++ instead of libstdc++ with Clang (default on OS X)
There is a couple of other possible options - feel free to read the `Makefile`.
-If you have a multi-core computer you'd probably want to add `-jX` to the options, where `X` should roughly be twice the number of cores you have available.
+If you have a multi-core computer you'd probably want to add `-jX` to the options, where `X` should
+roughly be twice the number of cores you have available.
Example: `make -j4 CLANG=1 CCACHE=1 NATIVE=linux64 RELEASE=1 TILES=1`
-The above will build a tiles release explicitly for 64 bit Linux, using Clang and ccache and 4 parallel processes.
+The above will build a tiles release explicitly for 64 bit Linux, using Clang and ccache and 4
+parallel processes.
Example: `make -j2`
-The above will build a debug-enabled curses version for the architecture you are using, using GCC and 2 parallel processes.
+The above will build a debug-enabled curses version for the architecture you are using, using GCC
+and 2 parallel processes.
-**Note on debug**:
-You should probably always build with `RELEASE=1` unless you experience segfaults and are willing to provide stack traces.
+**Note on debug**: You should probably always build with `RELEASE=1` unless you experience segfaults
+and are willing to provide stack traces.
## Compiling localization files
By default, only English language is available, and it does not require localization file.
-If you want to compile files for specific languages, you should add `LANGUAGES=" [lang_id_2] [...]"` option to make command:
+If you want to compile files for specific languages, you should add
+`LANGUAGES=" [lang_id_2] [...]"` option to make command:
make LANGUAGES="zh_CN zh_TW"
-You can get the language ID from the filenames of `*.po` in `lang/po` directory or use `LANGUAGES="all"` to compile all available localizations.
+You can get the language ID from the filenames of `*.po` in `lang/po` directory or use
+`LANGUAGES="all"` to compile all available localizations.
# Debian
-Instructions for compiling on a Debian-based system. The package names here are valid for Ubuntu 12.10 and may or may not work on your system.
+Instructions for compiling on a Debian-based system. The package names here are valid for Ubuntu
+12.10 and may or may not work on your system.
-Building instructions, below, always assume you are running them from the Cataclysm:BN source directory.
+Building instructions, below, always assume you are running them from the Cataclysm:BN source
+directory.
## Linux (native) ncurses builds
Dependencies:
- * ncurses or ncursesw (for multi-byte locales)
- * build essentials
+- ncurses or ncursesw (for multi-byte locales)
+- build essentials
Install:
@@ -172,22 +193,25 @@ Run:
Dependencies:
- * SDL
- * SDL_ttf
- * freetype
- * build essentials
- * libsdl2-mixer-dev - Used if compiling with sound support.
+- SDL
+- SDL_ttf
+- freetype
+- build essentials
+- libsdl2-mixer-dev - Used if compiling with sound support.
Install:
sudo apt-get install libsdl2-dev libsdl2-ttf-dev libsdl2-image-dev libsdl2-mixer-dev libfreetype6-dev build-essential
check correct version of SDL2 is installed by running:
+
```sh
> sdl2-config --version
2.0.22
```
-using old version of SDL could result in [IME not working.](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/1497)
+
+using old version of SDL could result in
+[IME not working.](https://github.com/cataclysmbnteam/Cataclysm-BN/issues/1497)
### Building
@@ -199,16 +223,17 @@ A more comprehensive alternative is:
make -j2 TILES=1 SOUND=1 RELEASE=1 USE_HOME_DIR=1
-The -j2 flag means it will compile with two parallel processes. It can be omitted or changed to -j4 in a more modern processor. If there is no desire to have sound, those flags can also be omitted. The USE_HOME_DIR flag places the user files, like configurations and saves into the home folder, making It easier for backups, and can also be omitted.
-
-
+The -j2 flag means it will compile with two parallel processes. It can be omitted or changed to -j4
+in a more modern processor. If there is no desire to have sound, those flags can also be omitted.
+The USE_HOME_DIR flag places the user files, like configurations and saves into the home folder,
+making It easier for backups, and can also be omitted.
## Cross-compiling to linux 32-bit from linux 64-bit
Dependencies:
- * 32-bit toolchain
- * 32-bit ncursesw (compatible with both multi-byte and 8-bit locales)
+- 32-bit toolchain
+- 32-bit ncursesw (compatible with both multi-byte and 8-bit locales)
Install:
@@ -221,9 +246,11 @@ Run:
make NATIVE=linux32
## Cross-compile to Windows from Linux
+
To cross-compile to Windows from Linux, you will need [MXE](http://mxe.cc).
-These instructions were written for Ubuntu 20.04, but should be applicable to any Debian-based environment. Please adjust all package manager instructions to match your environment.
+These instructions were written for Ubuntu 20.04, but should be applicable to any Debian-based
+environment. Please adjust all package manager instructions to match your environment.
MXE can be either installed from MXE apt repository (much faster) or compiled from source.
@@ -234,7 +261,8 @@ MXE can be either installed from MXE apt repository (much faster) or compiled fr
sudo apt-get update
sudo apt-get install astyle bzip2 git make mxe-{i686,x86-64}-w64-mingw32.static-{sdl2,sdl2-ttf,sdl2-image,sdl2-mixer}
-If you are not planning on building for both 32-bit and 64-bit, you might want to adjust the last apt-get invocation to install only `i686` or `x86-64` packages.
+If you are not planning on building for both 32-bit and 64-bit, you might want to adjust the last
+apt-get invocation to install only `i686` or `x86-64` packages.
Edit your `~/.profile` as follows:
@@ -244,6 +272,7 @@ Edit your `~/.profile` as follows:
This is to ensure that the variables for the `make` command will not get reset after a power cycle.
### Installing MXE from source
+
Install [MXE requirements](http://mxe.cc/#requirements) and build dependencies:
sudo apt install astyle autoconf automake autopoint bash bison bzip2 cmake flex gettext git g++ gperf intltool libffi-dev libgdk-pixbuf2.0-dev libtool libltdl-dev libssl-dev libxml-parser-perl lzip make mingw-w64 openssl p7zip-full patch perl pkg-config python ruby scons sed unzip wget xz-utils g++-multilib libc6-dev-i386 libtool-bin
@@ -256,9 +285,11 @@ Clone MXE repo and build packages required for CBN:
cd mxe
make -j$((`nproc`+0)) MXE_TARGETS='x86_64-w64-mingw32.static i686-w64-mingw32.static' sdl2 sdl2_ttf sdl2_image sdl2_mixer
-Building all these packages from MXE might take a while, even on a fast computer. Be patient; the `-j` flag will take advantage of all your processor cores.
+Building all these packages from MXE might take a while, even on a fast computer. Be patient; the
+`-j` flag will take advantage of all your processor cores.
-If you are not planning on building for both 32-bit and 64-bit, you might want to adjust your MXE_TARGETS.
+If you are not planning on building for both 32-bit and 64-bit, you might want to adjust your
+MXE_TARGETS.
Edit your `~/.profile` as follows:
@@ -266,6 +297,7 @@ Edit your `~/.profile` as follows:
export PLATFORM_64="~/src/mxe/usr/bin/x86_64-w64-mingw32.static-"
This is to ensure that the variables for the `make` command will not get reset after a power cycle.
+
### Building (SDL)
Run one of the following commands based on your targeted environment:
@@ -275,29 +307,34 @@ Run one of the following commands based on your targeted environment:
## Cross-compile to Mac OS X from Linux
-The procedure is very much similar to cross-compilation to Windows from Linux.
-Tested on ubuntu 14.04 LTS but should work on other distros as well.
+The procedure is very much similar to cross-compilation to Windows from Linux. Tested on ubuntu
+14.04 LTS but should work on other distros as well.
+
+Please note that due to historical difficulties with cross-compilation errors, run-time
+optimizations are disabled for cross-compilation to Mac OS X targets. (`-O0` is specified as a
+compilation flag.) See
+[Pull Request #26564](https://github.com/CleverRaven/Cataclysm-DDA/pull/26564) for details.
-Please note that due to historical difficulties with cross-compilation errors, run-time optimizations are disabled for cross-compilation to Mac OS X targets. (`-O0` is specified as a compilation flag.) See [Pull Request #26564](https://github.com/CleverRaven/Cataclysm-DDA/pull/26564) for details.
### Dependencies
- * OSX cross-compiling toolchain [osxcross](https://github.com/tpoechtrager/osxcross)
+- OSX cross-compiling toolchain [osxcross](https://github.com/tpoechtrager/osxcross)
- * `genisoimage` and [libdmg-hfsplus](https://github.com/planetbeing/libdmg-hfsplus.git) to create dmg distributions
+- `genisoimage` and [libdmg-hfsplus](https://github.com/planetbeing/libdmg-hfsplus.git) to create
+ dmg distributions
Make sure that all dependency tools are in search `PATH` before compiling.
### Setup
To set up the compiling environment execute the following commands
-`git clone https://github.com/tpoechtrager/osxcross.git` to clone the toolchain
-`cd osxcross`
-`cp ~/MacOSX10.11.sdk.tar.bz2 ./tarballs/` copy prepared MacOSX SDK tarball on place. [Read more about it](https://github.com/tpoechtrager/osxcross/blob/master/README.md#packaging-the-sdk)
-`OSX_VERSION_MIN=10.7 ./build.sh to build everything`
-Note the targeted minimum supported version of OSX.
+`git clone https://github.com/tpoechtrager/osxcross.git` to clone the toolchain `cd osxcross`
+`cp ~/MacOSX10.11.sdk.tar.bz2 ./tarballs/` copy prepared MacOSX SDK tarball on place.
+[Read more about it](https://github.com/tpoechtrager/osxcross/blob/master/README.md#packaging-the-sdk)
+`OSX_VERSION_MIN=10.7 ./build.sh to build everything` Note the targeted minimum supported version of
+OSX.
-Have a prepackaged set of libs and frameworks in place, since compiling with `osxcross` built-in MacPorts is rather difficult and not supported at the moment.
-Your directory tree should look like:
+Have a prepackaged set of libs and frameworks in place, since compiling with `osxcross` built-in
+MacPorts is rather difficult and not supported at the moment. Your directory tree should look like:
~/
├── Frameworks
@@ -310,10 +347,9 @@ Your directory tree should look like:
├── include
└── lib
-Populated with respective frameworks, dylibs and headers.
-Tested with lib version libncurses.5.4.dylib for ncurses.
-These libs were obtained from `homebrew` binary distribution at OS X 10.11
-Frameworks were obtained from SDL official website as described in the next [section](#sdl)
+Populated with respective frameworks, dylibs and headers. Tested with lib version
+libncurses.5.4.dylib for ncurses. These libs were obtained from `homebrew` binary distribution at OS
+X 10.11 Frameworks were obtained from SDL official website as described in the next [section](#sdl)
### Building (SDL)
@@ -336,17 +372,24 @@ Make sure that `x86_64-apple-darwin15-clang++` is in `PATH` environment variable
## Cross-compile to Android from Linux
-The Android build uses [Gradle](https://gradle.org/) to compile the java and native C++ code, and is based heavily off SDL's [Android project template](https://hg.libsdl.org/SDL/file/f1084c419f33/android-project). See the official SDL documentation [README-android.md](https://hg.libsdl.org/SDL/file/f1084c419f33/docs/README-android.md) for further information.
+The Android build uses [Gradle](https://gradle.org/) to compile the java and native C++ code, and is
+based heavily off SDL's
+[Android project template](https://hg.libsdl.org/SDL/file/f1084c419f33/android-project). See the
+official SDL documentation
+[README-android.md](https://hg.libsdl.org/SDL/file/f1084c419f33/docs/README-android.md) for further
+information.
-The Gradle project lives in the repository under `android/`. You can build it via the command line or open it in [Android Studio](https://developer.android.com/studio/). For simplicity, it only builds the SDL version with all features enabled, including tiles, sound and localization.
+The Gradle project lives in the repository under `android/`. You can build it via the command line
+or open it in [Android Studio](https://developer.android.com/studio/). For simplicity, it only
+builds the SDL version with all features enabled, including tiles, sound and localization.
### Dependencies
- * Java JDK 8
- * SDL2 (tested with 2.0.8, though a custom fork is recommended with project-specific bugfixes)
- * SDL2_ttf (tested with 2.0.14)
- * SDL2_mixer (tested with 2.0.2)
- * SDL2_image (tested with 2.0.3)
+- Java JDK 8
+- SDL2 (tested with 2.0.8, though a custom fork is recommended with project-specific bugfixes)
+- SDL2_ttf (tested with 2.0.14)
+- SDL2_mixer (tested with 2.0.2)
+- SDL2_image (tested with 2.0.3)
The Gradle build process automatically installs dependencies from [deps.zip](android/app/deps.zip).
@@ -383,14 +426,18 @@ You can also use this additional variables if you want to use `ccache` to speed
### Android device setup
-Enable [Developer options on your Android device](https://developer.android.com/studio/debug/dev-options). Connect your device to your PC via USB cable and run:
+Enable
+[Developer options on your Android device](https://developer.android.com/studio/debug/dev-options).
+Connect your device to your PC via USB cable and run:
adb devices
adb connect
### Building
-To build an APK, use the Gradle wrapper command line tool (gradlew). The Android Studio documentation provides a good summary of how to [build your app from the command line](https://developer.android.com/studio/build/building-cmdline).
+To build an APK, use the Gradle wrapper command line tool (gradlew). The Android Studio
+documentation provides a good summary of how to
+[build your app from the command line](https://developer.android.com/studio/build/building-cmdline).
To build a debug APK, from the `android/` subfolder of the repository run:
@@ -402,15 +449,21 @@ To build a debug APK and immediately deploy to your connected device over adb ru
./gradlew installDebug
-To build a signed release APK (ie. one that can be installed on a device), [build an unsigned release APK and sign it manually](https://developer.android.com/studio/publish/app-signing#signing-manually).
+To build a signed release APK (ie. one that can be installed on a device),
+[build an unsigned release APK and sign it manually](https://developer.android.com/studio/publish/app-signing#signing-manually).
### Additional notes
-The app stores data files on the device in `/sdcard/Android/data/com.cleverraven/cataclysmdda/files`. The data is backwards compatible with the desktop version.
+The app stores data files on the device in
+`/sdcard/Android/data/com.cleverraven/cataclysmdda/files`. The data is backwards compatible with the
+desktop version.
# Mac OS X
-To build Cataclysm on Mac you'll need [Command Line Tools for Xcode](https://developer.apple.com/downloads/) and the [Homebrew](http://brew.sh) package manager. With Homebrew, you can easily install or build Cataclysm using the [cataclysm](https://formulae.brew.sh/formula/cataclysm) forumla.
+To build Cataclysm on Mac you'll need
+[Command Line Tools for Xcode](https://developer.apple.com/downloads/) and the
+[Homebrew](http://brew.sh) package manager. With Homebrew, you can easily install or build Cataclysm
+using the [cataclysm](https://formulae.brew.sh/formula/cataclysm) forumla.
## Simple build using Homebrew
@@ -422,11 +475,14 @@ For a stable tiles build:
brew install cataclysm
-For an experimental tiles build built from the current HEAD of [upload](https://github.com/cataclysmbnteam/Cataclysm-BN/tree/upload/):
+For an experimental tiles build built from the current HEAD of
+[upload](https://github.com/cataclysmbnteam/Cataclysm-BN/tree/upload/):
brew install cataclysm --HEAD
-Whichever build you choose, Homebrew will install the appropriate dependencies as needed. The installation will be in `/usr/local/Cellar/cataclysm` with a symlink named `cataclysm` in `/usr/local/bin`.
+Whichever build you choose, Homebrew will install the appropriate dependencies as needed. The
+installation will be in `/usr/local/Cellar/cataclysm` with a symlink named `cataclysm` in
+`/usr/local/bin`.
To launch Cataclysm, just open Terminal and run `cataclysm`.
@@ -434,31 +490,35 @@ To update a stable tiles build simply run:
brew upgrade cataclysm
-To update an experimental build, you must uninstall Cataclysm then reinstall with `--HEAD`, triggering a new build from source.
+To update an experimental build, you must uninstall Cataclysm then reinstall with `--HEAD`,
+triggering a new build from source.
brew uninstall cataclysm
brew install cataclysm --HEAD
## Advanced info for Developers
-For most people, the simple Homebrew installation is enough. For developers, here are some more technical details on building Cataclysm on Mac OS X.
+For most people, the simple Homebrew installation is enough. For developers, here are some more
+technical details on building Cataclysm on Mac OS X.
### SDL
-SDL2, SDL2_image, and SDL2_ttf are needed for the tiles build. Optionally, you can add SDL2_mixer for sound support. Cataclysm can be built using either the SDL framework, or shared libraries built from source.
+SDL2, SDL2_image, and SDL2_ttf are needed for the tiles build. Optionally, you can add SDL2_mixer
+for sound support. Cataclysm can be built using either the SDL framework, or shared libraries built
+from source.
The SDL framework files can be downloaded here:
-* [**SDL2**](http://www.libsdl.org/download-2.0.php)
-* [**SDL2_image**](http://www.libsdl.org/projects/SDL_image/)
-* [**SDL2_ttf**](http://www.libsdl.org/projects/SDL_ttf/)
+- [**SDL2**](http://www.libsdl.org/download-2.0.php)
+- [**SDL2_image**](http://www.libsdl.org/projects/SDL_image/)
+- [**SDL2_ttf**](http://www.libsdl.org/projects/SDL_ttf/)
-Copy `SDL2.framework`, `SDL2_image.framework`, and `SDL2_ttf.framework`
-to `/Library/Frameworks` or `/Users/name/Library/Frameworks`.
+Copy `SDL2.framework`, `SDL2_image.framework`, and `SDL2_ttf.framework` to `/Library/Frameworks` or
+`/Users/name/Library/Frameworks`.
If you want sound support, you will need an additional SDL framework:
-* [**SDL2_mixer**](https://www.libsdl.org/projects/SDL_mixer/)
+- [**SDL2_mixer**](https://www.libsdl.org/projects/SDL_mixer/)
Copy `SDL2_mixer.framework` to `/Library/Frameworks` or `/Users/name/Library/Frameworks`.
@@ -482,7 +542,8 @@ with sound:
### ncurses
-ncurses with wide character support enabled is needed since Cataclysm makes extensive use of Unicode characters
+ncurses with wide character support enabled is needed since Cataclysm makes extensive use of Unicode
+characters
For Homebrew:
@@ -495,11 +556,17 @@ For MacPorts:
### gcc
-The version of gcc/g++ installed with the [Command Line Tools for Xcode](https://developer.apple.com/downloads/) is actually just a front end for the same Apple LLVM as clang. This doesn't necessarily cause issues, but this version of gcc/g++ will have clang error messages and essentially produce the same results as if using clang. To compile with the "real" gcc/g++, install it with homebrew:
+The version of gcc/g++ installed with the
+[Command Line Tools for Xcode](https://developer.apple.com/downloads/) is actually just a front end
+for the same Apple LLVM as clang. This doesn't necessarily cause issues, but this version of gcc/g++
+will have clang error messages and essentially produce the same results as if using clang. To
+compile with the "real" gcc/g++, install it with homebrew:
brew install gcc
-However, homebrew installs gcc as gcc-{version} (where {version} is the version) to avoid conflicts. The simplest way to use the homebrew version at `/usr/local/bin/gcc-{version}` instead of the Apple LLVM version at `/usr/bin/gcc` is to symlink the necessary.
+However, homebrew installs gcc as gcc-{version} (where {version} is the version) to avoid conflicts.
+The simplest way to use the homebrew version at `/usr/local/bin/gcc-{version}` instead of the Apple
+LLVM version at `/usr/bin/gcc` is to symlink the necessary.
cd /usr/local/bin
ln -s gcc-12 gcc
@@ -510,7 +577,8 @@ Or, to do this for everything in `/usr/local/bin/` ending with `-12`,
find /usr/local/bin -name "*-12" -exec sh -c 'ln -s "$1" $(echo "$1" | sed "s/..$//")' _ {} \;
-Also, you need to make sure that `/usr/local/bin` appears before `/usr/bin` in your `$PATH`, or else this will not work.
+Also, you need to make sure that `/usr/local/bin` appears before `/usr/bin` in your `$PATH`, or else
+this will not work.
Check that `gcc -v` shows the homebrew version you installed.
@@ -520,19 +588,28 @@ The Cataclysm source is compiled using `make`.
### Make options
-* `NATIVE=osx` build for OS X. Required for all Mac builds.
-* `OSX_MIN=version` sets `-mmacosx-version-min=`; default is 11.
-* `TILES=1` build the SDL version with graphical tiles (and graphical ASCII); omit to build with `ncurses`.
-* `SOUND=1` - if you want sound; this requires `TILES=1` and the additional dependencies mentioned above.
-* `FRAMEWORK=1` (tiles only) link to SDL libraries under the OS X Frameworks folders; omit to use SDL shared libraries from Homebrew or Macports.
-* `LANGUAGES="[lang_id_2][...]"` compile localization files for specified languages. e.g. `LANGUAGES="zh_CN zh_TW"`. You can also use `LANGUAGES=all` to compile all localization files.
-* `RELEASE=1` build an optimized release version; omit for debug build.
-* `CLANG=1` build with [Clang](http://clang.llvm.org/), the compiler that's included with the latest Command Line Tools for Xcode; omit to build using gcc/g++.
-* `MACPORTS=1` build against dependencies installed via Macports, currently only `ncurses`.
-* `USE_HOME_DIR=1` places user files (config, saves, graveyard, etc) in the user's home directory. For curses builds, this is `/Users//.cataclysm-bn`, for SDL builds it is `/Users//Library/Application Support/Cataclysm`.
-* `DEBUG_SYMBOLS=1` retains debug symbols when building an optimized release binary, making it easy for developers to spot the crash site.
-
-In addition to the options above, there is an `app` make target which will package the tiles build into `Cataclysm.app`, a complete tiles build in a Mac application that can run without Terminal.
+- `NATIVE=osx` build for OS X. Required for all Mac builds.
+- `OSX_MIN=version` sets `-mmacosx-version-min=`; default is 11.
+- `TILES=1` build the SDL version with graphical tiles (and graphical ASCII); omit to build with
+ `ncurses`.
+- `SOUND=1` - if you want sound; this requires `TILES=1` and the additional dependencies mentioned
+ above.
+- `FRAMEWORK=1` (tiles only) link to SDL libraries under the OS X Frameworks folders; omit to use
+ SDL shared libraries from Homebrew or Macports.
+- `LANGUAGES="[lang_id_2][...]"` compile localization files for specified languages. e.g.
+ `LANGUAGES="zh_CN zh_TW"`. You can also use `LANGUAGES=all` to compile all localization files.
+- `RELEASE=1` build an optimized release version; omit for debug build.
+- `CLANG=1` build with [Clang](http://clang.llvm.org/), the compiler that's included with the latest
+ Command Line Tools for Xcode; omit to build using gcc/g++.
+- `MACPORTS=1` build against dependencies installed via Macports, currently only `ncurses`.
+- `USE_HOME_DIR=1` places user files (config, saves, graveyard, etc) in the user's home directory.
+ For curses builds, this is `/Users//.cataclysm-bn`, for SDL builds it is
+ `/Users//Library/Application Support/Cataclysm`.
+- `DEBUG_SYMBOLS=1` retains debug symbols when building an optimized release binary, making it easy
+ for developers to spot the crash site.
+
+In addition to the options above, there is an `app` make target which will package the tiles build
+into `Cataclysm.app`, a complete tiles build in a Mac application that can run without Terminal.
For more info, see the comments in the `Makefile`.
@@ -542,7 +619,8 @@ Build a release SDL version using Clang:
make NATIVE=osx RELEASE=1 TILES=1 CLANG=1
-Build a release SDL version using Clang, link to libraries in the OS X Frameworks folders, build all language files, and package it into `Cataclysm.app`:
+Build a release SDL version using Clang, link to libraries in the OS X Frameworks folders, build all
+language files, and package it into `Cataclysm.app`:
make app NATIVE=osx RELEASE=1 TILES=1 FRAMEWORK=1 LANGUAGES=all CLANG=1
@@ -564,20 +642,26 @@ For `app` builds, launch Cataclysm.app from Finder.
### Test suite
-The build will also generate a test executable at tests/cata_test.
-Invoke it as you would any other executable and it will run the full suite of tests.
-Pass the ``--help`` flag to list options.
+The build will also generate a test executable at tests/cata_test. Invoke it as you would any other
+executable and it will run the full suite of tests. Pass the `--help` flag to list options.
### dmg distribution
-You can build a nice dmg distribution file with the `dmgdist` target. You will need a tool called [dmgbuild](https://pypi.python.org/pypi/dmgbuild). To install this tool, you will need Python first. If you are on Mac OS X >= 10.8, Python 2.7 is pre-installed with the OS. If you are on an older version of OS X, you can download Python [on their official website](https://www.python.org/downloads/) or install it with homebrew `brew install python`. Once you have Python, you should be able to install `dmgbuild` by running:
+You can build a nice dmg distribution file with the `dmgdist` target. You will need a tool called
+[dmgbuild](https://pypi.python.org/pypi/dmgbuild). To install this tool, you will need Python first.
+If you are on Mac OS X >= 10.8, Python 2.7 is pre-installed with the OS. If you are on an older
+version of OS X, you can download Python
+[on their official website](https://www.python.org/downloads/) or install it with homebrew
+`brew install python`. Once you have Python, you should be able to install `dmgbuild` by running:
# This install pip. It might not be required if it is already installed.
curl --silent --show-error --retry 5 https://bootstrap.pypa.io/get-pip.py | sudo python
# dmgbuild install
sudo pip install dmgbuild pyobjc-framework-Quartz
-Once `dmgbuild` is installed, you will be able to use the `dmgdist` target like this. The use of `USE_HOME_DIR=1` is important here because it will allow for an easy upgrade of the game while keeping the user config and his saves in his home directory.
+Once `dmgbuild` is installed, you will be able to use the `dmgdist` target like this. The use of
+`USE_HOME_DIR=1` is important here because it will allow for an easy upgrade of the game while
+keeping the user config and his saves in his home directory.
make dmgdist NATIVE=osx RELEASE=1 TILES=1 FRAMEWORK=1 CLANG=1 USE_HOME_DIR=1
@@ -587,38 +671,54 @@ You should see a `Cataclysm.dmg` file.
### ISSUE: Colors don't show up correctly
-Open Terminal's preferences, turn on "Use bright colors for bold text" in "Preferences -> Settings -> Text"
-
+Open Terminal's preferences, turn on "Use bright colors for bold text" in "Preferences -> Settings
+-> Text"
# Windows
-See [COMPILING-VS-VCPKG.md](COMPILING-VS-VCPKG.md) for instructions on how to set up and use a build environment using Visual Studio on windows.
+See [COMPILING-VS-VCPKG.md](COMPILING-VS-VCPKG.md) for instructions on how to set up and use a build
+environment using Visual Studio on windows.
-This is probably the easiest solution for someone used to working with Visual Studio and similar IDEs. -->
+This is probably the easiest solution for someone used to working with Visual Studio and similar
+IDEs. -->
## Building with MSYS2
-See [COMPILING-MSYS.md](COMPILING-MSYS.md) for instructions on how to set up and use a build environment using MSYS2 on windows.
+See [COMPILING-MSYS.md](COMPILING-MSYS.md) for instructions on how to set up and use a build
+environment using MSYS2 on windows.
-MSYS2 strikes a balance between a native Windows application and a UNIX-like environment. There's some command-line tools that our project uses (notably our JSON linter) that are harder to use without a command-line environment such as what MSYS2 or CYGWIN provide.
+MSYS2 strikes a balance between a native Windows application and a UNIX-like environment. There's
+some command-line tools that our project uses (notably our JSON linter) that are harder to use
+without a command-line environment such as what MSYS2 or CYGWIN provide.
## Building with CYGWIN
-See [COMPILING-CYGWIN.md](COMPILING-CYGWIN.md) for instructions on how to set up and use a build environment using CYGWIN on windows.
+See [COMPILING-CYGWIN.md](COMPILING-CYGWIN.md) for instructions on how to set up and use a build
+environment using CYGWIN on windows.
-CYGWIN attempts to more fully emulate a POSIX environment, to be "more unix" than MSYS2. It is a little less modern in some respects, and lacks the convenience of the MSYS2 package manager.
+CYGWIN attempts to more fully emulate a POSIX environment, to be "more unix" than MSYS2. It is a
+little less modern in some respects, and lacks the convenience of the MSYS2 package manager.
## Building with Clang and MinGW64
-Clang by default uses MSVC on Windows, but also supports the MinGW64 library. Simply replace `CLANG=1` with `"CLANG=clang++ -target x86_64-pc-windows-gnu -pthread"` in your batch script, and make sure MinGW64 is in your path. You may also need to apply [a patch](https://sourceforge.net/p/mingw-w64/mailman/message/36386405/) to `float.h` of MinGW64 for the unit test to compile.
+Clang by default uses MSVC on Windows, but also supports the MinGW64 library. Simply replace
+`CLANG=1` with `"CLANG=clang++ -target x86_64-pc-windows-gnu -pthread"` in your batch script, and
+make sure MinGW64 is in your path. You may also need to apply
+[a patch](https://sourceforge.net/p/mingw-w64/mailman/message/36386405/) to `float.h` of MinGW64 for
+the unit test to compile.
# BSDs
-There are reports of CBN building fine on recent OpenBSD and FreeBSD machines (with appropriately recent compilers), and there is some work being done on making the `Makefile` "just work", however we're far from that and BSDs support is mostly based on user contributions. Your mileage may vary. So far essentially all testing has been on amd64, but there is no (known) reason that other architectures shouldn't work, in principle.
+There are reports of CBN building fine on recent OpenBSD and FreeBSD machines (with appropriately
+recent compilers), and there is some work being done on making the `Makefile` "just work", however
+we're far from that and BSDs support is mostly based on user contributions. Your mileage may vary.
+So far essentially all testing has been on amd64, but there is no (known) reason that other
+architectures shouldn't work, in principle.
### Building on FreeBSD/amd64 10.1 with the system compiler
-FreeBSD uses clang as the default compiler as of 10.0, and combines it with libc++ to provide C++14 support out of the box. You will however need gmake (examples for binary packages):
+FreeBSD uses clang as the default compiler as of 10.0, and combines it with libc++ to provide C++14
+support out of the box. You will however need gmake (examples for binary packages):
`pkg install gmake`
@@ -626,7 +726,8 @@ Tiles builds will also require SDL2:
`pkg install sdl2 sdl2_image sdl2_mixer sdl2_ttf`
-Then you should be able to build with something like this (you can of course set CXXFLAGS and LDFLAGS in your .profile or something):
+Then you should be able to build with something like this (you can of course set CXXFLAGS and
+LDFLAGS in your .profile or something):
```
export CXXFLAGS="-I/usr/local/include" LDFLAGS="-L/usr/local/lib"
@@ -634,7 +735,8 @@ gmake # ncurses builds
gmake TILES=1 # tiles builds
```
-The author has not tested tiles builds, as the build VM lacks X; they do at least compile/link successfully.
+The author has not tested tiles builds, as the build VM lacks X; they do at least compile/link
+successfully.
### Building ncurses version on FreeBSD/amd64 9.3 with GCC 4.8.4 from ports
@@ -646,21 +748,24 @@ CXX = g++48
CXXFLAGS += -I/usr/local/lib/gcc48/include
LDFLAGS += -rpath=/usr/local/lib/gcc48
```
+
Note: or you can `setenv` the above (merging `OTHERS` into `CXXFLAGS`), but you knew that.
And then build with `gmake RELEASE=1`.
### Building on OpenBSD/amd64 5.8 with GCC 4.9.2 from ports/packages
-First, install g++, gmake, and libexecinfo from packages (g++ 4.8 or 4.9 should work; 4.9 has been tested):
+First, install g++, gmake, and libexecinfo from packages (g++ 4.8 or 4.9 should work; 4.9 has been
+tested):
`pkg_add g++ gmake libexecinfo`
-Then you should be able to build with something like:
+Then you should be able to build with something like:
`CXX=eg++ gmake`
-Only an ncurses build is possible on 5.8-release, as SDL2 is broken. On recent -current or snapshots, however, you can install the SDL2 packages:
+Only an ncurses build is possible on 5.8-release, as SDL2 is broken. On recent -current or
+snapshots, however, you can install the SDL2 packages:
`pkg_add sdl2 sdl2-image sdl2-mixer sdl2-ttf`
@@ -670,11 +775,14 @@ and build with:
### Building on NetBSD/amd64 7.0RC1 with the system compiler
-NetBSD has (or will have) gcc 4.8.4 as of version 7.0, which is new enough to build cataclysm. You will need to install gmake and ncursesw:
+NetBSD has (or will have) gcc 4.8.4 as of version 7.0, which is new enough to build cataclysm. You
+will need to install gmake and ncursesw:
`pkgin install gmake ncursesw`
-Then you should be able to build with something like this (LDFLAGS for ncurses builds are taken care of by the ncurses configuration script; you can of course set CXXFLAGS/LDFLAGS in your .profile or something):
+Then you should be able to build with something like this (LDFLAGS for ncurses builds are taken care
+of by the ncurses configuration script; you can of course set CXXFLAGS/LDFLAGS in your .profile or
+something):
```
export CXXFLAGS="-I/usr/pkg/include"
@@ -682,4 +790,5 @@ gmake # ncurses builds
LDFLAGS="-L/usr/pkg/lib" gmake TILES=1 # tiles builds
```
-SDL builds currently compile, but did not run in my testing - not only do they segfault, but gdb segfaults when reading the debug symbols! Perhaps your mileage will vary.
+SDL builds currently compile, but did not run in my testing - not only do they segfault, but gdb
+segfaults when reading the debug symbols! Perhaps your mileage will vary.
diff --git a/doc/CONTRIBUTING.ko.md b/doc/CONTRIBUTING.ko.md
index ba7f538b4bf2..f23090e04480 100644
--- a/doc/CONTRIBUTING.ko.md
+++ b/doc/CONTRIBUTING.ko.md
@@ -1,7 +1,6 @@
# 기여하기
-[![en][icon-en]][en]
-[![ko][icon-ko]][ko]
+[![en][icon-en]][en] [![ko][icon-ko]][ko]
[en]: ./CONTRIBUTING.md
[icon-en]: https://img.shields.io/badge/lang-en-red?style=flat-square
@@ -11,6 +10,9 @@
- [기여하기](#기여하기)
- [가이드라인](#가이드라인)
- [코드 스타일](#코드-스타일)
+ - [C++](#c)
+ - [JSON 스타일](#json-스타일)
+ - [마크다운](#마크다운)
- [번역](#번역)
- [공식 문서](#공식-문서)
- [독시젠(Doxygen) 주석](#독시젠doxygen-주석)
@@ -44,15 +46,18 @@
[pr]: https://docs.github.com/ko/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests
-> 카타클리즘: 밝은 밤은 크리에이티브 커먼즈 저작자표시-동일조건변경허락 3.0 라이선스에 따라 배포됩니다. 게임의 코드와 콘텐츠는 어떠한 목적에도 사용, 수정, 재배포할 수 있습니다. 자세한 내용은 http://creativecommons.org/licenses/by-sa/3.0/ 를 참고해주세요.
-> 그 말은, 이 프로젝트에 기여하면, 그 기여물도 동일한 라이선스에 의해 보호받는다는 것이며, 이 라이선스는 취소될 수 없다는 것입니다.
+> 카타클리즘: 밝은 밤은 크리에이티브 커먼즈 저작자표시-동일조건변경허락 3.0 라이선스에 따라
+> 배포됩니다. 게임의 코드와 콘텐츠는 어떠한 목적에도 사용, 수정, 재배포할 수 있습니다. 자세한 내용은
+> http://creativecommons.org/licenses/by-sa/3.0/ 를 참고해주세요. 그 말은, 이 프로젝트에 기여하면,
+> 그 기여물도 동일한 라이선스에 의해 보호받는다는 것이며, 이 라이선스는 취소될 수 없다는 것입니다.
## 가이드라인
몇 가지 지켰으면 하는 가이드라인이 있습니다:
- 이 저장소를 `upstream` [리모트][remote]로 추가해주세요.
-- `upload` 브랜치를 수정사항 없이 깨끗하게 유지해주세요. 원격 저장소의 최신 변경사항을 바로 끌어올 수 있게 하기 위함입니다.
+- `upload` 브랜치를 수정사항 없이 깨끗하게 유지해주세요. 원격 저장소의 최신 변경사항을 바로 끌어올
+ 수 있게 하기 위함입니다.
- 새 기능이나 버그 수정을 할 때마다 새 브랜치를 만들어주세요.
- 절대로 `upload` 브랜치에 로컬 브랜치를 병합하지 마세요. `upstream/upload`에서 끌어오기만 해주세요.
@@ -60,13 +65,37 @@
## 코드 스타일
-`astyle`로 일관된 코드 스타일을 강제하고 있습니다.
-자세한 내용은 [CODE_STYLE](../doc/CODE_STYLE.md)을 참고해주세요.
+### C++
+
+`astyle`로 일관된 코드 스타일을 강제하고 있습니다. 자세한 내용은
+[CODE_STYLE](../doc/CODE_STYLE.md)을 참고해주세요.
+
+### JSON 스타일
+
+`tools/format` 경로에 있는 포매터로 일관된 JSON 스타일을 강제하고 있습니다.
+[JSON Style Guide](../doc/JSON_STYLE.md) 을 참고해주세요.
+
+### 마크다운
+
+`doc/`같은 마크다운 파일들은 [`deno`](https://deno.com)를 사용해 포매팅하고 있습니다.
+[`deno fmt`](https://deno.land/manual/tools/formatter) 을 실행해 자동으로 마크다운 파일을
+포매팅하세요. VsCode를 사용중이라면 다음 설정으로 저장할 때마다 자동 포매팅을 실행할 수 있습니다:
+
+```json
+// .vscode/settings.json
+{
+ "[markdown]": {
+ "editor.formatOnSave": true,
+ "editor.defaultFormatter": "denoland.vscode-deno"
+ }
+}
+```
## 번역
카타클리즘: 밝은 밤의 번역은 Transifex에서 진행중입니다.
-[번역 프로젝트](https://app.transifex.com/bn-team/cataclysm-bright-nights/)에서 지원되는 언어를 실시간으로 확인할 수 있습니다.
+[번역 프로젝트](https://app.transifex.com/bn-team/cataclysm-bright-nights/)에서 지원되는 언어를
+실시간으로 확인할 수 있습니다.
[TRANSLATING](../doc/TRANSLATING.md)에서 더 자세한 내용을 확인할 수 있습니다:
@@ -81,11 +110,13 @@
-자동 생성된 문서를 [깃허브 페이지](https://cataclysmbnteam.github.io/Cataclysm-BN)에서 읽을 수 있습니다.
+자동 생성된 문서를 [깃허브 페이지](https://cataclysmbnteam.github.io/Cataclysm-BN)에서 읽을 수
+있습니다.
### 독시젠(Doxygen) 주석
-클래스와 클래스 멤버에 대한 상세한 문서가 있으면 새로운 기여자들이 코드를 읽고 이해하는데 도움이 됩니다. 기존 클래스에 독시젠 주석을 다는 것도 환영입니다.
+클래스와 클래스 멤버에 대한 상세한 문서가 있으면 새로운 기여자들이 코드를 읽고 이해하는데 도움이
+됩니다. 기존 클래스에 독시젠 주석을 다는 것도 환영입니다.
클래스에 주석을 달 때는 다음 템플릿을 사용해주세요:
@@ -120,15 +151,18 @@ int foo;
### 문서 추가 가이드라인
-- 독시젠 주석은 '밖에서 보았을 때' 동작을 설명해야 합니다. 숨겨진 내부 구현에 대한 설명은 하지 말아야 합니다. 하지만 카타클리즘의 많은 클래스들이 서로 연결되어 있어서, 구현에 대한 설명이 필요할 때도 있을 것입니다.
+- 독시젠 주석은 '밖에서 보았을 때' 동작을 설명해야 합니다. 숨겨진 내부 구현에 대한 설명은 하지
+ 말아야 합니다. 하지만 카타클리즘의 많은 클래스들이 서로 연결되어 있어서, 구현에 대한 설명이 필요할
+ 때도 있을 것입니다.
- 새 기여자들이 이름만으로는 이해하기 어려운 것들에 대해서만 설명해주세요.
- 동어 반복은 피해주세요: `/** Map **/; map* map;`는 도움이 되지 않습니다.
-- `X`에 대한 설명을 할 때, `X`와 다른 컴포넌트들 간의 상호작용에 대해서만 설명해주세요. `X` 자체가 하는 일에 대해서는 설명하지 말아주세요.
+- `X`에 대한 설명을 할 때, `X`와 다른 컴포넌트들 간의 상호작용에 대해서만 설명해주세요. `X` 자체가
+ 하는 일에 대해서는 설명하지 말아주세요.
### 문서를 로컬에서 빌드하기
- [독시젠 설치](https://www.doxygen.nl)
-- `doxygen doxygen_doc/doxygen_conf.txt `
+- `doxygen doxygen_doc/doxygen_conf.txt`
- `firefox doxygen_doc/html/index.html` (또는 다른 브라우저)
## 예시 워크플로우
@@ -149,7 +183,7 @@ $ git clone https://github.com/깃허브_사용자명/Cataclysm-BN.git
3. 커밋 메시지 템플릿을 설정합니다.
```sh
- $ git config --local commit.template .gitmessage
+$ git config --local commit.template .gitmessage
```
4. 원본 저장소를 원격 저장소로 추가합니다.
@@ -182,7 +216,8 @@ $ git pull --ff-only upstream upload
# "upstream" 원격 저장소의 "upload" 브랜치에서 변경사항을 가져옵니다.
```
-> **Note** 오류가 발생했다면, 로컬 `upload` 브랜치에 직접 커밋을 했다는 뜻입니다. [이 문제를 해결하는 방법을 보려면 여기를 클릭하세요](#git-pull---ff-only을-했더니-에러가-나요).
+> **Note** 오류가 발생했다면, 로컬 `upload` 브랜치에 직접 커밋을 했다는 뜻입니다.
+> [이 문제를 해결하는 방법을 보려면 여기를 클릭하세요](#git-pull---ff-only을-했더니-에러가-나요).
### 변경사항 만들기
@@ -204,28 +239,38 @@ $ git push origin new_feature
# origin은 복제할 때 자동으로 포크를 가리키도록 설정되어 있습니다.
```
-3. 브랜치에서 작업을 마치고, 모든 변경사항을 커밋하고 푸시했다면, `new_feature` 브랜치에서 이 저장소의 `upload` 브랜치로 풀 리퀘스트를 보내주세요.
+3. 브랜치에서 작업을 마치고, 모든 변경사항을 커밋하고 푸시했다면, `new_feature` 브랜치에서 이
+ 저장소의 `upload` 브랜치로 풀 리퀘스트를 보내주세요.
-> **Note** 깃허브의 `new_feature` 브랜치에 새 커밋이 생기면, 풀 리퀘스트에 자동으로 포함됩니다. 따라서 같은 브랜치에 관련된 변경사항만 커밋해주세요.
+> **Note** 깃허브의 `new_feature` 브랜치에 새 커밋이 생기면, 풀 리퀘스트에 자동으로 포함됩니다.
+> 따라서 같은 브랜치에 관련된 변경사항만 커밋해주세요.
## 풀 리퀘스트 초안
-풀 리퀘스트를 만들었지만 아직 작업 중이라면, [초안(draft)](https://docs.github.com/ko/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests#draft-pull-requests)으로 표시해주세요. 이렇게 하면 준비된 상태가 아니라는 것을 리뷰어에게 알려줘 리뷰 진행 속도를 높일 수 있습니다.
+풀 리퀘스트를 만들었지만 아직 작업 중이라면,
+[초안(draft)](https://docs.github.com/ko/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests#draft-pull-requests)으로
+표시해주세요. 이렇게 하면 준비된 상태가 아니라는 것을 리뷰어에게 알려줘 리뷰 진행 속도를 높일 수
+있습니다.
-풀 리퀘스트를 만들 때 이슈를 참조할 필요는 없지만, 참조하지 않느나면 PR이 어떤 문제룰 해결하려는지 자세히 설명해야 합니다.
+풀 리퀘스트를 만들 때 이슈를 참조할 필요는 없지만, 참조하지 않느나면 PR이 어떤 문제룰 해결하려는지
+자세히 설명해야 합니다.
### 모든 풀 리퀘스트에는 `"Summary"`줄이 있어야 합니다.
-개요(summary)는 [변경 내역](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/data/changelog.txt)에 추가할 한 줄 요약입니다.
+개요(summary)는
+[변경 내역](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/data/changelog.txt)에 추가할
+한 줄 요약입니다.
개요 형식: `SUMMARY: 카테고리 "설명"`
-고를 수 있는 카테고리는 Features, Content, Interface, Mods, Balance, Bugfixes, Performance, Infrastructure, Build, I18N이 있습니다.
+고를 수 있는 카테고리는 Features, Content, Interface, Mods, Balance, Bugfixes, Performance,
+Infrastructure, Build, I18N이 있습니다.
-예시: `SUMMARY: Content "Adds new mutation category 'Mouse'"`
-(해석: `SUMMARY: Content "새로운 변이 카테고리 'Mouse'를 추가합니다."`)
+예시: `SUMMARY: Content "Adds new mutation category 'Mouse'"` (해석:
+`SUMMARY: Content "새로운 변이 카테고리 'Mouse'를 추가합니다."`)
-[변경 내역 가이드라인](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/CHANGELOG_GUIDELINES.md)에서 카테고리에 대한 설명을 볼 수 있습니다.
+[변경 내역 가이드라인](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/CHANGELOG_GUIDELINES.md)에서
+카테고리에 대한 설명을 볼 수 있습니다.
### 키워드로 이슈 닫기
@@ -257,11 +302,14 @@ $ git push origin new_feature
- {키워드} #{이슈 번호}, {키워드} #{이슈 번호}
```
-더 자세한 설명은 [깃허브 공식 문서](https://docs.github.com/ko/issues/tracking-your-work-with-issues/linking-a-pull-request-to-an-issue)를 참고해주세요.
+더 자세한 설명은
+[깃허브 공식 문서](https://docs.github.com/ko/issues/tracking-your-work-with-issues/linking-a-pull-request-to-an-issue)를
+참고해주세요.
## 개발 도구 지원
-코딩 스타일을 지키도록 도와주는 여러 도구들이 있습니다. 자세한 내용은 [DEVELOPER_TOOLING](../doc/DEVELOPER_TOOLING.md)을 참고해주세요.
+코딩 스타일을 지키도록 도와주는 여러 도구들이 있습니다. 자세한 내용은
+[DEVELOPER_TOOLING](../doc/DEVELOPER_TOOLING.md)을 참고해주세요.
## 고급
@@ -269,8 +317,9 @@ $ git push origin new_feature
### 원격 추적 브랜치 사용하기
-Remote tracking branches allow you to easily stay in touch with this repository's `upload` branch, as they automatically know which remote branch to get changes from.
-원본 저장소의 `upload` 브랜치에 대한 원격 추적 브랜치를 설정하면 쉽게 최신 변경사항을 가져올 수 있습니다.
+Remote tracking branches allow you to easily stay in touch with this repository's `upload` branch,
+as they automatically know which remote branch to get changes from. 원본 저장소의 `upload` 브랜치에
+대한 원격 추적 브랜치를 설정하면 쉽게 최신 변경사항을 가져올 수 있습니다.
```sh
$ git branch -vv
@@ -278,7 +327,9 @@ $ git branch -vv
new_feature xxxx ....
```
-`upload` 브랜치는 `origin/upload` 브랜치를 추적하고 있고, `new_feature` 브랜치는 아무 브랜치도 추적하고 있지 않습니다. 그 말은 git이 어디에서 `new_feature` 에 대한 변경사항을 가져올지 모른다는 뜻입니다.
+`upload` 브랜치는 `origin/upload` 브랜치를 추적하고 있고, `new_feature` 브랜치는 아무 브랜치도
+추적하고 있지 않습니다. 그 말은 git이 어디에서 `new_feature` 에 대한 변경사항을 가져올지 모른다는
+뜻입니다.
```sh
$ git checkout new_feature
@@ -288,7 +339,8 @@ $ git pull
어떤 브랜치를 대상으로 병합할지 지정하십시오.
```
-`new_feature` 브랜치에서 `upstream/upload` 브랜치의 변경사항을 쉽게 가져오려면, git에 어떤 브랜치를 추적할지 알려줘야 합니다. (로컬 `upload` 브랜치에도 적용할 수 있습니다.)
+`new_feature` 브랜치에서 `upstream/upload` 브랜치의 변경사항을 쉽게 가져오려면, git에 어떤 브랜치를
+추적할지 알려줘야 합니다. (로컬 `upload` 브랜치에도 적용할 수 있습니다.)
```sh
$ git branch -u upstream/upload new_feature
@@ -305,7 +357,9 @@ $ git branch new_feature_2 --track upstream/upload
Branch new_feature_2 set up to track remote branch upload from upstream.
```
-> **Note** : 이렇게 하면 `upstream/upload` 브랜치에서 변경사항을 가져오는 것은 쉬워지지만, `git push`는 여전히 실패합니다. `git push`는 `upstream/upload` 브랜치에 변경사항을 푸시할 권한이 없기 때문입니다.
+> **Note** : 이렇게 하면 `upstream/upload` 브랜치에서 변경사항을 가져오는 것은 쉬워지지만,
+> `git push`는 여전히 실패합니다. `git push`는 `upstream/upload` 브랜치에 변경사항을 푸시할 권한이
+> 없기 때문입니다.
```sh
$ git push
@@ -319,8 +373,10 @@ xxxx..xxxx new_feature -> new_feature
## 단위 테스트
-`tests/` 경로에 단위 테스트가 있습니다. 게임 소스를 변경한 후에는 반드시 테스트를 실행해야 합니다. `make` 명령을 실행하면 `tests/cata_test` 실행 파일이 생성됩니다. 이 파일은 일반적인 실행 파일처럼 실행할 수 있습니다. `make check` 명령으로도 실행할 수 있습니다.
-아무 인자 없이 실행하면 전체 테스트를 실행합니다. `--help` 인자를 사용하면 실행 옵션을 볼 수 있습니다.
+`tests/` 경로에 단위 테스트가 있습니다. 게임 소스를 변경한 후에는 반드시 테스트를 실행해야 합니다.
+`make` 명령을 실행하면 `tests/cata_test` 실행 파일이 생성됩니다. 이 파일은 일반적인 실행 파일처럼
+실행할 수 있습니다. `make check` 명령으로도 실행할 수 있습니다. 아무 인자 없이 실행하면 전체
+테스트를 실행합니다. `--help` 인자를 사용하면 실행 옵션을 볼 수 있습니다.
```sh
$ make
@@ -337,9 +393,13 @@ The test took 41.772 seconds
## 게임 내에서 테스트하기, 테스트 환경, 디버그 메뉴
-새 기능을 구현하거나 버그를 수정하는 경우, 게임 내에서 변경사항을 테스트하는 것이 좋습니다. 평소처럼 게임을 플레이해서 정확한 조건을 만들어내기는 힘들지만, 디버그 메뉴를 사용하면 쉽게 테스트할 수 있습니다. 기본적으로 메뉴를 띄우는 단축키가 없으므로 먼저 단축키를 지정해야 합니다.
+새 기능을 구현하거나 버그를 수정하는 경우, 게임 내에서 변경사항을 테스트하는 것이 좋습니다. 평소처럼
+게임을 플레이해서 정확한 조건을 만들어내기는 힘들지만, 디버그 메뉴를 사용하면 쉽게 테스트할 수
+있습니다. 기본적으로 메뉴를 띄우는 단축키가 없으므로 먼저 단축키를 지정해야 합니다.
-단축키 설정 메뉴를 띄웁니다. (`Esc`키를 누른 다음 `1`키를 누릅니다.) 아래로 스크롤해서 _디버그 메뉴_ 항목을 찾고, `+` 키를 눌러 새로운 단축키를 추가합니다. 테스트를 한 다면 새 캐릭터로 하는 것이 좋습니다. 방금 설정한 단축키를 누르면 다음과 같은 화면이 나타날 것입니다.
+단축키 설정 메뉴를 띄웁니다. (`Esc`키를 누른 다음 `1`키를 누릅니다.) 아래로 스크롤해서 _디버그 메뉴_
+항목을 찾고, `+` 키를 눌러 새로운 단축키를 추가합니다. 테스트를 한 다면 새 캐릭터로 하는 것이
+좋습니다. 방금 설정한 단축키를 누르면 다음과 같은 화면이 나타날 것입니다.
```
┌─────────────────────────────────────────────────────┐
@@ -354,13 +414,17 @@ The test took 41.772 seconds
└─────────────────────────────────────────────────────┘
```
-위 명령어들을 사용해서 변경사항을 테스트할 수 있습니다. [DDA 위키](http://cddawiki.chezzo.com/cdda_wiki/index.php)에도 디버그 메뉴에 대한 정보가 있습니다.
+위 명령어들을 사용해서 변경사항을 테스트할 수 있습니다.
+[DDA 위키](http://cddawiki.chezzo.com/cdda_wiki/index.php)에도 디버그 메뉴에 대한 정보가 있습니다.
## 자주 묻는 질문
### `git pull --ff-only`을 했더니 에러가 나요
-`git pull --ff-only`를 실행했더니 에러가 났다면, `upload` 브랜치에 직접 커밋을 했기 때문입니다. 그 이유는 `upload` 브랜치의 내용이 원격과 로컬에서 각각 달라졌기 때문에, git이 원격과 로컬 중 무엇을 유지하고 무엇을 버릴 지 모르기 때문입니다. 이를 고치려면, 새 브랜치를 만들고, `upstream/upload` 브랜치와 분기된 지점을 찾은 다음, `upload` 브랜치를 그 지점으로 되돌려야 합니다.
+`git pull --ff-only`를 실행했더니 에러가 났다면, `upload` 브랜치에 직접 커밋을 했기 때문입니다. 그
+이유는 `upload` 브랜치의 내용이 원격과 로컬에서 각각 달라졌기 때문에, git이 원격과 로컬 중 무엇을
+유지하고 무엇을 버릴 지 모르기 때문입니다. 이를 고치려면, 새 브랜치를 만들고, `upstream/upload`
+브랜치와 분기된 지점을 찾은 다음, `upload` 브랜치를 그 지점으로 되돌려야 합니다.
```sh
$ git pull --ff-only upstream upload
@@ -374,7 +438,8 @@ $ git reset --hard cc31d0....
HEAD is now at cc31d0... ...
```
-이제 `upload`가 정리되었으니 `upstream/upload`에서 변경 내역을 끌어오고, `new_branch`에서 계속 작업할 수 있습니다.
+이제 `upload`가 정리되었으니 `upstream/upload`에서 변경 내역을 끌어오고, `new_branch`에서 계속
+작업할 수 있습니다.
```sh
$ git pull --ff-only upstream upload
diff --git a/doc/CONTRIBUTING.md b/doc/CONTRIBUTING.md
index 86948f56091f..5cb202d96e08 100644
--- a/doc/CONTRIBUTING.md
+++ b/doc/CONTRIBUTING.md
@@ -1,7 +1,6 @@
# Contribute
-[![en][icon-en]][en]
-[![ko][icon-ko]][ko]
+[![en][icon-en]][en] [![ko][icon-ko]][ko]
[en]: ./CONTRIBUTING.md
[icon-en]: https://img.shields.io/badge/lang-en-red?style=flat-square
@@ -11,6 +10,9 @@
- [Contribute](#contribute)
- [Guidelines](#guidelines)
- [Code Style](#code-style)
+ - [C++](#c)
+ - [JSON](#json)
+ - [Markdown](#markdown)
- [Translations](#translations)
- [Documentation](#documentation)
- [Doxygen Comments](#doxygen-comments)
@@ -42,27 +44,56 @@ Contributing to Cataclysm: Bright Nights is easy:
1. Make your changes.
1. Send us a pull request.
-> Cataclysm: Bright Nights is released under the Creative Commons Attribution ShareAlike 3.0 license. The code and content of the game is free to use, modify, and redistribute for any purpose whatsoever. See http://creativecommons.org/licenses/by-sa/3.0/ for details.
-> This means any contribution you make to the project will also be covered by the same license, and this license is irrevocable.
+> Cataclysm: Bright Nights is released under the Creative Commons Attribution ShareAlike 3.0
+> license. The code and content of the game is free to use, modify, and redistribute for any purpose
+> whatsoever. See http://creativecommons.org/licenses/by-sa/3.0/ for details. This means any
+> contribution you make to the project will also be covered by the same license, and this license is
+> irrevocable.
## Guidelines
There are a couple of guidelines we suggest sticking to:
- Add this repository as an `upstream` remote.
-- Keep your `upload` branch clean. This means you can easily pull changes made to this repository into yours.
+- Keep your `upload` branch clean. This means you can easily pull changes made to this repository
+ into yours.
- Create a new branch for each new feature or set of related bug fixes.
-- Never merge from your local branches into your `upload` branch. Only update that by pulling from `upstream/upload`.
+- Never merge from your local branches into your `upload` branch. Only update that by pulling from
+ `upstream/upload`.
## Code Style
-Code style is enforced across the codebase by `astyle`.
-See [CODE_STYLE](../doc/CODE_STYLE.md) for details.
+### C++
+
+Code style is enforced across the codebase by `astyle`. See [CODE_STYLE](../doc/CODE_STYLE.md) for
+details.
+
+### JSON
+
+JSON files are formatted using custom formatter available in `tools/format`. Visit
+[JSON Style Guide](../doc/JSON_STYLE.md) for details.
+
+### Markdown
+
+Markdown files such as `doc/` are formatted using [`deno`](https://deno.com)'s built-in formatter.
+Run [`deno fmt`](https://deno.land/manual/tools/formatter) anywhere to format markdown files. On
+VSCode, you can set following configuration to auto-format markdown files on save:
+
+```json
+// .vscode/settings.json
+{
+ "[markdown]": {
+ "editor.formatOnSave": true,
+ "editor.defaultFormatter": "denoland.vscode-deno"
+ }
+}
+```
## Translations
-The translation of Cataclysm: BN is done using Transifex.
-Look at the [translation project](https://app.transifex.com/bn-team/cataclysm-bright-nights/) for an up-to-date list of supported languages.
+The translation of Cataclysm: BN is done using Transifex. Look at the
+[translation project](https://app.transifex.com/bn-team/cataclysm-bright-nights/) for an up-to-date
+list of supported languages.
See [TRANSLATING](../doc/TRANSLATING.md) for more information:
@@ -77,11 +108,13 @@ See [TRANSLATING](../doc/TRANSLATING.md) for more information:
-Autogenerated documentation is hosted on [GitHub Pages](https://cataclysmbnteam.github.io/Cataclysm-BN).
+Autogenerated documentation is hosted on
+[GitHub Pages](https://cataclysmbnteam.github.io/Cataclysm-BN).
### Doxygen Comments
-Extensive documentation of classes and class members will make the code more readable to new contributors. New doxygen comments for existing classes are a welcomed contribution.
+Extensive documentation of classes and class members will make the code more readable to new
+contributors. New doxygen comments for existing classes are a welcomed contribution.
Use the following template for commenting classes:
@@ -116,7 +149,8 @@ int foo;
### Guidelines for adding documentation
-- Doxygen comments should describe behavior towards the outside, not implementation, but since many classes in Cataclysm are intertwined, it's often necessary to describe implementation.
+- Doxygen comments should describe behavior towards the outside, not implementation, but since many
+ classes in Cataclysm are intertwined, it's often necessary to describe implementation.
- Describe things that aren't obvious to newcomers just from the name.
- Don't describe redundantly: `/** Map **/; map* map;` is not a helpful comment.
- When documenting X, describe how X interacts with other components, not just what X itself does.
@@ -124,7 +158,7 @@ int foo;
### Building the documentation for viewing it locally
- Install doxygen
-- `doxygen doxygen_doc/doxygen_conf.txt `
+- `doxygen doxygen_doc/doxygen_conf.txt`
- `firefox doxygen_doc/html/index.html` (replace firefox with your browser of choice)
## Example Workflow
@@ -145,7 +179,7 @@ $ git clone https://github.com/YOUR_USERNAME/Cataclysm-BN.git
3. Set commit message template.
```sh
- $ git config --local commit.template .gitmessage
+$ git config --local commit.template .gitmessage
```
4. Add this repository as a remote.
@@ -178,7 +212,9 @@ $ git pull --ff-only upstream upload
# gets changes from "upload" branch on the "upstream" remote
```
-> **Note** If this gives you an error, it means you have committed directly to your local `upload` branch. [Click here for instructions on how to fix this issue](#why-does-git-pull---ff-only-result-in-an-error).
+> **Note** If this gives you an error, it means you have committed directly to your local `upload`
+> branch.
+> [Click here for instructions on how to fix this issue](#why-does-git-pull---ff-only-result-in-an-error).
### Make your changes
@@ -200,31 +236,42 @@ $ git push origin new_feature
# origin was automatically set to point to your fork when you cloned it
```
-3. Once you're finished working on your branch, and have committed and pushed all your changes, submit a pull request from your `new_feature` branch to this repository's `upload` branch.
+3. Once you're finished working on your branch, and have committed and pushed all your changes,
+ submit a pull request from your `new_feature` branch to this repository's `upload` branch.
-> **Note** any new commits to the `new_feature` branch on GitHub will automatically be included in the pull request, so make sure to only commit related changes to the same branch.
+> **Note** any new commits to the `new_feature` branch on GitHub will automatically be included in
+> the pull request, so make sure to only commit related changes to the same branch.
## Pull Request Notes
-If you file a PR but you're still working on it, please mark it as [draft](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests#draft-pull-requests). This can help speed up our review process by allowing us to only review the things that are ready for it, and will prevent anything that isn't completely ready from being merged in.
+If you file a PR but you're still working on it, please mark it as
+[draft](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests#draft-pull-requests).
+This can help speed up our review process by allowing us to only review the things that are ready
+for it, and will prevent anything that isn't completely ready from being merged in.
-It is not required to solve or reference an open issue to file a PR, however, if you do so, you need to explain the problem your PR is solving in full detail.
+It is not required to solve or reference an open issue to file a PR, however, if you do so, you need
+to explain the problem your PR is solving in full detail.
### All PRs should have a `"Summary"` line
-Summary is a one-line description of your change that will be extracted and added to the project changelog at
+Summary is a one-line description of your change that will be extracted and added to the project
+changelog at
The format is: `SUMMARY: Category "description"`
-The categories to choose from are: Features, Content, Interface, Mods, Balance, Bugfixes, Performance, Infrastructure, Build, I18N.
+The categories to choose from are: Features, Content, Interface, Mods, Balance, Bugfixes,
+Performance, Infrastructure, Build, I18N.
Example: `SUMMARY: Content "Adds new mutation category 'Mouse'"`
-See the [Changelog Guidelines](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/CHANGELOG_GUIDELINES.md) for explanations of the categories.
+See the
+[Changelog Guidelines](https://github.com/cataclysmbnteam/Cataclysm-BN/blob/upload/doc/CHANGELOG_GUIDELINES.md)
+for explanations of the categories.
### Closing issues using keywords
-One more thing: when marking your PR as closing, fixing, or resolving issues, please include this somewhere in the description:
+One more thing: when marking your PR as closing, fixing, or resolving issues, please include this
+somewhere in the description:
```
- {keyword} #{issue}
@@ -244,7 +291,8 @@ for example: `- fixed #12345`
and `{issue}` is the number of the issue you're closing after PR gets merged.
-This would automatically close the issue when the PR is pulled in, and allows merges to work slightly faster.
+This would automatically close the issue when the PR is pulled in, and allows merges to work
+slightly faster.
### closing multiple issues at once
@@ -256,7 +304,8 @@ See for more.
## Tooling support
-Various tools are available to help you keep your contributions conforming to the appropriate style. See [DEVELOPER_TOOLING](../doc/DEVELOPER_TOOLING.md) for more details.
+Various tools are available to help you keep your contributions conforming to the appropriate style.
+See [DEVELOPER_TOOLING](../doc/DEVELOPER_TOOLING.md) for more details.
## Advanced Techniques
@@ -264,7 +313,8 @@ These guidelines aren't essential, but they can make keeping things in order muc
### Using remote tracking branches
-Remote tracking branches allow you to easily stay in touch with this repository's `upload` branch, as they automatically know which remote branch to get changes from.
+Remote tracking branches allow you to easily stay in touch with this repository's `upload` branch,
+as they automatically know which remote branch to get changes from.
```sh
$ git branch -vv
@@ -272,7 +322,9 @@ $ git branch -vv
new_feature xxxx ....
```
-Here you can see we have two branches; `upload` which is tracking `origin/upload`, and `new_feature` which isn't tracking any branch. In practice, what this means is that git won't know where to get changes from.
+Here you can see we have two branches; `upload` which is tracking `origin/upload`, and `new_feature`
+which isn't tracking any branch. In practice, what this means is that git won't know where to get
+changes from.
```sh
$ git checkout new_feature
@@ -282,7 +334,8 @@ There is no tracking information for the current branch.
Please specify which branch you want to merge with.
```
-In order to easily pull changes from `upstream/upload` into the `new_feature` branch, we can tell git which branch it should track. (You can even do this for your local upload branch.)
+In order to easily pull changes from `upstream/upload` into the `new_feature` branch, we can tell
+git which branch it should track. (You can even do this for your local upload branch.)
```sh
$ git branch -u upstream/upload new_feature
@@ -299,7 +352,9 @@ $ git branch new_feature_2 --track upstream/upload
Branch new_feature_2 set up to track remote branch upload from upstream.
```
-> **Note**: Although this makes it easier to pull from `upstream/upload`, it doesn't change anything with regards to pushing. `git push` fails because you don't have permission to push to `upstream/upload`.
+> **Note**: Although this makes it easier to pull from `upstream/upload`, it doesn't change anything
+> with regards to pushing. `git push` fails because you don't have permission to push to
+> `upstream/upload`.
```sh
$ git push
@@ -313,11 +368,11 @@ xxxx..xxxx new_feature -> new_feature
## Unit tests
-There is a suite of tests built into the source tree at tests/
-You should run the test suite after ANY change to the game source.
-An ordinary invocation of `make` will build the test executable at tests/cata_test, and it can be invoked like any ordinary executable, or via `make check`.
-With no arguments it will run the entire test suite.
-With `--help` it will print a number of invocation options you can use to adjust its operation.
+There is a suite of tests built into the source tree at tests/ You should run the test suite after
+ANY change to the game source. An ordinary invocation of `make` will build the test executable at
+tests/cata_test, and it can be invoked like any ordinary executable, or via `make check`. With no
+arguments it will run the entire test suite. With `--help` it will print a number of invocation
+options you can use to adjust its operation.
```sh
$ make
@@ -334,9 +389,16 @@ I recommend habitually invoking make like `make YOUR BUILD OPTIONS && make check
## In-game testing, test environment and the debug menu
-Whether you are implementing a new feature or whether you are fixing a bug, it is always a good practice to test your changes in-game. It can be a hard task to create the exact conditions by playing a normal game to be able to test your changes, which is why there is a debug menu. There is no default key to bring up the menu so you will need to assign one first.
+Whether you are implementing a new feature or whether you are fixing a bug, it is always a good
+practice to test your changes in-game. It can be a hard task to create the exact conditions by
+playing a normal game to be able to test your changes, which is why there is a debug menu. There is
+no default key to bring up the menu so you will need to assign one first.
-Bring up the keybindings menu (press `Escape` then `1`), scroll down almost to the bottom and press `+` to add a new key binding. Press the letter that corresponds to the _Debug menu_ item, then press the key you want to use to bring up the debug menu. To test your changes, create a new world with a new character. Once you are in that world, press the key you just assigned for the debug menu and you should see something like this:
+Bring up the keybindings menu (press `Escape` then `1`), scroll down almost to the bottom and press
+`+` to add a new key binding. Press the letter that corresponds to the _Debug menu_ item, then press
+the key you want to use to bring up the debug menu. To test your changes, create a new world with a
+new character. Once you are in that world, press the key you just assigned for the debug menu and
+you should see something like this:
```
┌─────────────────────────────────────────────────────┐
@@ -351,13 +413,17 @@ Bring up the keybindings menu (press `Escape` then `1`), scroll down almost to t
└─────────────────────────────────────────────────────┘
```
-With these commands, you should be able to recreate the proper conditions to test your changes. The [DDA wiki](http://cddawiki.chezzo.com/cdda_wiki/index.php) may have useful informations regarding debug menu.
+With these commands, you should be able to recreate the proper conditions to test your changes. The
+[DDA wiki](http://cddawiki.chezzo.com/cdda_wiki/index.php) may have useful informations regarding
+debug menu.
## Frequently Asked Questions
### Why does `git pull --ff-only` result in an error?
-If `git pull --ff-only` shows an error, it means that you've committed directly to your local `upload` branch. To fix this, we create a new branch with these commits, find the point at which we diverged from `upstream/upload`, and then reset `upload` to that point.
+If `git pull --ff-only` shows an error, it means that you've committed directly to your local
+`upload` branch. To fix this, we create a new branch with these commits, find the point at which we
+diverged from `upstream/upload`, and then reset `upload` to that point.
```sh
$ git pull --ff-only upstream upload
@@ -371,7 +437,8 @@ $ git reset --hard cc31d0....
HEAD is now at cc31d0... ...
```
-Now that `upload` has been cleaned up, we can easily pull from `upstream/upload`, and then continue working on `new_branch`.
+Now that `upload` has been cleaned up, we can easily pull from `upstream/upload`, and then continue
+working on `new_branch`.
```sh
$ git pull --ff-only upstream upload
diff --git a/doc/DEVELOPER_FAQ.md b/doc/DEVELOPER_FAQ.md
index ca011146ce34..cb2d88693077 100644
--- a/doc/DEVELOPER_FAQ.md
+++ b/doc/DEVELOPER_FAQ.md
@@ -2,23 +2,40 @@
## Adding a monster
-1. Edit `data/json/monsters.json` or create a new json file and insert the definition of your new monster there (probably copy an existing entry).
+1. Edit `data/json/monsters.json` or create a new json file and insert the definition of your new
+ monster there (probably copy an existing entry).
2. Make sure the id value is unique among all other monster types.
-3. Your monster type is now valid, but won't be spawned. If you want it to be spawned among similar monsters, edit monstergroups.json. Find the appropriate array, and insert the identifier for your monster (e.g, `mon_zombie`). `cost_multiplier`, makes it more expensive to spawn. The higher the cost, the more 'slots' it takes up, and freq is how frequent they spawn. See `mongroupdef.cpp`
-4. If you want your monster to drop items, edit `monster_drops.json`. Make a new array for your monster type with all the map item groups it may carry, and a chance value for each.
-5. Your monster may have a special attack, a `monattack::function` reference. Edit `monattack.h` and include the function in the class definition; edit `monstergenerator.cpp` and add the translation, then edit `monattack.cpp` and define your function. Functions may be shared among different monster types. Be aware that the function should contain a statement that the monster uses to decide whether or not to use the attack, and if they do, should reset the monster's attack timer.
-6. Just like attacks, some monsters may have a special function called when they die. This works the same as attacks, but the relevant files are `mondeath.h` and `mondeath.cpp`.
-7. If you add flags, document them in `JSON_FLAGS.md`, and `mtype.h`. Please. Or we will replace your blood with acid in the night.
+3. Your monster type is now valid, but won't be spawned. If you want it to be spawned among similar
+ monsters, edit monstergroups.json. Find the appropriate array, and insert the identifier for your
+ monster (e.g, `mon_zombie`). `cost_multiplier`, makes it more expensive to spawn. The higher the
+ cost, the more 'slots' it takes up, and freq is how frequent they spawn. See `mongroupdef.cpp`
+4. If you want your monster to drop items, edit `monster_drops.json`. Make a new array for your
+ monster type with all the map item groups it may carry, and a chance value for each.
+5. Your monster may have a special attack, a `monattack::function` reference. Edit `monattack.h` and
+ include the function in the class definition; edit `monstergenerator.cpp` and add the
+ translation, then edit `monattack.cpp` and define your function. Functions may be shared among
+ different monster types. Be aware that the function should contain a statement that the monster
+ uses to decide whether or not to use the attack, and if they do, should reset the monster's
+ attack timer.
+6. Just like attacks, some monsters may have a special function called when they die. This works the
+ same as attacks, but the relevant files are `mondeath.h` and `mondeath.cpp`.
+7. If you add flags, document them in `JSON_FLAGS.md`, and `mtype.h`. Please. Or we will replace
+ your blood with acid in the night.
## Adding structures to the map
-Most "regular" buildings are spawned in cities (large clusters of buildings which are located rather close to each other).
+Most "regular" buildings are spawned in cities (large clusters of buildings which are located rather
+close to each other).
-In file `omdata.h` in the enum `oter_id` structure define names (code identifiers) for your building.
+In file `omdata.h` in the enum `oter_id` structure define names (code identifiers) for your
+building.
-If you want your building to be displayed at overmap in different orientations, you should add 4 identifiers for each orientation (`south`, `east`, `west` and `north` correspondingly).
+If you want your building to be displayed at overmap in different orientations, you should add 4
+identifiers for each orientation (`south`, `east`, `west` and `north` correspondingly).
-In the same file in structure `const oter_t oterlist[num_ter_types]` we should define how these buildings will be displayed, how much they obscure vision and which extras set they have. For example:
+In the same file in structure `const oter_t oterlist[num_ter_types]` we should define how these
+buildings will be displayed, how much they obscure vision and which extras set they have. For
+example:
```C++
{"mil. surplus", '^', c_white, 5, build_extras, false, false},
@@ -27,37 +44,54 @@ In the same file in structure `const oter_t oterlist[num_ter_types]` we should d
{"mil. surplus", '<', c_white, 5, build_extras, false, false}
```
-Comments at the beginning of this structure are rather useful. In the file `mapgen.cpp` find the subroutine called `draw_map(...);` where you should find a huge variant operator ("switch"). This is where you should put your code defining the new building by adding new case-statement.
+Comments at the beginning of this structure are rather useful. In the file `mapgen.cpp` find the
+subroutine called `draw_map(...);` where you should find a huge variant operator ("switch"). This is
+where you should put your code defining the new building by adding new case-statement.
It should be mentioned that most buildings are built on the square `SEEX*2 x SEEY*2` tiles.
-If you want your building to be spawned not only in city limits you should refer to structures in file `omdata.h` (starting from the line `#define OMSPEC_FREQ 7`).
+If you want your building to be spawned not only in city limits you should refer to structures in
+file `omdata.h` (starting from the line `#define OMSPEC_FREQ 7`).
-These structures are also commented in source code. Add new identifier in enum `omspec_id` structure before `NUM_OMSPECS` and then add a record in `const overmap_special overmap_specials[NUM_OMSPECS]` array. For example:
+These structures are also commented in source code. Add new identifier in enum `omspec_id` structure
+before `NUM_OMSPECS` and then add a record in `const overmap_special overmap_specials[NUM_OMSPECS]`
+array. For example:
```C++
{ot_toxic_dump, 0, 5, 15, -1, mcat_null, 0, 0, 0, 0, &omspec_place::wilderness,0}
```
-The comments given in source code to structure `struct overmap_special` explain the meaning of these constants in the example above.
+The comments given in source code to structure `struct overmap_special` explain the meaning of these
+constants in the example above.
## Adding a bionic
-1. Edit `data/json/bionics.json` and add your bionic near similar types. See `JSON_INFO.md` for a more in-depth review of the individual fields.
-2. If you want the bionic to be available in the game world as an item, add it to `item_groups.json`, and add a bionic item to `data/json/items/bionics.json`.
-3. Manually code in effects into the appropriate files, for activated bionics edit the `player::activate_bionic` function in `bionics.cpp`.
-4. For bionic ranged weapons add the bionic weapon counterparts to `ranged.json`, give them the `BIONIC_WEAPON` flags.
-5. For bionic close combat weapons add the bionic weapon to `data/json/items/melee.json` give them `NON_STUCK`, `NO_UNWIELD` at least.
+1. Edit `data/json/bionics.json` and add your bionic near similar types. See `JSON_INFO.md` for a
+ more in-depth review of the individual fields.
+2. If you want the bionic to be available in the game world as an item, add it to
+ `item_groups.json`, and add a bionic item to `data/json/items/bionics.json`.
+3. Manually code in effects into the appropriate files, for activated bionics edit the
+ `player::activate_bionic` function in `bionics.cpp`.
+4. For bionic ranged weapons add the bionic weapon counterparts to `ranged.json`, give them the
+ `BIONIC_WEAPON` flags.
+5. For bionic close combat weapons add the bionic weapon to `data/json/items/melee.json` give them
+ `NON_STUCK`, `NO_UNWIELD` at least.
## How armor protection is calculated
-1. When the player is hit at a specific body part, armor coverage determines whether the armor is hit, or an uncovered part of the player is hit (roll `1d100` against coverage).
-2. If the above roll fails (ie roll value is above coverage), then the armor does not absorb any damage from the blow, neither does it become damaged.
-3. If the above roll succeeds, the armor is hit, possibly absorbing some damage and possibly getting damaged in the process.
+1. When the player is hit at a specific body part, armor coverage determines whether the armor is
+ hit, or an uncovered part of the player is hit (roll `1d100` against coverage).
+2. If the above roll fails (ie roll value is above coverage), then the armor does not absorb any
+ damage from the blow, neither does it become damaged.
+3. If the above roll succeeds, the armor is hit, possibly absorbing some damage and possibly getting
+ damaged in the process.
4. The above steps are repeated for each layer of armor on the body part.
-5. Armor protects against bash and cut damage. These are determined by multiplying the armor thickness by the material bash/cut resistance factor respectively, given in `materials.json`.
-6. If the armor is made from 2 materials types, then it takes a weighted average of the primary material (`66%`) and secondary material (`33%`).
-7. Materials resistance factors are given relative to `PAPER` as a material (this probably needs some fine-tuning for balance).
+5. Armor protects against bash and cut damage. These are determined by multiplying the armor
+ thickness by the material bash/cut resistance factor respectively, given in `materials.json`.
+6. If the armor is made from 2 materials types, then it takes a weighted average of the primary
+ material (`66%`) and secondary material (`33%`).
+7. Materials resistance factors are given relative to `PAPER` as a material (this probably needs
+ some fine-tuning for balance).
## Adding an iuse function.
@@ -67,7 +101,8 @@ The comments given in source code to structure `struct overmap_special` explain
## Acid resistance
-This determines how items react to acid fields. Item acid resistances are a weighted average of the materials acid resistance (see `item::acid_resist`):
+This determines how items react to acid fields. Item acid resistances are a weighted average of the
+materials acid resistance (see `item::acid_resist`):
- An item acid resistance of zero means it will get corroded every turn;
- An item acid resistance above zero means it has a chance of getting corroded every turn;
@@ -88,8 +123,22 @@ A: Add the `TRADER_AVOID` flag to it.
**Q: What the heck is up with the map objects?**
-A: The pertinent map objects are submap, mapbuffer, map, and overmap. The submap contains the actual map data, in SEEXxSEEY chunks. Vehicles and spawns are stored in vectors because they are relatively sparse. Submaps live in the single global mapbuffer, MAPBUFFER. Map encapsulates the area around the player that is currently active. It contains an array of MAPSIZE * MAPSIZE submap pointers called grid. This is a 2D array, but is mapped to a 1D array for (questionable) indexing purposes. When the player moves from one submap to another, `map::shift()` is called. It moves the pointers in grid to keep the player centered. The leading edge of grid is populated by `map::load()`. If the submap has been visited before, it's loaded from MAPBUFFER. Otherwise a 2x2 chunk of submaps is generated based on the corresponding overmap square type. An overmap encapsulates the large-scale structure of a map (such as location and layout of cities, forests, rivers, roads, etc...). There are an arbitrary number of overmaps. Additional overmaps are generated as the player enters new areas.
-
-**Q: What is the relation of NPCs/Monsters vs maps. How are they stored, where are they stored? Are they stored?**
-
-A: All npcs are now stored in the overmap. Active npcs only contain the currently active npcs. There is a difference between npc active npc coordinates, and overmap coordinates. So these will need to be changed when saving the npcs. Or else the npcs will be saved at the wrong location. And yes they are stored.
+A: The pertinent map objects are submap, mapbuffer, map, and overmap. The submap contains the actual
+map data, in SEEXxSEEY chunks. Vehicles and spawns are stored in vectors because they are relatively
+sparse. Submaps live in the single global mapbuffer, MAPBUFFER. Map encapsulates the area around the
+player that is currently active. It contains an array of MAPSIZE * MAPSIZE submap pointers called
+grid. This is a 2D array, but is mapped to a 1D array for (questionable) indexing purposes. When the
+player moves from one submap to another, `map::shift()` is called. It moves the pointers in grid to
+keep the player centered. The leading edge of grid is populated by `map::load()`. If the submap has
+been visited before, it's loaded from MAPBUFFER. Otherwise a 2x2 chunk of submaps is generated based
+on the corresponding overmap square type. An overmap encapsulates the large-scale structure of a map
+(such as location and layout of cities, forests, rivers, roads, etc...). There are an arbitrary
+number of overmaps. Additional overmaps are generated as the player enters new areas.
+
+**Q: What is the relation of NPCs/Monsters vs maps. How are they stored, where are they stored? Are
+they stored?**
+
+A: All npcs are now stored in the overmap. Active npcs only contain the currently active npcs. There
+is a difference between npc active npc coordinates, and overmap coordinates. So these will need to
+be changed when saving the npcs. Or else the npcs will be saved at the wrong location. And yes they
+are stored.
diff --git a/doc/DEVELOPER_TOOLING.md b/doc/DEVELOPER_TOOLING.md
index 05dc810f8d3e..afd3da703686 100644
--- a/doc/DEVELOPER_TOOLING.md
+++ b/doc/DEVELOPER_TOOLING.md
@@ -1,8 +1,10 @@
## Code style (astyle)
-Automatic formatting of the source code is performed by [Artistic Style](http://astyle.sourceforge.net/), or `astyle` for short.
+Automatic formatting of the source code is performed by
+[Artistic Style](http://astyle.sourceforge.net/), or `astyle` for short.
-There are multiple ways to invoke it on the codebase, depending on your system or personal preferences.
+There are multiple ways to invoke it on the codebase, depending on your system or personal
+preferences.
### Invoking astyle directly
@@ -22,9 +24,9 @@ make astyle
### Invoking astyle via pre-commit hook
-If you have all the relevant tools installed, you can have git automatically
-check the style of code and json by adding these commands to your git
-pre-commit hook (typically at `.git/hooks/pre-commit`):
+If you have all the relevant tools installed, you can have git automatically check the style of code
+and json by adding these commands to your git pre-commit hook (typically at
+`.git/hooks/pre-commit`):
```BASH
git diff --cached --name-only -z HEAD | grep -z 'data/.*\.json' | \
@@ -35,33 +37,49 @@ make astyle-check || exit 1
### Astyle extensions for Visual Studio
-There are astyle extensions in the Visual Studio Marketplace, but none of them have been confirmed (yet) to correctly work for our purposes on VS2019 or VS2022.
+There are astyle extensions in the Visual Studio Marketplace, but none of them have been confirmed
+(yet) to correctly work for our purposes on VS2019 or VS2022.
#### Visual Studio 2022
-Head over to https://github.com/olanti-p/BN_Astyle and follow instructions in the [README.md](https://github.com/olanti-p/BN_Astyle/blob/master/README.md). You may compile and install the extension from source, or take advantage of the pre-built version in [releases section](https://github.com/olanti-p/BN_Astyle/releases).
+Head over to https://github.com/olanti-p/BN_Astyle and follow instructions in the
+[README.md](https://github.com/olanti-p/BN_Astyle/blob/master/README.md). You may compile and
+install the extension from source, or take advantage of the pre-built version in
+[releases section](https://github.com/olanti-p/BN_Astyle/releases).
#### Visual Studio 2019
-Extensions's source code lives over at https://github.com/lukamicoder/astyle-extension.
-To install and compile it:
-1. Add the `Visual Studio extension development` workload through Visual Studio installer to your VS2019
-2. Download and extract the source code, or clone the repository (a simple `git clone --depth 1 https://github.com/lukamicoder/astyle-extension.git` should do).
+Extensions's source code lives over at https://github.com/lukamicoder/astyle-extension. To install
+and compile it:
+
+1. Add the `Visual Studio extension development` workload through Visual Studio installer to your
+ VS2019
+2. Download and extract the source code, or clone the repository (a simple
+ `git clone --depth 1 https://github.com/lukamicoder/astyle-extension.git` should do).
3. From the root folder, open `astyle-extension/AStyleExtension2017.sln`
-4. Select `Release` build configuration (most likely VS will select `Debug` configuration by default)
+4. Select `Release` build configuration (most likely VS will select `Debug` configuration by
+ default)
5. Build the solution
-6. If the build succeeded, you'll see the compiled extension in `AStyleExtension\bin\Release`. Double click it to install.
-7. Configure the extension according to [Configuration instructions (Visual Studio 2019 or older)](#configuration-instructions-visual-studio-2019-or-older) section.
+6. If the build succeeded, you'll see the compiled extension in `AStyleExtension\bin\Release`.
+ Double click it to install.
+7. Configure the extension according to
+ [Configuration instructions (Visual Studio 2019 or older)](#configuration-instructions-visual-studio-2019-or-older)
+ section.
#### Visual Studio 2017 or earlier
-You may follow the steps for VS2019 to compile from source, but there are pre-built versions [available](https://marketplace.visualstudio.com/items?itemName=Lukamicoder.AStyleExtension2017) on Visual Studio Marketplace, you should be able to install the extension through VS's extension manager and then configure it the same way.
+You may follow the steps for VS2019 to compile from source, but there are pre-built versions
+[available](https://marketplace.visualstudio.com/items?itemName=Lukamicoder.AStyleExtension2017) on
+Visual Studio Marketplace, you should be able to install the extension through VS's extension
+manager and then configure it the same way.
#### Configuration instructions (Visual Studio 2019 or older):
1. Go to `Tools` - `Options` - `AStyle Formatter` - `General`.
-2. Import `https://github.com/CleverRaven/Cataclysm-DDA/blob/master/msvc-full-features/AStyleExtension-Cataclysm-DDA.cfg` on `Export/Import` tab using `Import` button:
+2. Import
+ `https://github.com/CleverRaven/Cataclysm-DDA/blob/master/msvc-full-features/AStyleExtension-Cataclysm-DDA.cfg`
+ on `Export/Import` tab using `Import` button:
![image](img/VS_Astyle_Step_1.png)
@@ -69,11 +87,13 @@ You may follow the steps for VS2019 to compile from source, but there are pre-bu
![image](img/VS_Astyle_Step_2.png)
-4. Close `Options` menu, open file to be astyled and use `Format Document (Astyle)` or `Format Selection (Astyle)` commands from `Edit` - `Advanced` menu.
+4. Close `Options` menu, open file to be astyled and use `Format Document (Astyle)` or
+ `Format Selection (Astyle)` commands from `Edit` - `Advanced` menu.
![image](img/VS_Astyle_Step_3.png)
-*Note:* You can also configure keybindings for aforementioned commands in `Tools` - `Options` - `Environment` - `Keybindings` menu:
+_Note:_ You can also configure keybindings for aforementioned commands in `Tools` - `Options` -
+`Environment` - `Keybindings` menu:
![image](img/VS_Astyle_Step_4.png)
@@ -83,78 +103,81 @@ See the [JSON style guide](JSON_STYLE.md).
## ctags
-In addition to the usual means of creating a `tags` file via e.g. [`ctags`](http://ctags.sourceforge.net/), we provide `tools/json_tools/cddatags.py` to augment a `tags` file with locations of definitions taken from CDDA JSON data. `cddatags.py` is designed to safely update a tags file containing source code tags, so if you want both types of tag in your `tags` file then you can run `ctags -R . && tools/json_tools/cddatags.py`. Alternatively, there is a rule in the `Makefile` to do this for you; just run `make ctags` or `make etags`.
-
+In addition to the usual means of creating a `tags` file via e.g.
+[`ctags`](http://ctags.sourceforge.net/), we provide `tools/json_tools/cddatags.py` to augment a
+`tags` file with locations of definitions taken from CDDA JSON data. `cddatags.py` is designed to
+safely update a tags file containing source code tags, so if you want both types of tag in your
+`tags` file then you can run `ctags -R . && tools/json_tools/cddatags.py`. Alternatively, there is a
+rule in the `Makefile` to do this for you; just run `make ctags` or `make etags`.
## clang-tidy
-Cataclysm has a [clang-tidy configuration file](../.clang-tidy) and if you have
-`clang-tidy` available you can run it to perform static analysis of the
-codebase. We test with `clang-tidy` from LLVM 12.0.0 with CI, so for the most
-consistent results, you might want to use that version.
+Cataclysm has a [clang-tidy configuration file](../.clang-tidy) and if you have `clang-tidy`
+available you can run it to perform static analysis of the codebase. We test with `clang-tidy` from
+LLVM 12.0.0 with CI, so for the most consistent results, you might want to use that version.
To run it, you have a few options.
-* `clang-tidy` ships with a wrapper script `run-clang-tidy.py`.
+- `clang-tidy` ships with a wrapper script `run-clang-tidy.py`.
-* Use CMake's built-in support by adding `-DCMAKE_CXX_CLANG_TIDY=clang-tidy`
- or similar, pointing it to your chosen clang-tidy version.
+- Use CMake's built-in support by adding `-DCMAKE_CXX_CLANG_TIDY=clang-tidy` or similar, pointing it
+ to your chosen clang-tidy version.
+
+- To run `clang-tidy` directly try something like
-* To run `clang-tidy` directly try something like
```sh
grep '"file": "' build/compile_commands.json | \
sed "s+.*$PWD/++;s+\"$++" | \
egrep '.' | \
xargs -P 9 -n 1 clang-tidy -quiet
```
-To focus on a subset of files add their names into the `egrep` regex in the
-middle of the command-line.
+
+To focus on a subset of files add their names into the `egrep` regex in the middle of the
+command-line.
### Custom clang-tidy plugin
-We have written our own clang-tidy checks in a custom plugin. Unfortunately,
-`clang-tidy` as distributed by LLVM doesn't support plugins, so making this
-work requires some extra steps.
+We have written our own clang-tidy checks in a custom plugin. Unfortunately, `clang-tidy` as
+distributed by LLVM doesn't support plugins, so making this work requires some extra steps.
#### Ubuntu Focal
-If you are on Ubuntu Focal then you might be able to get it working the same
-way our CI does. Add the LLVM 12 Focal source [listed
-here](https://apt.llvm.org/) to your `sources.list`, install the needed packages (`clang-12
-libclang-12-dev llvm-12-dev llvm-12-tools`), and build Cataclysm with CMake,
-adding `-DCATA_CLANG_TIDY_PLUGIN=ON`.
+If you are on Ubuntu Focal then you might be able to get it working the same way our CI does. Add
+the LLVM 12 Focal source [listed here](https://apt.llvm.org/) to your `sources.list`, install the
+needed packages (`clang-12 libclang-12-dev llvm-12-dev llvm-12-tools`), and build Cataclysm with
+CMake, adding `-DCATA_CLANG_TIDY_PLUGIN=ON`.
On other distributions you will probably need to build `clang-tidy` yourself.
-* Check out the `llvm`, `clang`, and `clang-tools-extra` repositories in the
- required layout (as described for example
+
+- Check out the `llvm`, `clang`, and `clang-tools-extra` repositories in the required layout (as
+ described for example
[here](https://quuxplusone.github.io/blog/2018/04/16/building-llvm-from-source/).
-* Patch in plugin support for `clang-tidy` using [this
- patch](https://github.com/jbytheway/clang-tidy-plugin-support/blob/master/plugin-support.patch).
-* Configure LLVM using CMake, including the
- `-DCMAKE_EXE_LINKER_FLAGS="-rdynamic"` option.
-* Add the `build/bin` directory to your path so that `clang-tidy` and
- `FileCheck` are found from there.
-
-Then you can use your locally build `clang-tidy` to compile Cataclysm. You'll
-need to use the CMake version of the Cataclysm build rather than the `Makefile`
-build. Add the following CMake options:
+- Patch in plugin support for `clang-tidy` using
+ [this patch](https://github.com/jbytheway/clang-tidy-plugin-support/blob/master/plugin-support.patch).
+- Configure LLVM using CMake, including the `-DCMAKE_EXE_LINKER_FLAGS="-rdynamic"` option.
+- Add the `build/bin` directory to your path so that `clang-tidy` and `FileCheck` are found from
+ there.
+
+Then you can use your locally build `clang-tidy` to compile Cataclysm. You'll need to use the CMake
+version of the Cataclysm build rather than the `Makefile` build. Add the following CMake options:
+
```sh
-DCATA_CLANG_TIDY_PLUGIN=ON
-DCATA_CLANG_TIDY_INCLUDE_DIR="$extra_dir/clang-tidy"
-DCATA_CHECK_CLANG_TIDY="$extra_dir/test/clang-tidy/check_clang_tidy.py"
```
+
where `$extra_dir` is the location of your `clang-tools-extra` checkout.
To run `clang-tidy` with this plugin enabled add the
-`'-plugins=$build_dir/tools/clang-tidy-plugin/libCataAnalyzerPlugin.so'` option
-to your `clang-tidy` command line.
+`'-plugins=$build_dir/tools/clang-tidy-plugin/libCataAnalyzerPlugin.so'` option to your `clang-tidy`
+command line.
-If you wish to run the tests for the custom clang-tidy plugin you will also
-need `lit`. This will be built as part of LLVM, or you can install it via
-`pip` or your local package manager if you prefer.
+If you wish to run the tests for the custom clang-tidy plugin you will also need `lit`. This will be
+built as part of LLVM, or you can install it via `pip` or your local package manager if you prefer.
+
+Then, assuming `build` is your Cataclysm build directory, you can run the tests with
-Then, assuming `build` is your Cataclysm build directory, you can run the tests
-with
```sh
lit -v build/tools/clang-tidy-plugin/test
```
@@ -164,32 +187,33 @@ lit -v build/tools/clang-tidy-plugin/test
##### Build LLVM
To build LLVM on Windows, you'll first need to get some tools installed.
+
- Cmake
- Python 3
-- MinGW-w64 (other compilers may or may not work. Clang itself does not seem to be
-building LLVM on Windows correctly.)
+- MinGW-w64 (other compilers may or may not work. Clang itself does not seem to be building LLVM on
+ Windows correctly.)
- A shell environment
-After the tools are installed, a patch still needs to be applied before building
-LLVM, since `clang-tidy` as distributed by LLVM doesn't support plugins.
+After the tools are installed, a patch still needs to be applied before building LLVM, since
+`clang-tidy` as distributed by LLVM doesn't support plugins.
-First, clone the LLVM repo from, for example, [the official github repo](https://github.com/llvm/llvm-project.git).
-Checkout the `release/12.x` branch, since that's what our patch was based on.
+First, clone the LLVM repo from, for example,
+[the official github repo](https://github.com/llvm/llvm-project.git). Checkout the `release/12.x`
+branch, since that's what our patch was based on.
On Windows, in addition to applying `plugin-support.patch` mentioned in the previous section, you
should also apply
[`clang-tidy-scripts.patch`](https://github.com/jbytheway/clang-tidy-plugin-support/blob/master/clang-tidy-scripts.patch)
-so you can run the lit test with the custom clang-tidy executable and let
-clang-tidy apply suggestions automatically.
+so you can run the lit test with the custom clang-tidy executable and let clang-tidy apply
+suggestions automatically.
-After the patch is applied, you can then build the LLVM code. Unfortunately, it
-seems that clang itself cannot correctly compile the LLVM code on Windows (gives
-some sort of relocation error). Luckily, MinGW-w64 can be used instead to compile
-the code.
+After the patch is applied, you can then build the LLVM code. Unfortunately, it seems that clang
+itself cannot correctly compile the LLVM code on Windows (gives some sort of relocation error).
+Luckily, MinGW-w64 can be used instead to compile the code.
-The first step to build the code is to run CMake to generate the makefile. On
-the root dir of LLVM, run the following script (substitute values inside `<>`
-with the actual paths). Make sure CMake, python, and MinGW-w64 are on the path.
+The first step to build the code is to run CMake to generate the makefile. On the root dir of LLVM,
+run the following script (substitute values inside `<>` with the actual paths). Make sure CMake,
+python, and MinGW-w64 are on the path.
```sh
mkdir -p build
@@ -204,10 +228,10 @@ cmake \
../llvm
```
-The next step is to call `make` to actually build clang-tidy as a library.
-When using MinGW-w64 to build, you should call `mingw32-make` instead.
-Also, because `FileCheck` is not shipped with Windows, you'll also need to build
-it yourself using LLVM sources by adding the `FileCheck` target to the make command.
+The next step is to call `make` to actually build clang-tidy as a library. When using MinGW-w64 to
+build, you should call `mingw32-make` instead. Also, because `FileCheck` is not shipped with
+Windows, you'll also need to build it yourself using LLVM sources by adding the `FileCheck` target
+to the make command.
```sh
mkdir -p build
@@ -215,31 +239,33 @@ cd build
mingw32-make -j4 clang-tidy clangTidyMain FileCheck
```
-Here `clang-tidy` is only added to trigger the building of several targets that
-are needed to build our custom clang-tidy executable later.
+Here `clang-tidy` is only added to trigger the building of several targets that are needed to build
+our custom clang-tidy executable later.
##### Build clang-tidy with custom checks
-After building clang-tidy as a library from the LLVM source, the next step is to
-build clang-tidy as an executable, with the custom checks from the CDDA source.
+After building clang-tidy as a library from the LLVM source, the next step is to build clang-tidy as
+an executable, with the custom checks from the CDDA source.
In this step, the following tools are required.
+
- Python 3
- CMake
- MinGW-w64
- FileCheck (built from the LLVM source)
- A shell environment
-You also need to install yaml for python 3 to work. Download the `.whl` installer
-corresponding to your python version from [here](https://pyyaml.org/wiki/PyYAML)
-and execute the following command inside the `/Scripts` directory
+You also need to install yaml for python 3 to work. Download the `.whl` installer corresponding to
+your python version from [here](https://pyyaml.org/wiki/PyYAML) and execute the following command
+inside the `/Scripts` directory
+
```sh
pip install path/to/your/downloaded/file.whl
```
-Currently, the CDDA source is still building the custom checks as a plugin,
-which unfortunately is not supported on Windows, so the following patch needs to
-be applied before the custom checks can be built as an executable.
+Currently, the CDDA source is still building the custom checks as a plugin, which unfortunately is
+not supported on Windows, so the following patch needs to be applied before the custom checks can be
+built as an executable.
```patch
diff --git a/tools/clang-tidy-plugin/CMakeLists.txt b/tools/clang-tidy-plugin/CMakeLists.txt
@@ -309,18 +335,17 @@ index 496804316a..43beb49653 100644
config.substitutions.append(('%cata_plugin', cata_plugin))
```
-The next step is to run CMake to generate the compilation database. The compilation
-database contains compiler flags that clang-tidy uses to check the source files.
+The next step is to run CMake to generate the compilation database. The compilation database
+contains compiler flags that clang-tidy uses to check the source files.
-Make sure Python 3, CMake, MinGW-w64, and FileCheck are on the path.
-Note that two `bin` directories of MinGW-w64 should be on the path: `/bin`,
-and `/x86_64-w64-mingw32/bin`. FileCheck's path is `/build/bin`,
-if you built it with the instructions in the previous section.
+Make sure Python 3, CMake, MinGW-w64, and FileCheck are on the path. Note that two `bin` directories
+of MinGW-w64 should be on the path: `/bin`, and
+`/x86_64-w64-mingw32/bin`. FileCheck's path is `/build/bin`, if
+you built it with the instructions in the previous section.
-Then add the following CMake options to generate the compilation database
-(substitute values inside `<>` with the actual paths), and build the CDDA source
-and the custom clang-tidy executable with `mingw32-make`. In this tutorial we
-run CMake and `mingw32-make` in the `build` subdirectory.
+Then add the following CMake options to generate the compilation database (substitute values inside
+`<>` with the actual paths), and build the CDDA source and the custom clang-tidy executable with
+`mingw32-make`. In this tutorial we run CMake and `mingw32-make` in the `build` subdirectory.
```sh
-DCMAKE_EXPORT_COMPILE_COMMANDS=ON
@@ -333,22 +358,20 @@ run CMake and `mingw32-make` in the `build` subdirectory.
-DCATA_CHECK_CLANG_TIDY="/clang-tools-extra/test/clang-tidy/check_clang_tidy.py -clang-tidy=/build/tools/clang-tidy-plugin/CataAnalyzerPlugin.exe"
```
-Next, change the directory back to the source root, and run `tools/fix-compilation-database.py`
-with Python 3 to fix some errors in the compilation database. Then the compilation
-database should be usable by clang-tidy.
+Next, change the directory back to the source root, and run `tools/fix-compilation-database.py` with
+Python 3 to fix some errors in the compilation database. Then the compilation database should be
+usable by clang-tidy.
-If you want to check if the custom checks are working correctly, run the following
-script.
+If you want to check if the custom checks are working correctly, run the following script.
```sh
python3 /llvm/utils/lit/lit.py -v build/tools/clang-tidy-plugin/test
```
-Finally, use the following command to run clang-tidy with the custom checks.
-In the following command, the first line of "-extra-arg"s are used to tell
-clang-tidy to mimic g++ behavior, and the second line of "-extra-arg"s are
-used to tell clang-tidy to use clang's x86intrin.h instead of g++'s, so as
-to avoid compiler errors.
+Finally, use the following command to run clang-tidy with the custom checks. In the following
+command, the first line of "-extra-arg"s are used to tell clang-tidy to mimic g++ behavior, and the
+second line of "-extra-arg"s are used to tell clang-tidy to use clang's x86intrin.h instead of
+g++'s, so as to avoid compiler errors.
```sh
python3 /clang-tools-extra/clang-tidy/tool/run-clang-tidy.py \
@@ -358,24 +381,24 @@ python3 /clang-tools-extra/clang-tidy/tool/run-clang-tidy.py \
-extra-arg=-isystem -extra-arg=/clang/lib/Headers
```
-You can also add `-fix-errors` to apply fixes reported by the checks, or
-`-checks="-*,xxx,yyy"` to specify the checks you would like to run.
+You can also add `-fix-errors` to apply fixes reported by the checks, or `-checks="-*,xxx,yyy"` to
+specify the checks you would like to run.
## include-what-you-use
-[include-what-you-use](https://github.com/include-what-you-use/include-what-you-use)
-(IWYU) is a project intended to optimise includes. It will calculate the
-required headers and add and remove includes as appropriate.
+[include-what-you-use](https://github.com/include-what-you-use/include-what-you-use) (IWYU) is a
+project intended to optimise includes. It will calculate the required headers and add and remove
+includes as appropriate.
-Running on this codebase revealed some issues. You will need a version of IWYU
-where the following PR has been merged (which has not yet happened at time of
-writing, but with luck might make it into the clang-10 release of IWYU):
+Running on this codebase revealed some issues. You will need a version of IWYU where the following
+PR has been merged (which has not yet happened at time of writing, but with luck might make it into
+the clang-10 release of IWYU):
-* https://github.com/include-what-you-use/include-what-you-use/pull/775
+- https://github.com/include-what-you-use/include-what-you-use/pull/775
-Once you have IWYU built, build the codebase using cmake, with
-`CMAKE_EXPORT_COMPILE_COMMANDS=ON` on to create a compilation database
-(Look for `compile_commands.json` in the build dir to see whether that worked).
+Once you have IWYU built, build the codebase using cmake, with `CMAKE_EXPORT_COMPILE_COMMANDS=ON` on
+to create a compilation database (Look for `compile_commands.json` in the build dir to see whether
+that worked).
Then run:
@@ -383,46 +406,40 @@ Then run:
iwyu_tool.py -p $CMAKE_BUILD_DIR/compile_commands.json -- -Xiwyu --mapping_file=$PWD/tools/iwyu/cata.imp | fix_includes.py --nosafe_headers --reorder
```
-IWYU will sometimes add C-style library headers which clang-tidy doesn't like,
-so you might need to run clang-tidy (as described above) and then re-run IWYU a
-second time.
-
-There are mapping files in `tools/iwyu` intended to help IWYU pick the right
-headers. Mostly they should be fairly obvious, but the SDL mappings might
-warrant further explanation. We want to force most SDL includes to go via
-`sdl_wrappers.h`, because that handles the platform-dependence issues (the
-include paths are different on Windows). There are a couple of exceptions
-(`SDL_version.h` and `SDL_mixer.h`). The former is because `main.cpp` can't
-include all SDL headers, because they `#define WinMain`. All the mappings in
-`sdl.imp` are designed to make this happen.
-
-We have to use IWYU pragmas in some situations. Some of the reasons are:
-
-* IWYU has a concept of [associated
- headers](https://github.com/include-what-you-use/include-what-you-use/blob/master/docs/IWYUPragmas.md#iwyu-pragma-associated),
- where each cpp file can have some number of such headers. The cpp file is
- expected to define the things declared in those headers. In Cata, the
- mapping between headers and cpp files is not nearly so simple, so there are
- files with multiple associated headers, and files with none. Headers that
- are not the associated header of any cpp file will not get their includes
- updated, which could lead to broken builds, so ideally all headers would be
- associated to some cpp file. You can use the following command to get a list
- of headers which are not currently associated to any cpp file (requires GNU
- sed):
+IWYU will sometimes add C-style library headers which clang-tidy doesn't like, so you might need to
+run clang-tidy (as described above) and then re-run IWYU a second time.
+
+There are mapping files in `tools/iwyu` intended to help IWYU pick the right headers. Mostly they
+should be fairly obvious, but the SDL mappings might warrant further explanation. We want to force
+most SDL includes to go via `sdl_wrappers.h`, because that handles the platform-dependence issues
+(the include paths are different on Windows). There are a couple of exceptions (`SDL_version.h` and
+`SDL_mixer.h`). The former is because `main.cpp` can't include all SDL headers, because they
+`#define WinMain`. All the mappings in `sdl.imp` are designed to make this happen.
+
+We have to use IWYU pragmas in some situations. Some of the reasons are:
+
+- IWYU has a concept of
+ [associated headers](https://github.com/include-what-you-use/include-what-you-use/blob/master/docs/IWYUPragmas.md#iwyu-pragma-associated),
+ where each cpp file can have some number of such headers. The cpp file is expected to define the
+ things declared in those headers. In Cata, the mapping between headers and cpp files is not nearly
+ so simple, so there are files with multiple associated headers, and files with none. Headers that
+ are not the associated header of any cpp file will not get their includes updated, which could
+ lead to broken builds, so ideally all headers would be associated to some cpp file. You can use
+ the following command to get a list of headers which are not currently associated to any cpp file
+ (requires GNU sed):
```
diff <(ls src/*.h | sed 's!.*/!!') <(for i in src/*.cpp; do echo $i; sed -n '/^#include/{p; :loop n; p; /^$/q; b loop}' $i; done | grep 'e "' | grep -o '"[^"]*"' | sort -u | tr -d '"')
```
-* Due to a [clang bug](https://bugs.llvm.org/show_bug.cgi?id=20666), uses in
- template arguments to explicit instantiations are not counted, which leads to
- some need for `IWYU pragma: keep`.
+- Due to a [clang bug](https://bugs.llvm.org/show_bug.cgi?id=20666), uses in template arguments to
+ explicit instantiations are not counted, which leads to some need for `IWYU pragma: keep`.
-* Due to
+- Due to
[these](https://github.com/include-what-you-use/include-what-you-use/blob/4909f206b46809775e9b5381f852eda62cbf4bf7/iwyu.cc#L1617)
[missing](https://github.com/include-what-you-use/include-what-you-use/blob/4909f206b46809775e9b5381f852eda62cbf4bf7/iwyu.cc#L1629)
- features of IWYU, it does not count uses in template arguments to return
- types, which leads to other requirements for `IWYU pragma: keep`.
+ features of IWYU, it does not count uses in template arguments to return types, which leads to
+ other requirements for `IWYU pragma: keep`.
-* IWYU seems to have particular trouble with types used in maps. Have not looked into this in detail, but again worked
- around it with pragmas.
+- IWYU seems to have particular trouble with types used in maps. Have not looked into this in
+ detail, but again worked around it with pragmas.
diff --git a/doc/EFFECTS_JSON.md b/doc/EFFECTS_JSON.md
index 5ac83156ad3a..0d7d3e2e3416 100644
--- a/doc/EFFECTS_JSON.md
+++ b/doc/EFFECTS_JSON.md
@@ -1,31 +1,39 @@
# Effect data
## How to give effects in-game?
+
### Comestibles
-The first way to give a player an effect in-game is through the drug system. To do this your item must have a use_action of type "consume_drug".
-```C++
- "use_action" : {
- "type" : "consume_drug",
- "activation_message" : "You take some oxycodone.",
- "effects" : [
- {
- "id": "pkill3",
- "duration": 20
- },
- {
- "id": "pkill2",
- "duration": 200
- }
- ]
- },
+
+The first way to give a player an effect in-game is through the drug system. To do this your item
+must have a use_action of type "consume_drug".
+
+```C++
+"use_action" : {
+ "type" : "consume_drug",
+ "activation_message" : "You take some oxycodone.",
+ "effects" : [
+ {
+ "id": "pkill3",
+ "duration": 20
+ },
+ {
+ "id": "pkill2",
+ "duration": 200
+ }
+ ]
+},
```
+
Notice the "effects" field. Each effect has four potential fields:
+
```C++
"id" - Required
"duration" - Required
"bp" - This will cause the effect to target this body part specifically
```
+
Valid "bp" entries are (no entry means the effect is untargeted):
+
```C++
"torso"
"head"
@@ -42,120 +50,145 @@ Valid "bp" entries are (no entry means the effect is untargeted):
```
### Creature attacks
-Creatures have an effect field similar to the "consume_drug" entry for items. You can make a creature's attacks apply effects by adding an "attack_effs" entry for the creature.
+
+Creatures have an effect field similar to the "consume_drug" entry for items. You can make a
+creature's attacks apply effects by adding an "attack_effs" entry for the creature.
+
```C++
- "attack_effs": [
- {
- "//": "applying this multiple times makes intensity go up by 3 instead of 1",
- "id": "paralyzepoison",
- "duration": 33
- },
- {
- "id": "paralyzepoison",
- "duration": 33
- },
- {
- "id": "paralyzepoison",
- "duration": 33
- }
- ],
+"attack_effs": [
+ {
+ "//": "applying this multiple times makes intensity go up by 3 instead of 1",
+ "id": "paralyzepoison",
+ "duration": 33
+ },
+ {
+ "id": "paralyzepoison",
+ "duration": 33
+ },
+ {
+ "id": "paralyzepoison",
+ "duration": 33
+ }
+],
```
-The fields for "attack_effs" function identically to the ones for "consume_drug". However, creatures have an additional field:
+
+The fields for "attack_effs" function identically to the ones for "consume_drug". However, creatures
+have an additional field:
+
```C++
"chance" - The percentage chance of the effect being applied on a good hit, defaults to 100%
```
-If a creature successfully damages the player and their chance roll succeeds they will apply
-all of the listed effects to the player. The effects are added one after another.
+
+If a creature successfully damages the player and their chance roll succeeds they will apply all of
+the listed effects to the player. The effects are added one after another.
## Required fields
+
```C++
- "type": "effect_type", - Required
- "id": "xxxx" - Must be unique
+"type": "effect_type", - Required
+"id": "xxxx" - Must be unique
```
## Optional fields
### Max intensity
+
```C++
- "max_intensity": 3 - Used for many later fields, defaults to 1
- "max_effective_intensity" - How many intensity levels will apply effects.
- Other intensity levels will only increase duration.
+"max_intensity": 3 - Used for many later fields, defaults to 1
+"max_effective_intensity" - How many intensity levels will apply effects.
+ Other intensity levels will only increase duration.
```
### Name
+
```C++
- "name": ["XYZ"]
- or
- "name": [
- "ABC",
- "XYZ",
- "123"
- ]
+"name": ["XYZ"]
+or
+"name": [
+ "ABC",
+ "XYZ",
+ "123"
+]
```
-If "max_intensity" > 1 and the number of entries in "name" >= "max_intensity" then
-it will attempt to use the proper intensity name. In this case that means an intensity
-of 1 would give the name "ABC", 2 would give "XYZ", and 3 would give "123". If "max_intensity" == 1
-or the number of entries in "name" is less than "max_intensity", it will use the first entry followed by
-the intensity in brackets if the current intensity > 1, i.e. "ABC", "ABC [2]", "ABC [3]". If the desired
-entry of "name" is the empty string ("") or "name" is missing then the effect will not display to the player
+
+If "max_intensity" > 1 and the number of entries in "name" >= "max_intensity" then it will attempt
+to use the proper intensity name. In this case that means an intensity of 1 would give the name
+"ABC", 2 would give "XYZ", and 3 would give "123". If "max_intensity" == 1 or the number of entries
+in "name" is less than "max_intensity", it will use the first entry followed by the intensity in
+brackets if the current intensity > 1, i.e. "ABC", "ABC [2]", "ABC [3]". If the desired entry of
+"name" is the empty string ("") or "name" is missing then the effect will not display to the player
in the status screen.
Each entry in "name" can also have an optional context:
+
```JSON
- "name": [ { "ctxt": "ECIG", "str": "Smoke" } ]
+"name": [ { "ctxt": "ECIG", "str": "Smoke" } ]
```
-In this case, the game will translate the name with the given context "ECIG",
-which makes it possible to distinguish the verb "Smoke" from the noun "Smoke"
-in other languages.
+
+In this case, the game will translate the name with the given context "ECIG", which makes it
+possible to distinguish the verb "Smoke" from the noun "Smoke" in other languages.
```C++
- "speed_name" : "XYZ" - Defaults to the first name value
+"speed_name" : "XYZ" - Defaults to the first name value
```
-This is the value used in the list of modifiers on a player's speed. It will default to the first entry in "name"
-if it doesn't exist, and if neither one exists or if "speed_name" is the empty string (""), then it will not
-appear in the list of modifiers on the players speed (though the effect might still have an effect).
+
+This is the value used in the list of modifiers on a player's speed. It will default to the first
+entry in "name" if it doesn't exist, and if neither one exists or if "speed_name" is the empty
+string (""), then it will not appear in the list of modifiers on the players speed (though the
+effect might still have an effect).
### Descriptions
+
```C++
- "desc": ["XYZ"]
- or
- "desc": [
- "ABC",
- "XYZ",
- "123"
- ]
+"desc": ["XYZ"]
+or
+"desc": [
+ "ABC",
+ "XYZ",
+ "123"
+]
```
-Descriptions operate identically to the name field when picking which one to use. In general, descriptions
-should be only 1 line. Stats and effects do not need to be included, and will be automatically generated
-from the other effect data. Should a description line be the empty string ("") it will only display the
-stat changes in the effect description.
+
+Descriptions operate identically to the name field when picking which one to use. In general,
+descriptions should be only 1 line. Stats and effects do not need to be included, and will be
+automatically generated from the other effect data. Should a description line be the empty string
+("") it will only display the stat changes in the effect description.
Descriptions also have a second field that can act as a modifier:
+
```C++
- "part_descs": true - Defaults to false if not present
+"part_descs": true - Defaults to false if not present
```
-If "part_descs" == true then descriptions are preceded by "Your X", where X is the body part name, meaning
-the prior descriptions would appear as "Your left arm ABC".
+
+If "part_descs" == true then descriptions are preceded by "Your X", where X is the body part name,
+meaning the prior descriptions would appear as "Your left arm ABC".
Descriptions can also have a reduced form:
+
```C++
- "reduced_desc": ["XYZ"]
- or
- "reduced_desc": [
- "ABC",
- "XYZ",
- "123"
- ]
+"reduced_desc": ["XYZ"]
+or
+"reduced_desc": [
+ "ABC",
+ "XYZ",
+ "123"
+]
```
-This is the description that will be used if an effect is reduced. By default this will use the normal description
-if it doesn't exist.
+
+This is the description that will be used if an effect is reduced. By default this will use the
+normal description if it doesn't exist.
### Rating
+
```C++
- "rating": "good" - Defaults to "neutral" if missing
+"rating": "good" - Defaults to "neutral" if missing
```
-This is used for how the messages when the effect is applied and removed are displayed. Also this affects "blood_analysis_description" (see below) field: effects with "good" rating will be colored green, effects with any other rating will be colored red when character conducts a blood analysis through some means.
-Valid entries are:
+
+This is used for how the messages when the effect is applied and removed are displayed. Also this
+affects "blood_analysis_description" (see below) field: effects with "good" rating will be colored
+green, effects with any other rating will be colored red when character conducts a blood analysis
+through some means. Valid entries are:
+
```C++
"good"
"neutral"
@@ -164,154 +197,195 @@ Valid entries are:
```
### Messages
+
```C++
- "apply_message": "message",
- "remove_message": "message"
+"apply_message": "message",
+"remove_message": "message"
```
-If the "apply_message" or "remove_message" fields exist, the respective message will be
-displayed upon the addition or removal of the effect. Note: "apply_message" will only display
-if the effect is being added, not if it is simply incrementing a current effect (so only new bites, etc.).
+
+If the "apply_message" or "remove_message" fields exist, the respective message will be displayed
+upon the addition or removal of the effect. Note: "apply_message" will only display if the effect is
+being added, not if it is simply incrementing a current effect (so only new bites, etc.).
### Memorial Log
+
```C++
- "apply_memorial_log": "log",
- "remove_memorial_log": "log"
+"apply_memorial_log": "log",
+"remove_memorial_log": "log"
```
-If the "apply_memorial_log" or "remove_memorial_log" fields exist, the game will add the
-respective message to the memorial log on addition or removal of the effect. Similar to
-the message fields the "apply_memorial_log" will only be added to the log for new effect additions.
+
+If the "apply_memorial_log" or "remove_memorial_log" fields exist, the game will add the respective
+message to the memorial log on addition or removal of the effect. Similar to the message fields the
+"apply_memorial_log" will only be added to the log for new effect additions.
### Resistances
+
```C++
- "resist_trait": "NOPAIN",
- "resist_effect": "flumed"
+"resist_trait": "NOPAIN",
+"resist_effect": "flumed"
```
+
These fields are used to determine if an effect is being resisted or not. If the player has the
-matching trait or effect then they are "resisting" the effect, which changes its effects and description.
-Effects can only have one "resist_trait" and one "resist_effect" at a time.
+matching trait or effect then they are "resisting" the effect, which changes its effects and
+description. Effects can only have one "resist_trait" and one "resist_effect" at a time.
### Removes effects
+
```C++
- "removes_effects": ["bite", "flu"]
+"removes_effects": ["bite", "flu"]
```
-This field will cause an effect to automatically remove any other copies of the listed effects if they are present.
-In the example above the placed effect would automatically cure any bite wounds or flu the player had. Any values here
-automatically count for "blocks_effects" as well, no need to duplicate them there.
+
+This field will cause an effect to automatically remove any other copies of the listed effects if
+they are present. In the example above the placed effect would automatically cure any bite wounds or
+flu the player had. Any values here automatically count for "blocks_effects" as well, no need to
+duplicate them there.
### Blocks effects
+
```C++
- "blocks_effects": ["cold", "flu"]
+"blocks_effects": ["cold", "flu"]
```
-This field will cause an effect to prevent the placement of the listed effects. In the example above the effect would
-prevent the player from catching the cold or the flu (BUT WOULD NOT CURE ANY ONGOING COLDS OR FLUS). Any effects present
-in "removes_effects" are automatically added to "blocks_effects", no need for manual duplication.
+
+This field will cause an effect to prevent the placement of the listed effects. In the example above
+the effect would prevent the player from catching the cold or the flu (BUT WOULD NOT CURE ANY
+ONGOING COLDS OR FLUS). Any effects present in "removes_effects" are automatically added to
+"blocks_effects", no need for manual duplication.
### Effect limiters
+
```C++
- "max_duration": 100,
- "dur_add_perc": 150 - Defaults to 100%
+"max_duration": 100,
+"dur_add_perc": 150 - Defaults to 100%
```
-These are utilized when adding to currently existing effects. "max_duration" limits the overall duration of the effect.
-"dur_add_perc" is the percentage value of the normal duration for adding to an existing. An example:
-1) I add effect A to the player for 100 ticks.
-2) I add effect A to the player again for 100 ticks.
-Because the "dur_add_perc" = 150 in the example above, the second addition adds a total of 100 * 150% = 150 ticks, for
-a total value of 250 ticks from the two. This can also be below 100%, and should be able to even be negative, leading to
-future applications decreasing the overall time left.
+
+These are utilized when adding to currently existing effects. "max_duration" limits the overall
+duration of the effect. "dur_add_perc" is the percentage value of the normal duration for adding to
+an existing. An example:
+
+1. I add effect A to the player for 100 ticks.
+2. I add effect A to the player again for 100 ticks. Because the "dur_add_perc" = 150 in the example
+ above, the second addition adds a total of 100 * 150% = 150 ticks, for a total value of 250 ticks
+ from the two. This can also be below 100%, and should be able to even be negative, leading to
+ future applications decreasing the overall time left.
### Intensities
+
Intensities are used to control effect effects, names, and descriptions. They are defined with:
+
```C++
- "int_add_val": 2 - Defaults to 0! This means future applications will not increase intensity unless changed!
- and/or
- "int_decay_step": -2, - Defaults to -1
- "int_decay_tick": 10
- or
- "int_dur_factor": 700
+"int_add_val": 2 - Defaults to 0! This means future applications will not increase intensity unless changed!
+and/or
+"int_decay_step": -2, - Defaults to -1
+"int_decay_tick": 10
+or
+"int_dur_factor": 700
```
-The first value is the amount an intensity will be incremented if adding to an already existing effect. As an example:
-1) I add effect A to the player
-2) I add effect A to the player again
-Because "int_add_val" = 2, the second addition will change the effect intensity from 1 to 1 + 2 = 3.
-NOTE: You must have at least one of the 3 intensity data sets for intensity to do anything!
-"int_decay_step" and "int_decay_tick" require one another to do anything. If both exist then the game will automatically
-increment the current effect intensities by "int_decay_step" every "int_decay_tick" ticks, capping the result at [1, "max_intensity"].
-This can be used to make effects automatically increase or decrease in intensity over time.
+The first value is the amount an intensity will be incremented if adding to an already existing
+effect. As an example:
+
+1. I add effect A to the player
+2. I add effect A to the player again Because "int_add_val" = 2, the second addition will change the
+ effect intensity from 1 to 1 + 2 = 3. NOTE: You must have at least one of the 3 intensity data
+ sets for intensity to do anything!
+
+"int_decay_step" and "int_decay_tick" require one another to do anything. If both exist then the
+game will automatically increment the current effect intensities by "int_decay_step" every
+"int_decay_tick" ticks, capping the result at [1, "max_intensity"]. This can be used to make effects
+automatically increase or decrease in intensity over time.
-"int_dur_factor" overrides the other three intensities fields, and forces the intensity to be a number defined as
-intensity = duration / "int_dur_factor" rounded up (so from 0 to "int_dur_factor" is intensity 1).
+"int_dur_factor" overrides the other three intensities fields, and forces the intensity to be a
+number defined as intensity = duration / "int_dur_factor" rounded up (so from 0 to "int_dur_factor"
+is intensity 1).
### Permanence
-An effect that is permanent does not lose duration with time.
-That is, even if its duration is 1 turn, it will last until removed.
+
+An effect that is permanent does not lose duration with time. That is, even if its duration is 1
+turn, it will last until removed.
+
```C++
- "permanent": true
+"permanent": true
```
### Miss messages
+
```C++
- "miss_messages": [["Your blisters distract you", 1]]
- or
- "miss_messages": [
- ["Your blisters distract you", 1],
- ["Your blisters don't like you", 10],
- ]
+"miss_messages": [["Your blisters distract you", 1]]
+or
+"miss_messages": [
+ ["Your blisters distract you", 1],
+ ["Your blisters don't like you", 10],
+]
```
+
This will add the following miss messages at the given chances while the effect is in effect.
### Decay messages
+
```C++
- "decay_messages": [["The jet injector's chemicals wear off. You feel AWFUL!", "bad"]]
- or
- "decay_messages": [
- ["The jet injector's chemicals wear off. You feel AWFUL!", "bad"],
- ["OOGA-BOOGA. You feel AWFUL!", "bad"],
- ]
+"decay_messages": [["The jet injector's chemicals wear off. You feel AWFUL!", "bad"]]
+or
+"decay_messages": [
+ ["The jet injector's chemicals wear off. You feel AWFUL!", "bad"],
+ ["OOGA-BOOGA. You feel AWFUL!", "bad"],
+]
```
-The messages are matched to intensities, so the first message is for intensity 1, the second for intensity 2, and
-so on. The messages will print whenever the intensity decreases to their matching intensity from a higher intensity,
-whether through decay ticks or through "int_dur_factor". So if it decayed to intensity 2 from 3+ it would display
-"OOGA-BOOGA. You feel AWFUL!" as a bad message to the player.
+
+The messages are matched to intensities, so the first message is for intensity 1, the second for
+intensity 2, and so on. The messages will print whenever the intensity decreases to their matching
+intensity from a higher intensity, whether through decay ticks or through "int_dur_factor". So if it
+decayed to intensity 2 from 3+ it would display "OOGA-BOOGA. You feel AWFUL!" as a bad message to
+the player.
### Targeting modifiers
+
```C++
- "main_parts_only": true - Defaults to false
+"main_parts_only": true - Defaults to false
```
+
This automatically retargets any effect on a non-main part (hands, eyes, feet, etc.) to the matching
main part (arms, head, legs, etc.).
### Effect modifiers
+
```C++
- "pkill_addict_reduces": true, - Defaults to false
- "pain_sizing": true, - Defaults to false
- "hurt_sizing": true, - Defaults to false
- "harmful_cough": true - Defaults to false
+"pkill_addict_reduces": true, - Defaults to false
+"pain_sizing": true, - Defaults to false
+"hurt_sizing": true, - Defaults to false
+"harmful_cough": true - Defaults to false
```
-"pkill_addict_reduces" makes a player's addiction to painkillers reduce the chance of the effect giving
-them more pkill. "pain_sizing" and "hurt_sizing" cause large/huge mutations to affect the chance of pain
-and hurt effects triggering. "harmful_cough" means that the coughs caused by this effect can hurt the player.
+
+"pkill_addict_reduces" makes a player's addiction to painkillers reduce the chance of the effect
+giving them more pkill. "pain_sizing" and "hurt_sizing" cause large/huge mutations to affect the
+chance of pain and hurt effects triggering. "harmful_cough" means that the coughs caused by this
+effect can hurt the player.
### Morale
+
```C++
- "morale": "morale_high"
+"morale": "morale_high"
```
-Type of morale effect provided. Mandatory if there is a morale effect, must not be specified otherwise.
+
+Type of morale effect provided. Mandatory if there is a morale effect, must not be specified
+otherwise.
### Effect effects
+
```C++
- "base_mods" : {
- arguments
- },
- "scaling_mods": {
- arguments
- }
+"base_mods" : {
+ arguments
+},
+"scaling_mods": {
+ arguments
+}
```
-This is where the real meat of the effect JSON definition lies. Each one can take a variety of arguments.
-Decimals are valid but must be formatted as "0.X" or "-0.X". The game will round towards zero at the end
-when calculating actually applied values
+
+This is where the real meat of the effect JSON definition lies. Each one can take a variety of
+arguments. Decimals are valid but must be formatted as "0.X" or "-0.X". The game will round towards
+zero at the end when calculating actually applied values
Basic definitions:
+
```C++
"X_amount" - Amount applied of X when effect is placed. Like apply messages it will only trigger on new effects
"X_min" - Minimum amount of X applied when roll triggers
@@ -324,6 +398,7 @@ Basic definitions:
```
Valid arguments:
+
```C++
"str_mod" - Positive values raises stat, negative values lowers stat
"dex_mod" - Positive values raises stat, negative values lowers stat
@@ -445,73 +520,79 @@ Valid arguments:
"healing_torso" - Percentage of healing value for torso
"morale" - Amount of morale provided. Must be a single number (resistance not supported).
-
```
+
Each argument can also take either one or two values.
+
```C++
- "thirst_min": [1]
- or
- "thirst_min": [1, 2]
+"thirst_min": [1]
+or
+"thirst_min": [1, 2]
```
-If an effect is "resisted" (either through "resist_effect" or "resist_trait") then it will use the second
-value. If there is only one value given it will always use that amount.
-Base mods and Scaling mods:
-While on intensity = 1 an effect will only have the basic effects of its "base_mods". However for each
-additional intensity it gains it adds the value of each of its "scaling_mods" to the calculations. So:
+If an effect is "resisted" (either through "resist_effect" or "resist_trait") then it will use the
+second value. If there is only one value given it will always use that amount.
+
+Base mods and Scaling mods: While on intensity = 1 an effect will only have the basic effects of its
+"base_mods". However for each additional intensity it gains it adds the value of each of its
+"scaling_mods" to the calculations. So:
+
```C++
Intensity 1 values = base_mods values
Intensity 2 values = base_mods values + scaling_mods values
Intensity 3 values = base_mods values + 2 * scaling_mods values
Intensity 4 values = base_mods values + 3 * scaling_mods values
```
+
and so on.
-Special case:
-The only special case is if base_mods' "X_chance_bot" + intensity * scaling_mods' "X_chance_bot" = 0 then it treats it
-as if it were equal to 1 (i.e. trigger every time)
+Special case: The only special case is if base_mods' "X_chance_bot" + intensity * scaling_mods'
+"X_chance_bot" = 0 then it treats it as if it were equal to 1 (i.e. trigger every time)
## Example Effect
+
```C++
- "type": "effect_type",
- "id": "drunk",
- "name": [
- "Tipsy",
- "Drunk",
- "Trashed",
- "Wasted"
- ],
- "max_intensity": 4,
- "apply_message": "You feel lightheaded.",
- "int_dur_factor": 1000,
- "miss_messages": [["You feel woozy.", 1]],
- "morale": "morale_drunk",
- "base_mods": {
- "str_mod": [1],
- "vomit_chance": [-43],
- "sleep_chance": [-1003],
- "sleep_min": [2500],
- "sleep_max": [3500],
- "morale": [ 5 ]
- },
- "scaling_mods": {
- "str_mod": [-0.67],
- "per_mod": [-1],
- "dex_mod": [-1],
- "int_mod": [-1.42],
- "vomit_chance": [21],
- "sleep_chance": [501],
- "morale": [ 10 ]
- }
-```
-First when "drunk" is applied to the player if they aren't already drunk it prints the message,
-"You feel lightheaded". It also adds the "You feel woozy" miss message for as long as it is applied.
-It has "int_dur_factor": 1000, meaning that its intensity will always be equal to its duration / 1000 rounded up, and
-it has "max_intensity": 4 meaning the highest its intensity will go is 4 at a duration of 3000 or higher.
-As it moves up through the different intensities, its name will change. Its description will simply display the stat
-changes, with no additional description added.
+"type": "effect_type",
+"id": "drunk",
+"name": [
+ "Tipsy",
+ "Drunk",
+ "Trashed",
+ "Wasted"
+],
+"max_intensity": 4,
+"apply_message": "You feel lightheaded.",
+"int_dur_factor": 1000,
+"miss_messages": [["You feel woozy.", 1]],
+"morale": "morale_drunk",
+"base_mods": {
+ "str_mod": [1],
+ "vomit_chance": [-43],
+ "sleep_chance": [-1003],
+ "sleep_min": [2500],
+ "sleep_max": [3500],
+ "morale": [ 5 ]
+},
+"scaling_mods": {
+ "str_mod": [-0.67],
+ "per_mod": [-1],
+ "dex_mod": [-1],
+ "int_mod": [-1.42],
+ "vomit_chance": [21],
+ "sleep_chance": [501],
+ "morale": [ 10 ]
+}
+```
+
+First when "drunk" is applied to the player if they aren't already drunk it prints the message, "You
+feel lightheaded". It also adds the "You feel woozy" miss message for as long as it is applied. It
+has "int_dur_factor": 1000, meaning that its intensity will always be equal to its duration / 1000
+rounded up, and it has "max_intensity": 4 meaning the highest its intensity will go is 4 at a
+duration of 3000 or higher. As it moves up through the different intensities, its name will change.
+Its description will simply display the stat changes, with no additional description added.
As it moves up through the intensity levels its effects will be:
+
```C++
Intensity 1
+1 STR
@@ -544,7 +625,10 @@ Intensity 4
```
### Blood analysis description
+
```C++
- "blood_analysis_description": "Minor Painkiller"
+"blood_analysis_description": "Minor Painkiller"
```
-This description will be displayed for every effect which has this field when character conducts a blood analysis (for example, through Blood Analysis CBM).
\ No newline at end of file
+
+This description will be displayed for every effect which has this field when character conducts a
+blood analysis (for example, through Blood Analysis CBM).
diff --git a/doc/FACTIONS.md b/doc/FACTIONS.md
index 7ac113c732ec..0ce0c9a2fc95 100644
--- a/doc/FACTIONS.md
+++ b/doc/FACTIONS.md
@@ -3,78 +3,84 @@
An NPC faction looks like this:
```json
- {
- "type": "faction",
- "id": "free_merchants",
- "name": "The Free Merchants",
- "likes_u": 30,
- "respects_u": 30,
- "known_by_u": false,
- "size": 100,
- "power": 100,
- "food_supply": 115200,
- "lone_wolf_faction": true,
- "wealth": 75000000,
- "currency": "FMCNote",
- "relations": {
- "free_merchants": {
- "kill on sight": false,
- "watch your back": true,
- "share my stuff": true,
- "guard your stuff": true,
- "lets you in": true,
- "defends your space": true,
- "knows your voice": true
- },
- "old_guard": {
- "kill on sight": false,
- "watch your back": true,
- "share my stuff": false,
- "guard your stuff": true,
- "lets you in": true,
- "defends your space": true,
- "knows your voice": true
- },
- "hells_raiders": {
- "kill on sight": true
- }
- },
- "description": "A conglomeration of entrepreneurs and businessmen that stand together to hammer-out an existence through trade and industry."
- },
+{
+ "type": "faction",
+ "id": "free_merchants",
+ "name": "The Free Merchants",
+ "likes_u": 30,
+ "respects_u": 30,
+ "known_by_u": false,
+ "size": 100,
+ "power": 100,
+ "food_supply": 115200,
+ "lone_wolf_faction": true,
+ "wealth": 75000000,
+ "currency": "FMCNote",
+ "relations": {
+ "free_merchants": {
+ "kill on sight": false,
+ "watch your back": true,
+ "share my stuff": true,
+ "guard your stuff": true,
+ "lets you in": true,
+ "defends your space": true,
+ "knows your voice": true
+ },
+ "old_guard": {
+ "kill on sight": false,
+ "watch your back": true,
+ "share my stuff": false,
+ "guard your stuff": true,
+ "lets you in": true,
+ "defends your space": true,
+ "knows your voice": true
+ },
+ "hells_raiders": {
+ "kill on sight": true
+ }
+ },
+ "description": "A conglomeration of entrepreneurs and businessmen that stand together to hammer-out an existence through trade and industry."
+},
```
-Field | Meaning
--- | --
-`"type"` | string, must be `"faction"`
-`"id"` | string, unique faction id
-`"name"` | string, the faction's common name
-`"likes_u"` | integer, the faction's starting opinion of the player. `"likes_u"` can be increased or decreased in play. If it goes below -10, members of the faction will be hostile.
-`"respects_u"` | integer, the faction's starting opinionof the player. Has no meaningful effect in game and may be removed in the future.
-`"known_by_u"` | boolean, whether the player has met members of the faction. Can be changed in play. Unknown factions will not be displayed in the faction menu.
-`"size"` | integer, an approximate count of the members of the faction. Has no effect in play currently.
-`"power"` | integer, an approximation of the faction's power. Has no effect in play currently.
-`"food_supply"` | integer, the number of calories available to the faction. Has no effect in play currently.
-`"wealth"` | integer, number of post-apocalyptic currency in cents that that faction has to purchase stuff.
-`"currency"` | string, the item `"id"` of the faction's preferred currency. Faction shopkeeps will trade faction current at 100% value, for both selling and buying.
-`"relations"` | dictionary, a description of how the faction sees other factions. See below
-`"mon_faction"` | string, optional. The monster faction `"name"` of the monster faction that this faction counts as. Defaults to "human" if unspecified.
-`"lone_wolf_faction"` | bool, optional. This is a proto/micro faction template that is used to generate 1-person factions for dynamically spawned NPCs, defaults to "false" if unspecified.
+| Field | Meaning |
+| --------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `"type"` | string, must be `"faction"` |
+| `"id"` | string, unique faction id |
+| `"name"` | string, the faction's common name |
+| `"likes_u"` | integer, the faction's starting opinion of the player. `"likes_u"` can be increased or decreased in play. If it goes below -10, members of the faction will be hostile. |
+| `"respects_u"` | integer, the faction's starting opinionof the player. Has no meaningful effect in game and may be removed in the future. |
+| `"known_by_u"` | boolean, whether the player has met members of the faction. Can be changed in play. Unknown factions will not be displayed in the faction menu. |
+| `"size"` | integer, an approximate count of the members of the faction. Has no effect in play currently. |
+| `"power"` | integer, an approximation of the faction's power. Has no effect in play currently. |
+| `"food_supply"` | integer, the number of calories available to the faction. Has no effect in play currently. |
+| `"wealth"` | integer, number of post-apocalyptic currency in cents that that faction has to purchase stuff. |
+| `"currency"` | string, the item `"id"` of the faction's preferred currency. Faction shopkeeps will trade faction current at 100% value, for both selling and buying. |
+| `"relations"` | dictionary, a description of how the faction sees other factions. See below |
+| `"mon_faction"` | string, optional. The monster faction `"name"` of the monster faction that this faction counts as. Defaults to "human" if unspecified. |
+| `"lone_wolf_faction"` | bool, optional. This is a proto/micro faction template that is used to generate 1-person factions for dynamically spawned NPCs, defaults to "false" if unspecified. |
## Faction relations
-Factions can have relations with each other that apply to each member of the faction. Faction relationships are not reciprocal: members of the Free Merchants will defend members of the Lobby Beggars, but members of the Lobby Beggars will not defend members of the Free Merchants.
-Faction relationships are stored in a dictionary, with the dictionary keys being the name of the faction and the values being a second dictionary of boolean flags. All flags are optional and default to false. The faction with the dictionary is the acting faction and the other faction is the object faction.
+Factions can have relations with each other that apply to each member of the faction. Faction
+relationships are not reciprocal: members of the Free Merchants will defend members of the Lobby
+Beggars, but members of the Lobby Beggars will not defend members of the Free Merchants.
+
+Faction relationships are stored in a dictionary, with the dictionary keys being the name of the
+faction and the values being a second dictionary of boolean flags. All flags are optional and
+default to false. The faction with the dictionary is the acting faction and the other faction is the
+object faction.
The flags have the following meanings:
-Flag | Meaning
--- | --
-`"kill on sight"` | Members of the acting faction are always hostile to members of the object faction.
-`"watch your back"` | Members of the acting faction will treat attacks on members of the object faction as attacks on themselves.
-`"share my stuff"` | Members of the acting faction will not object if members of the object faction take items owned by the acting faction.
-`"guard your stuff"` | Members of the acting faction will object if someone takes items owned by the object faction.
-`"lets you in"` | Members of the acting faction will not object if a member of the object faction enters territory controlled by the acting faction.
-`"defends your space"` | Members of the acting faction will become hostile if someone enters territory controlled by the object faction.
-`"knows your voice"` | Members of the acting faction will not comment on speech by members of the object faction.
+| Flag | Meaning |
+| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------- |
+| `"kill on sight"` | Members of the acting faction are always hostile to members of the object faction. |
+| `"watch your back"` | Members of the acting faction will treat attacks on members of the object faction as attacks on themselves. |
+| `"share my stuff"` | Members of the acting faction will not object if members of the object faction take items owned by the acting faction. |
+| `"guard your stuff"` | Members of the acting faction will object if someone takes items owned by the object faction. |
+| `"lets you in"` | Members of the acting faction will not object if a member of the object faction enters territory controlled by the acting faction. |
+| `"defends your space"` | Members of the acting faction will become hostile if someone enters territory controlled by the object faction. |
+| `"knows your voice"` | Members of the acting faction will not comment on speech by members of the object faction. |
So far, only `"kill on sight"`, `"knows your voice"`, and `"watch your back"` have been implemented.
diff --git a/doc/GAME_BALANCE.md b/doc/GAME_BALANCE.md
index 2e7ef664f27f..0284f437c0ea 100644
--- a/doc/GAME_BALANCE.md
+++ b/doc/GAME_BALANCE.md
@@ -1,4 +1,5 @@
# Stat system scaling:
+
Minimum stat: 0 (should only happen due to penalties, instant failure in most scenarios)
Nominal stat: 8 ("average" person)
@@ -7,14 +8,14 @@ Very high stat: 14 (realistic world class human, maximum cost-effective in charg
Maximal stat: 20 (higher may be achievable, but we're not worried about balancing at that point.)
-
# Skill system scaling:
+
Minimum skill: 0 (no training)
Maximum skill: 10 (requires regular training to maintain, "professional" level)
-
# Monster melee skill scaling:
+
Minimum skill: 0 (no melee potential; turret, fungal wall)
Nominal skill: 4 (average critter; most zeds & giant insects)
@@ -25,264 +26,416 @@ Very high skill: 8 (dangerous opponent; dark wyrm, vinebeast)
Maximal skill: 10 (highest for balance purposes; jabberwock, tribot, shoggoth, gracken)
-
# Speeds:
-Zombies are a bit faster than "shambling". Zombified versions of fast critters will remain fast, but in general the process slows the undead version. Further, under no circumstances should a zed be more than 50% faster than base character speed. Currently, this means "capped at 150".
+Zombies are a bit faster than "shambling". Zombified versions of fast critters will remain fast, but
+in general the process slows the undead version. Further, under no circumstances should a zed be
+more than 50% faster than base character speed. Currently, this means "capped at 150".
# Dodge System assumptions:
+
Dodge chance is based on attacker's melee skill and target's dex stat and dodge skill.
Successful dodges negate the attack and impose a cumulative penalty on dodges within the same turn.
## Dodge Use Cases:
-An individual with no skill and nominal stats in ideal circumstances against a basic opponent should occasionally be able to dodge.
-An individual with no skill and nominal stats in ideal circumstances against a skilled opponent should rarely if ever be able to dodge.
+An individual with no skill and nominal stats in ideal circumstances against a basic opponent should
+occasionally be able to dodge.
-An individual with world-class dodging ability, in ideal circumstances against a basic opponent should have a negligible chance of failure.
+An individual with no skill and nominal stats in ideal circumstances against a skilled opponent
+should rarely if ever be able to dodge.
-An individual with world-class dodging ability, in ideal circumstances against a skilled opponent should have a moderate chance of failure.
+An individual with world-class dodging ability, in ideal circumstances against a basic opponent
+should have a negligible chance of failure.
-The effect of increasing dodge skill has a growth rate with diminishing returns that accelerates sharply at the point where you move beyond the dodge a "regular" character is likely to achieve (7 and above)
+An individual with world-class dodging ability, in ideal circumstances against a skilled opponent
+should have a moderate chance of failure.
-The balance of melee versus dodge should favor dodge which, after all, isn't effective against a wide variety of other types of attacks.
+The effect of increasing dodge skill has a growth rate with diminishing returns that accelerates
+sharply at the point where you move beyond the dodge a "regular" character is likely to achieve (7
+and above)
-Even a world class dodger should not be able to dodge continuously when attacked many times a turn.
+The balance of melee versus dodge should favor dodge which, after all, isn't effective against a
+wide variety of other types of attacks.
+Even a world class dodger should not be able to dodge continuously when attacked many times a turn.
# MELEE WEAPONS:
+
## To-Hit Bonuses
-To-hit bonuses start at '-2' and are modified as follows for weapons that have the following properties:
+
+To-hit bonuses start at '-2' and are modified as follows for weapons that have the following
+properties:
### Grip
+
Grip is a measure of how well you can control the weapon to quickly respond to situational changes.
--1 - Particularly hard to grip items, (especially those that are innately slipper or very rounded with no obvious gripping edge) such as basketballs and barrels, or which are dangerous to hold because of very sharp edges, like scrap metal and broken glass.
+-1 - Particularly hard to grip items, (especially those that are innately slipper or very rounded
+with no obvious gripping edge) such as basketballs and barrels, or which are dangerous to hold
+because of very sharp edges, like scrap metal and broken glass.
-+0 - Any object that doesn't fall into one of the categories below. Examples include 2x4s, computer monitors, wires, stingers and clothing. Basically, anything that has a grippable component, but which is too thick, too thin, or too flimsy to grab comfortably in a way that can reliably control the object.
++0 - Any object that doesn't fall into one of the categories below. Examples include 2x4s, computer
+monitors, wires, stingers and clothing. Basically, anything that has a grippable component, but
+which is too thick, too thin, or too flimsy to grab comfortably in a way that can reliably control
+the object.
+1 - A weapon with a fairly solid grip, like a pipe, a rock, guitar neck, pool cue or a heavy stick.
-+2 - A weapon with a dedicated grip shaped to the hand, like a sword, axe, knife, or police baton, or that is strapped to the body (or is a piece of the body). Fists would get a +2 bonus here, bringing them to "0" total, since none of the others would apply.
++2 - A weapon with a dedicated grip shaped to the hand, like a sword, axe, knife, or police baton,
+or that is strapped to the body (or is a piece of the body). Fists would get a +2 bonus here,
+bringing them to "0" total, since none of the others would apply.
### Length
-Length allows more surface area for potential contact, and reduces the need to control the positioning of the body to guarantee a hit. It also allows the player to strike from a safer distance, allowing them to worry more about trying to hit without being hit in return, and allows for swings with larger arcs, making dodging such a strike more difficult.
+
+Length allows more surface area for potential contact, and reduces the need to control the
+positioning of the body to guarantee a hit. It also allows the player to strike from a safer
+distance, allowing them to worry more about trying to hit without being hit in return, and allows
+for swings with larger arcs, making dodging such a strike more difficult.
+0 - Any object without a length bonus.
-+1 - Objects that, when held, extend over a foot (1/3 of a meter) in length from the hand. A normal American 12inch ruler is the handy boundary guide for when an item should switch over to a +1 bonus (the ruler, losing several inches when held, does not get one - unless you added a handle to it!).
++1 - Objects that, when held, extend over a foot (1/3 of a meter) in length from the hand. A normal
+American 12inch ruler is the handy boundary guide for when an item should switch over to a +1 bonus
+(the ruler, losing several inches when held, does not get one - unless you added a handle to it!).
-+2 - An object that is over 3 feet in length from the point where it is held. Includes swords, spears, quarterstaffs, poles, and a lot of other stuff.
++2 - An object that is over 3 feet in length from the point where it is held. Includes swords,
+spears, quarterstaffs, poles, and a lot of other stuff.
### Striking Surface
-Some weapons need to strike in a certain way to be effective. Others are more difficult to use "incorrectly".
--2 - Single-Point weapons - Picks, spears, syringes. Any weapon that has a single point that must contact the enemy in a specific way in order to deal a decent amount of damage. Also, weapons with difficult attack angles, like scythes, where the damaging part of the weapon is faced away from the enemy.
+Some weapons need to strike in a certain way to be effective. Others are more difficult to use
+"incorrectly".
--1 - Line of damage weapons - Swords, knives, and other weapons that require a solid strike along a particular piece of the weapon, where the weapon can be said to have an attack angle, fall here. Weapons that have point attacks but are still effective without any solid hit, such as a nailboard, would also fall here.
+-2 - Single-Point weapons - Picks, spears, syringes. Any weapon that has a single point that must
+contact the enemy in a specific way in order to deal a decent amount of damage. Also, weapons with
+difficult attack angles, like scythes, where the damaging part of the weapon is faced away from the
+enemy.
-+0 - attack-anywhere weapons - Clubs, pipes, maces, etc, where the weapon will be dealing full damage with a solid blow no matter how it is angled, because every surface is a striking surface.
+-1 - Line of damage weapons - Swords, knives, and other weapons that require a solid strike along a
+particular piece of the weapon, where the weapon can be said to have an attack angle, fall here.
+Weapons that have point attacks but are still effective without any solid hit, such as a nailboard,
+would also fall here.
-+1 - Weapons that can still do significant damage even with glancing blows would fall here. Jagged tearing weapons and electric weapons like a stun baton would fall here.
++0 - attack-anywhere weapons - Clubs, pipes, maces, etc, where the weapon will be dealing full
+damage with a solid blow no matter how it is angled, because every surface is a striking surface.
+
++1 - Weapons that can still do significant damage even with glancing blows would fall here. Jagged
+tearing weapons and electric weapons like a stun baton would fall here.
### Balance
-A measure of how well-suited the item is for being swung/thrust/etc. This factors in overall balance of the weapon, weight is accounted for separately.
--2 - Very clumsy or lopsided items ill-suited for swinging or thrusting. Characterized by requiring effort just to hold steady. frying pan or pot, chainsaw, chair, vacuum cleaner.
+A measure of how well-suited the item is for being swung/thrust/etc. This factors in overall balance
+of the weapon, weight is accounted for separately.
+
+-2 - Very clumsy or lopsided items ill-suited for swinging or thrusting. Characterized by requiring
+effort just to hold steady. frying pan or pot, chainsaw, chair, vacuum cleaner.
--1 - Balance of the object is uneven, but in a way that at least doesn't interfere with swinging. axes, sledgehammer, rifle, scythe, most polearms.
+-1 - Balance of the object is uneven, but in a way that at least doesn't interfere with swinging.
+axes, sledgehammer, rifle, scythe, most polearms.
-+0 - Neutral balance, neither well nor poorly weighted for the typical use. Heavy stick, rock, pool stick, kitchen knives, claw hammer, metal pipe, crowbar, handguns.
++0 - Neutral balance, neither well nor poorly weighted for the typical use. Heavy stick, rock, pool
+stick, kitchen knives, claw hammer, metal pipe, crowbar, handguns.
-+1 - Well-balanced for swinging or stabbing. Baseball bat, golf club, swords, quarterstaff, knives.
++1 - Well-balanced for swinging or stabbing. Baseball bat, golf club, swords, quarterstaff, knives.
## Damage
-Weapon's relative strength is based on an approximate formula involving its damage, to-hit, techniques and few other factors.
+
+Weapon's relative strength is based on an approximate formula involving its damage, to-hit,
+techniques and few other factors.
### Damage per second
-A melee's weapon damage per second (dps) is calculated past armor against a sample group of monsters with a range of dodge and armor values: a soldier zombie (low dodge, high bash and cut armor), a survivor zombie (medium dodge, some bash and cut armor), and a smoker zombie (high dodge, no armor). This should correctly weigh accuracy, criticals, and damage without over valuing any of them.
-In code, this is calculated using the item::effective_dps() function, which takes a character and a monster. It calculates the relative accuracy of the character and weapon against the monster's defenses and determines the hit rate from a table lookup. It also determines the number of critical hits. Number of hits is hit rate * 10,000, and number of misses is 10,000 - number of hits.
+A melee's weapon damage per second (dps) is calculated past armor against a sample group of monsters
+with a range of dodge and armor values: a soldier zombie (low dodge, high bash and cut armor), a
+survivor zombie (medium dodge, some bash and cut armor), and a smoker zombie (high dodge, no armor).
+This should correctly weigh accuracy, criticals, and damage without over valuing any of them.
+
+In code, this is calculated using the item::effective_dps() function, which takes a character and a
+monster. It calculates the relative accuracy of the character and weapon against the monster's
+defenses and determines the hit rate from a table lookup. It also determines the number of critical
+hits. Number of hits is hit rate * 10,000, and number of misses is 10,000 - number of hits.
-For both critical and non-critical hits, average damage is calculated based on the weapon's stats and the user's skill. Monster armor absorbs the damage, and then the damage is multiplied by the number of hits: either critical hits for the critical hit case, or total hits - critical hits for the non critical hit case. If the weapon has the rapid strike technique, the total damage is halved, and then the average damage is recalculated, multiplied by 0.66, and absorbed by monster armor again to account for rapid strikes.
+For both critical and non-critical hits, average damage is calculated based on the weapon's stats
+and the user's skill. Monster armor absorbs the damage, and then the damage is multiplied by the
+number of hits: either critical hits for the critical hit case, or total hits - critical hits for
+the non critical hit case. If the weapon has the rapid strike technique, the total damage is halved,
+and then the average damage is recalculated, multiplied by 0.66, and absorbed by monster armor again
+to account for rapid strikes.
-Number of moves is calculated as attack speed * ( number of misses + number of non-critical hits + number of critical hits ) for weapons without rapid strike, or attack speed * ( number of misses + number of non-critical hits / 2 + number of critical hits / 2 ) + attack speed / 2 * ( number of non-critical hits / 2 + number of critical hits / 2 ) for weapons without rapid strikes.
+Number of moves is calculated as attack speed * ( number of misses + number of non-critical hits +
+number of critical hits ) for weapons without rapid strike, or attack speed * ( number of misses +
+number of non-critical hits / 2 + number of critical hits / 2 ) + attack speed / 2 * ( number of
+non-critical hits / 2 + number of critical hits / 2 ) for weapons without rapid strikes.
-Damage per second against a particular monster is total damage * 100 / number of moves (100 for the 100 moves/second). Overall dps is the average of the dps against the three reference monsters.
+Damage per second against a particular monster is total damage * 100 / number of moves (100 for the
+100 moves/second). Overall dps is the average of the dps against the three reference monsters.
### Critical hits
-A double critical can occcur when a second hit roll is made against 1.5 * the monster's dodge. Double critical hits have a higher chance of occurring than normal critical hits. For each hit, the chance of achieving either a double critical hit or a normal critical hit is calculated, and then if a random number is less than the critical chance, the critical occurs. Both double and normal critical hits have the same effect, but the chance of them occurring is different.
-**Note** The critical hit system is stupid and complicated and produces weird results. Double critical hits should have a chance of occuring when the original hit roll is more than 1 standard deviation above the mean, which is simple and faster to calculate than the current system.
+A double critical can occcur when a second hit roll is made against 1.5 * the monster's dodge.
+Double critical hits have a higher chance of occurring than normal critical hits. For each hit, the
+chance of achieving either a double critical hit or a normal critical hit is calculated, and then if
+a random number is less than the critical chance, the critical occurs. Both double and normal
+critical hits have the same effect, but the chance of them occurring is different.
+
+**Note** The critical hit system is stupid and complicated and produces weird results. Double
+critical hits should have a chance of occuring when the original hit roll is more than 1 standard
+deviation above the mean, which is simple and faster to calculate than the current system.
### Other factors
+
Reach is worth +20% at reach 2, +35% at reach 3.
A weapon that is usuable by a known martial art is worth +50%.
### Weapon tiers
+
Relative value should put the weapon into one of those categories:
-<2 - Not weapons. Those items may be pressed into service, but are unlikely to be better than fists. Plastic bottles, rocks, boots.
+<2 - Not weapons. Those items may be pressed into service, but are unlikely to be better than fists.
+Plastic bottles, rocks, boots.
2-5 - Tools not meant to strike and improvised weapons. Two-by-fours, pointy sticks, pipes, hammers.
-6-11 - Dangerous tools or crude dedicated weapons. Golf clubs, two-by-swords, wooden spears, knife spears, hatchets, switchblades, tonfas, quarterstaves.
+6-11 - Dangerous tools or crude dedicated weapons. Golf clubs, two-by-swords, wooden spears, knife
+spears, hatchets, switchblades, tonfas, quarterstaves.
-12-15 - Good dedicated weapons or the most dangerous of tools. Wood and fire axes, steel spears, electric carvers, kukris, bokken, machetes, barbed wire bats.
+12-15 - Good dedicated weapons or the most dangerous of tools. Wood and fire axes, steel spears,
+electric carvers, kukris, bokken, machetes, barbed wire bats.
-20-35 - Weapons of war, well designed to kill humans. Wakizashis, katanas, broadswords, zweihanders, combat knifes, battle axes, war hammers, maces, morningstars.
+20-35 - Weapons of war, well designed to kill humans. Wakizashis, katanas, broadswords, zweihanders,
+combat knifes, battle axes, war hammers, maces, morningstars.
35+ - Sci-fi stuff. Diamond katanas, monomolecular blades, lightsabers and chainswords.
-Specific weapon balancing points:
-20 - combat knifes
-22 - short blades
-24 - long blades, short axes, and short flails
-26 - two handed blades, long axes, most spears
-28 - two handed axes and polearms
+Specific weapon balancing points: 20 - combat knifes 22 - short blades 24 - long blades, short axes,
+and short flails 26 - two handed blades, long axes, most spears 28 - two handed axes and polearms
30 - combat spears
Improvised weapons generally have about 75% of the value of a real weapon.
## Other melee balancing factors
+
### Attack speed
-Out of two weapons with same dpt, the faster one is generally better.
-Faster weapons allow more damage granularity (less overkill), make it less likely to miss a turn (and thus dodge/block recharges) and make positioning easier.
-Slower weapons will pierce armor better, but currently most enemies are very lightly armored.
+
+Out of two weapons with same dpt, the faster one is generally better. Faster weapons allow more
+damage granularity (less overkill), make it less likely to miss a turn (and thus dodge/block
+recharges) and make positioning easier. Slower weapons will pierce armor better, but currently most
+enemies are very lightly armored.
### Damage type
-At low skill, piercing damage suffers from scaling and bashing damage from damage limit due to low strength and skill. Cutting damage is not affected.
-At high skill, bashing damage is generally the strongest, but still suffers from the damage limit.
-Exotic damage types (currently only fire) do not scale with skills or crits.
+
+At low skill, piercing damage suffers from scaling and bashing damage from damage limit due to low
+strength and skill. Cutting damage is not affected. At high skill, bashing damage is generally the
+strongest, but still suffers from the damage limit. Exotic damage types (currently only fire) do not
+scale with skills or crits.
# RANGE WEAPONS
+
## Automatic Fire
-Guns with automatic fire are balanced around 1-second of cyclic fire, unless the cyclic or practical fire rate is less than 1 every second. Rates of fire less than 1 shot every second are increased to 2.
+
+Guns with automatic fire are balanced around 1-second of cyclic fire, unless the cyclic or practical
+fire rate is less than 1 every second. Rates of fire less than 1 shot every second are increased
+to 2.
## Magazines
+
### Reload times
-The overall balance is that magazines themselves are slow to reload whereas changing a magazine should be fast. For standard box magazines a default `reload_time` of 100 (per round) is appropriate with this value increasing for poor quality or extended magazines. Guns themselves should also specify `reload` of 100 (per magazine) unless their magazines are particularly awkward to reload (eg. ammo belts). The game logic intrinsically handles higher volume magazines consuming more time to attach to a gun so you need not consider this.
+
+The overall balance is that magazines themselves are slow to reload whereas changing a magazine
+should be fast. For standard box magazines a default `reload_time` of 100 (per round) is appropriate
+with this value increasing for poor quality or extended magazines. Guns themselves should also
+specify `reload` of 100 (per magazine) unless their magazines are particularly awkward to reload
+(eg. ammo belts). The game logic intrinsically handles higher volume magazines consuming more time
+to attach to a gun so you need not consider this.
### Weight
-Increases proportional to capacity and should have a comparable ratio to similar magazines. Consider the base item to be a 10-round .223 factory specification box magazine which has a capacity:weight of 1:10. Increase the ratio markedly for poor quality magazines or more slightly for extended magazines. Smaller calibers should use a lower ratio. The `material` should have some effect, with plastic magazines weighing less.
+
+Increases proportional to capacity and should have a comparable ratio to similar magazines. Consider
+the base item to be a 10-round .223 factory specification box magazine which has a capacity:weight
+of 1:10. Increase the ratio markedly for poor quality magazines or more slightly for extended
+magazines. Smaller calibers should use a lower ratio. The `material` should have some effect, with
+plastic magazines weighing less.
### Volume
-Scaled based upon the capacity relative to the `stack_size` of the ammo. For example 223 has a `stack size` of 20 so for 10 and 30 round magazines the volume would be 1 and 2. Extended magazine should always have larger volume than the standard type and for very large drum magazines consider applying an extra penalty. By default most handgun magazines should be volume 1 and most rifle magazines volume 2. Ammo belts should not specify volume as this will be determined from their length.
+
+Scaled based upon the capacity relative to the `stack_size` of the ammo. For example 223 has a
+`stack size` of 20 so for 10 and 30 round magazines the volume would be 1 and 2. Extended magazine
+should always have larger volume than the standard type and for very large drum magazines consider
+applying an extra penalty. By default most handgun magazines should be volume 1 and most rifle
+magazines volume 2. Ammo belts should not specify volume as this will be determined from their
+length.
### Reliability
-Should be specified first considering the below and then scaled against any equivalent magazines. For example if an extended version of a magazine exists place it one rank below the standard capacity version. Damaged guns or magazines will further adversely affect reliability.
-10 - **Perfectly reliable**. Factory specification or milspec only. Never extended magazines. Very rare.
+Should be specified first considering the below and then scaled against any equivalent magazines.
+For example if an extended version of a magazine exists place it one rank below the standard
+capacity version. Damaged guns or magazines will further adversely affect reliability.
-9 - **Reliable**. Failures only in burst fire. Factory or milspec magazines only. Never extended magazines. Uncommon.
+10 - **Perfectly reliable**. Factory specification or milspec only. Never extended magazines. Very
+rare.
-8 - **Dependable**. Failures infrequently in any fire mode. Highest reliability possible for extended magazines and those crafted using gunsmithing tools. Most common.
+9 - **Reliable**. Failures only in burst fire. Factory or milspec magazines only. Never extended
+magazines. Uncommon.
-7 - **Serviceable**. Fail infrequently in semi-automatic, more frequently in burst. Includes many extended and aftermarket gunsmithing tools. Common.
+8 - **Dependable**. Failures infrequently in any fire mode. Highest reliability possible for
+extended magazines and those crafted using gunsmithing tools. Most common.
-6 - **Acceptable**. Failures can be problematic. Highest reliability possible for magazines crafted **without** gunsmithing tools. Includes most ammo belts.
+7 - **Serviceable**. Fail infrequently in semi-automatic, more frequently in burst. Includes many
+extended and aftermarket gunsmithing tools. Common.
-5 - **Usable**. Failures can be problematic and more serious. Mostly poor quality hand-crafted magazines.
+6 - **Acceptable**. Failures can be problematic. Highest reliability possible for magazines crafted
+**without** gunsmithing tools. Includes most ammo belts.
-<4 - **Poor**. Significant risk of catastrophic failure. Not applied by default to any item but can be acquired by damage or other factors.
+5 - **Usable**. Failures can be problematic and more serious. Mostly poor quality hand-crafted
+magazines.
+
+<4 - **Poor**. Significant risk of catastrophic failure. Not applied by default to any item but can
+be acquired by damage or other factors.
### Rarity
-Overall balance is that pistol magazines are twice as common as rifle magazines and that for guns that spawn with magazines these are always the standard capacity versions. Consider 9x19mm and .223 to be the defaults with everything else more rare. Some locations have more specific balance requirements:
-
-Location | Description | With guns | Damaged | Example
-------------------|-----------------------------------------------------------|-----------|-----------|--------------------------
-Military site | Only source of milspec magazines and ammo belts | Never | Never | LW-56, .223 ammo belt
-Gun store | Standard and extended capacity magazines | Never | Never | STANAG-30, Glock extended
-Police armory | Mostly pistol magazines, especially 9x19mm, never extended| Sometimes | Never | Glock, MP5 magazine
-SWAT truck | Police or military magazines, occasionally extended | Sometimes | Rarely | MP5 extended
-Survivor basement | Anything except milspec weighted towards common types | Often | Sometimes | Saiga mag, M1911 extended
-Military surplus | Older military magazines that are not current issue | Never | Rarely | M9 mag, STEN magazine
-Pawn shop | Anything except milspec weighted towards unusual calibers | Never | Rarely | Makarov mag, AK-74 mag
-Everywhere else | Predominately 9mm and 223. Always with standard magazine | Often | Sometimes | Ruger 223 mag, M1911 mag
+
+Overall balance is that pistol magazines are twice as common as rifle magazines and that for guns
+that spawn with magazines these are always the standard capacity versions. Consider 9x19mm and .223
+to be the defaults with everything else more rare. Some locations have more specific balance
+requirements:
+
+| Location | Description | With guns | Damaged | Example |
+| ----------------- | ---------------------------------------------------------- | --------- | --------- | ------------------------- |
+| Military site | Only source of milspec magazines and ammo belts | Never | Never | LW-56, .223 ammo belt |
+| Gun store | Standard and extended capacity magazines | Never | Never | STANAG-30, Glock extended |
+| Police armory | Mostly pistol magazines, especially 9x19mm, never extended | Sometimes | Never | Glock, MP5 magazine |
+| SWAT truck | Police or military magazines, occasionally extended | Sometimes | Rarely | MP5 extended |
+| Survivor basement | Anything except milspec weighted towards common types | Often | Sometimes | Saiga mag, M1911 extended |
+| Military surplus | Older military magazines that are not current issue | Never | Rarely | M9 mag, STEN magazine |
+| Pawn shop | Anything except milspec weighted towards unusual calibers | Never | Rarely | Makarov mag, AK-74 mag |
+| Everywhere else | Predominately 9mm and 223. Always with standard magazine | Often | Sometimes | Ruger 223 mag, M1911 mag |
## Archery damage
-Bow damage is based on the momentum achieved in the projectile. Since arrows and bolts have sharp cutting surfaces, the penetration and therefore damage achieved is based on the projectile's capacity for slicing through tissues. The arrow has a modifier based on construction, material and design, most critically centered around the effectiveness of the head. Base damage is calculated from momentum by taking momentum in Slug-foot-seconds, multiplying by 150 and subtracting 32. This was arrived at by taking well-regarded bowhunting guidelines and determining the damage numbers necessary for a kill of various game on a critical hit, see tests/archery_damage_test.cpp for details.
+
+Bow damage is based on the momentum achieved in the projectile. Since arrows and bolts have sharp
+cutting surfaces, the penetration and therefore damage achieved is based on the projectile's
+capacity for slicing through tissues. The arrow has a modifier based on construction, material and
+design, most critically centered around the effectiveness of the head. Base damage is calculated
+from momentum by taking momentum in Slug-foot-seconds, multiplying by 150 and subtracting 32. This
+was arrived at by taking well-regarded bowhunting guidelines and determining the damage numbers
+necessary for a kill of various game on a critical hit, see tests/archery_damage_test.cpp for
+details.
## Ammo stats
+
### Base Damage
-The damage (**Dmg**) of firearm ammunition starts with the square root of a round's muzzle energy in joules (**Energy, J**) rounded to the nearest integer with an arbitrary increase or decrease to account for terminal ballistics. These numbers are modified depending on certain criteria (see Adjustment Criteria, below). Damage of handloaded ammo is set to 92% (rounded down) of their factory counterparts. A similar system for calculating recoil is planned but not currently being worked on. The figures used to calculate stats and any other relevant information are presented in table below.
+
+The damage (**Dmg**) of firearm ammunition starts with the square root of a round's muzzle energy in
+joules (**Energy, J**) rounded to the nearest integer with an arbitrary increase or decrease to
+account for terminal ballistics. These numbers are modified depending on certain criteria (see
+Adjustment Criteria, below). Damage of handloaded ammo is set to 92% (rounded down) of their factory
+counterparts. A similar system for calculating recoil is planned but not currently being worked on.
+The figures used to calculate stats and any other relevant information are presented in table below.
### Base Barrel Length
-Each cartridge also has a Base Barrel Length (**Base Brl**) listed; this determines the damage for the connected guns. A firearm has its damage modifier determined by it's real life barrel length; for every three inches between it and the listed baseline here, the gun takes a 1 point bonus or penalty, rounding to the nearest modifier. For example, a .45 ACP gun with a 7 inch barrel would get a +1 bonus (against a baseline of 5 inches).
-
-Ammo ID | Description | Energy, J | Dmg | Base Brl | Applied Modifiers / Comments |
--------------------|-----------------------------|-----------|-----|----------|-------------------------------
-.22 CB | 18gr CB bullet | 39 | 6 | 7.87in | |
-.22LR | 40gr unjacketed bullet | 141 | 12 | 6in | |
-.22LR FMJ | 30gr FMJ bullet | 277 | 17 | 6in | |
-.32 ACP | 60gr JHP bullet | 218 | 15 | 4in | |
-7.62x25mm | 85gr JHP bullet | 544 | 23 | 4.7in | |
-7.62x25mm Type P | 120gr bullet | 245 | 15 | 9.6in | Fired from the Type 64 SMG; need more data here |
-9x18mm 57-N-181S | 93gr FMJ bullet | 251 | 16 | 3.8in | |
-9x18mm SP-7 | 93gr bullet | 417 | 20 | 3.8in | |
-9x18mm RG028 | 93gr hardened steel core bullet | 317 | 18 | 3.8in | damage reduced by 4 |
-9x19mm FMJ | 115gr FMJ bullet | 420 | 24 | 5.9in | |
-9x19mm JHP | 115gr JHP bullet | 533 | 23 | 5.9in |damage increased by 3 |
-9x19mm +P | 115gr JHP bullet | 632 | 25 | 5.9in | |
-9x19mm +P+ | 115gr JHP bullet | 678 | 26 | 5.9in | |
-.38 Special | 130gr FMJ bullet | 256 | 16 | 4in | |
-.38 FMJ | 130gr FMJ bullet | 256 | 16 | 4in | |
-.38 Super | 147gr JHP bullet | 660 | 26 | 4in | |
-10mm Auto | 180gr FMJ bullet | 960 | 31 | 4in | |
-.40 S&W | 135gr JHP bullet | 575 | 24 | 4in | |
-.40 FMJ | 180gr FMJ bullet | 598 | 24 | 4in | |
-.44 Magnum | 240gr JHP bullet | 1570 | 40 | 7.5in | |
-.45 ACP JHP | 185gr JHP bullet | 614 | 25 | 5in | |
-.45 ACP FMJ | 230gr FMJ bullet | 447 | 21 | 5in | |
-.45 ACP +P | 200gr JHP bullet | 702 | 26 | 5in | |
-.454 Casull | 300gr JSP bullet | 2459 | 50 | 7.5in | |
-.45 Colt JHP | 250gr JHP bullet | 610 | 25 | 7.5in | |
-.500 S&W Magnum | 500gr bullet | 3056 | 55 | 8.4in | |
-4.6x30mm | 31gr copper plated steel bullet | 505 | 22 | 7.1in | damage reduced by 4 |
-5.7x28mm SS190 | 31gr AP FMJ bullet | 534 | 23 | 10.4in | damage reduced by 3 |
-7.62x39mm | 123gr FMJ bullet | 2179 | 46 | 16.3in | |
-7.62x39mm 57-N-231 | 121.9gr steel core FMJ bullet | 2036 | 45 | 16.3in | |
-7.62x39mm M67 | 123gr FMJ bullet | 2141 | 46 | 16.3in | TODO |
-5.45x39mm 7N10 | 56gr FMJ bullet | 1402 | 37 | 16.3in | damage increased by 3 |
-5.45x39mm 7N22 | 57gr steel core FMJ bullet | 1461 | 38 | 16.3in | |
-.223 Remington | 36gr JHP bullet | 1524 | 39 | 20in |Uses 5.56 NATO barrel baseline; damage increased by 5 |
-5.56x45mm M855A1 | 62gr copper core FMJBT bullet | 1843 | 43 | 20in | |
-.300BLK supersonic | 125gr OTM | 1840 | 43 | 16in |
-,300BLK subsonic | 220gr OTM | 675 | 26 | 16in | subsonic
-7.62x54mmR | 150gr FMJ bullet | 3629 | 60 | 28in | |
-.308 Winchester | 168gr hollow point bullet | 3570 | 60 | 24in | |
-7.62x51mm NATO M80 | 147gr FMJ bullet | 3304 | 57 | 24in | |
-7.62x51mm NATO M62 | 142gr tracer bullet | 3232 | 57 | 24in | Belt with 1/5 tracer rounds |
-.270 Winchester | 130gr soft point bullet | 3663 | 61 | 24in | |
-.30-06 Springfield | 165gr soft point bullet | 3894 | 62 | 24in | damage increased by 4 |
-.30-06 M2 | 165.7gr AP bullet | 3676 | 60 | 24in | damage reduced by 10 |
-.30-06 M14A1 | Incendiary ammunition | 3894 | 62 | 24in | damage reduced by 10 |
-.45-70 Govt. | 300gr soft point bullet | 3867 | 66 | 24in | damage increased by 4 |
-.300 Winchester Magnum | 220gr JHP bullet | 5299 | 73 | 24in | damage increased by 5 |
-.700 NX | 1000gr JSP bullet | 12100 | 110 | 28in | |
-.50 BMG Ball | 750gr FMJ-BT bullet | 17083 | 131 | 45in | |
-.50 BMG M33 Ball | 706.7gr bullet | 18013 | 134 | 45in | |
-.50 BMG M903 SLAP | 355gr tungsten AP bullet | 17083 | 131 | 45in | Can't be used with M107A1 |
-.410 000 shot | 5 000 pellets | 1530 | 39 | 18in | |
-
-###Adjustment Criteria
-If the resulting base damage is below specific thresholds, apply one of three multipliers. If the base damage is less than 20, the multiplier is 1.333. Else, if the base damage less than 30, the multiplier is 1.222. Else, if the base damage is less than 40, the multiplier is 1.111. Ammunition with damage of 40 or higher will generally have no arbitrary multiplier given to its basic variant.
-
-For liminal cases where the base damage is 20, 30, or 40 there is room for discretion regarding which balance increase to apply, if any. Discretion should be exercised based on which solution makes that ammunition distinct from other ammunition of similar power, as having two different ammotypes with identical power should be avoided.
-
-As for terminal ballistics, hollowpoints variant should be at least 25% more effective against a completely unarmored target, in order for the difference to be considered relevant. Conversely, the base FMJ variation should have a combined damage and armor penetration whose total is at least 15% greater than the damage of the hollowpoint variant. This recommendation will inform how much armor penetration the two variants should have. Armor-piercing variants have seven-eigths the damage of the standard FMJ variant, and enough penetration to make the combined value 30% higher than the damage of hollowpoint.
-
-The relative combined damage plus armor penetration for each variant can likewise be summarized as follows:
+
+Each cartridge also has a Base Barrel Length (**Base Brl**) listed; this determines the damage for
+the connected guns. A firearm has its damage modifier determined by it's real life barrel length;
+for every three inches between it and the listed baseline here, the gun takes a 1 point bonus or
+penalty, rounding to the nearest modifier. For example, a .45 ACP gun with a 7 inch barrel would get
+a +1 bonus (against a baseline of 5 inches).
+
+| Ammo ID | Description | Energy, J | Dmg | Base Brl | Applied Modifiers / Comments |
+| ---------------------- | ------------------------------- | --------- | --- | -------- | ----------------------------------------------------- |
+| .22 CB | 18gr CB bullet | 39 | 6 | 7.87in | |
+| .22LR | 40gr unjacketed bullet | 141 | 12 | 6in | |
+| .22LR FMJ | 30gr FMJ bullet | 277 | 17 | 6in | |
+| .32 ACP | 60gr JHP bullet | 218 | 15 | 4in | |
+| 7.62x25mm | 85gr JHP bullet | 544 | 23 | 4.7in | |
+| 7.62x25mm Type P | 120gr bullet | 245 | 15 | 9.6in | Fired from the Type 64 SMG; need more data here |
+| 9x18mm 57-N-181S | 93gr FMJ bullet | 251 | 16 | 3.8in | |
+| 9x18mm SP-7 | 93gr bullet | 417 | 20 | 3.8in | |
+| 9x18mm RG028 | 93gr hardened steel core bullet | 317 | 18 | 3.8in | damage reduced by 4 |
+| 9x19mm FMJ | 115gr FMJ bullet | 420 | 24 | 5.9in | |
+| 9x19mm JHP | 115gr JHP bullet | 533 | 23 | 5.9in | damage increased by 3 |
+| 9x19mm +P | 115gr JHP bullet | 632 | 25 | 5.9in | |
+| 9x19mm +P+ | 115gr JHP bullet | 678 | 26 | 5.9in | |
+| .38 Special | 130gr FMJ bullet | 256 | 16 | 4in | |
+| .38 FMJ | 130gr FMJ bullet | 256 | 16 | 4in | |
+| .38 Super | 147gr JHP bullet | 660 | 26 | 4in | |
+| 10mm Auto | 180gr FMJ bullet | 960 | 31 | 4in | |
+| .40 S&W | 135gr JHP bullet | 575 | 24 | 4in | |
+| .40 FMJ | 180gr FMJ bullet | 598 | 24 | 4in | |
+| .44 Magnum | 240gr JHP bullet | 1570 | 40 | 7.5in | |
+| .45 ACP JHP | 185gr JHP bullet | 614 | 25 | 5in | |
+| .45 ACP FMJ | 230gr FMJ bullet | 447 | 21 | 5in | |
+| .45 ACP +P | 200gr JHP bullet | 702 | 26 | 5in | |
+| .454 Casull | 300gr JSP bullet | 2459 | 50 | 7.5in | |
+| .45 Colt JHP | 250gr JHP bullet | 610 | 25 | 7.5in | |
+| .500 S&W Magnum | 500gr bullet | 3056 | 55 | 8.4in | |
+| 4.6x30mm | 31gr copper plated steel bullet | 505 | 22 | 7.1in | damage reduced by 4 |
+| 5.7x28mm SS190 | 31gr AP FMJ bullet | 534 | 23 | 10.4in | damage reduced by 3 |
+| 7.62x39mm | 123gr FMJ bullet | 2179 | 46 | 16.3in | |
+| 7.62x39mm 57-N-231 | 121.9gr steel core FMJ bullet | 2036 | 45 | 16.3in | |
+| 7.62x39mm M67 | 123gr FMJ bullet | 2141 | 46 | 16.3in | TODO |
+| 5.45x39mm 7N10 | 56gr FMJ bullet | 1402 | 37 | 16.3in | damage increased by 3 |
+| 5.45x39mm 7N22 | 57gr steel core FMJ bullet | 1461 | 38 | 16.3in | |
+| .223 Remington | 36gr JHP bullet | 1524 | 39 | 20in | Uses 5.56 NATO barrel baseline; damage increased by 5 |
+| 5.56x45mm M855A1 | 62gr copper core FMJBT bullet | 1843 | 43 | 20in | |
+| .300BLK supersonic | 125gr OTM | 1840 | 43 | 16in | |
+| ,300BLK subsonic | 220gr OTM | 675 | 26 | 16in | subsonic |
+| 7.62x54mmR | 150gr FMJ bullet | 3629 | 60 | 28in | |
+| .308 Winchester | 168gr hollow point bullet | 3570 | 60 | 24in | |
+| 7.62x51mm NATO M80 | 147gr FMJ bullet | 3304 | 57 | 24in | |
+| 7.62x51mm NATO M62 | 142gr tracer bullet | 3232 | 57 | 24in | Belt with 1/5 tracer rounds |
+| .270 Winchester | 130gr soft point bullet | 3663 | 61 | 24in | |
+| .30-06 Springfield | 165gr soft point bullet | 3894 | 62 | 24in | damage increased by 4 |
+| .30-06 M2 | 165.7gr AP bullet | 3676 | 60 | 24in | damage reduced by 10 |
+| .30-06 M14A1 | Incendiary ammunition | 3894 | 62 | 24in | damage reduced by 10 |
+| .45-70 Govt. | 300gr soft point bullet | 3867 | 66 | 24in | damage increased by 4 |
+| .300 Winchester Magnum | 220gr JHP bullet | 5299 | 73 | 24in | damage increased by 5 |
+| .700 NX | 1000gr JSP bullet | 12100 | 110 | 28in | |
+| .50 BMG Ball | 750gr FMJ-BT bullet | 17083 | 131 | 45in | |
+| .50 BMG M33 Ball | 706.7gr bullet | 18013 | 134 | 45in | |
+| .50 BMG M903 SLAP | 355gr tungsten AP bullet | 17083 | 131 | 45in | Can't be used with M107A1 |
+| .410 000 shot | 5 000 pellets | 1530 | 39 | 18in | |
+
+###Adjustment Criteria If the resulting base damage is below specific thresholds, apply one of three
+multipliers. If the base damage is less than 20, the multiplier is 1.333. Else, if the base damage
+less than 30, the multiplier is 1.222. Else, if the base damage is less than 40, the multiplier is
+1.111. Ammunition with damage of 40 or higher will generally have no arbitrary multiplier given to
+its basic variant.
+
+For liminal cases where the base damage is 20, 30, or 40 there is room for discretion regarding
+which balance increase to apply, if any. Discretion should be exercised based on which solution
+makes that ammunition distinct from other ammunition of similar power, as having two different
+ammotypes with identical power should be avoided.
+
+As for terminal ballistics, hollowpoints variant should be at least 25% more effective against a
+completely unarmored target, in order for the difference to be considered relevant. Conversely, the
+base FMJ variation should have a combined damage and armor penetration whose total is at least 15%
+greater than the damage of the hollowpoint variant. This recommendation will inform how much armor
+penetration the two variants should have. Armor-piercing variants have seven-eigths the damage of
+the standard FMJ variant, and enough penetration to make the combined value 30% higher than the
+damage of hollowpoint.
+
+The relative combined damage plus armor penetration for each variant can likewise be summarized as
+follows:
+
1. Hollowpoints are considered to have 100% combined damage (example: 100 damage, 0 arpen)
-2. Standard/FMJ variants are considered to have 115% combined damage, 80% damage and 35% arpen (example: 80 damage, 35 arpen)
-3. AP variants are considered to have 130% combined damage, 70% damage and 60% arpen (example: 70 damage, 60 arpen)
+2. Standard/FMJ variants are considered to have 115% combined damage, 80% damage and 35% arpen
+ (example: 80 damage, 35 arpen)
+3. AP variants are considered to have 130% combined damage, 70% damage and 60% arpen (example: 70
+ damage, 60 arpen)
# LIQUIDS:
-Multi-charge items are weighed by the charge/use. If you have an item that contains 40 uses, it'll weigh 40x as much (when found in-game) as you entered in the JSON. Liquids are priced by the 250mL unit, but handled in containers. This can cause problems if you create something that comes in (say) a gallon jug (15 charges) and price it at the cost of a jug's worth: it'll be 15x as expensive as intended.
-To that end, here's a list of containers with non-one volume. If you have something spawn in 'em, divide the "shelf" price by this value to arrive at the correct price to list in the JSON.
+Multi-charge items are weighed by the charge/use. If you have an item that contains 40 uses, it'll
+weigh 40x as much (when found in-game) as you entered in the JSON. Liquids are priced by the 250mL
+unit, but handled in containers. This can cause problems if you create something that comes in (say)
+a gallon jug (15 charges) and price it at the cost of a jug's worth: it'll be 15x as expensive as
+intended.
+
+To that end, here's a list of containers with non-one volume. If you have something spawn in 'em,
+divide the "shelf" price by this value to arrive at the correct price to list in the JSON.
- plastic bottle: 2
@@ -297,6 +450,6 @@ To that end, here's a list of containers with non-one volume. If you have somet
- gallon jug: 15
# Diamond weapons
-Diamond weapons should be uniform in their CVD machine requirements.
-Coal requirements are `floor((weapon_volume+1)/2)*25`.
-Hydrogen requirements are `coal_requirements/2.5`.
+
+Diamond weapons should be uniform in their CVD machine requirements. Coal requirements are
+`floor((weapon_volume+1)/2)*25`. Hydrogen requirements are `coal_requirements/2.5`.
diff --git a/doc/GUIDE_COMESTIBLES.md b/doc/GUIDE_COMESTIBLES.md
index 993763cfe5da..6465417c126c 100644
--- a/doc/GUIDE_COMESTIBLES.md
+++ b/doc/GUIDE_COMESTIBLES.md
@@ -1,44 +1,56 @@
# Guide to add Comestibles
-There are a large number of files in `json/items/comestibles`, and this guide will help you decide where to put your new comestible!
+There are a large number of files in `json/items/comestibles`, and this guide will help you decide
+where to put your new comestible!
## List of special comestibles:
-`med.json` -- comestible that has an addiction effect besides alcohol and caffeine, plus bandages and antiseptic type items.
+
+`med.json` -- comestible that has an addiction effect besides alcohol and caffeine, plus bandages
+and antiseptic type items.
`mre.json` -- items and comestibles related to MREs
`mutagen.json` -- comestible that has any mutagen effect
-`carnivore.json` -- item that would normally be butchered from an animal or monster, and their cooked versions. examples: chunks of meat, tainted bones, boiled stomach
+`carnivore.json` -- item that would normally be butchered from an animal or monster, and their
+cooked versions. examples: chunks of meat, tainted bones, boiled stomach
`protein.json` -- comestibles based on protein powder
-`spice.json` -- comestibles that normally do not have nutrients, but are used to flavor a dish. examples: salt, thyme, black pepper
+`spice.json` -- comestibles that normally do not have nutrients, but are used to flavor a dish.
+examples: salt, thyme, black pepper
`frozen.json` -- comestibles best eaten frozen
`brewing.json` -- items used in the brewing process
## List of normal comestibles, in order of priority:
-When you have a comestible you want to add to the files, just go down this list, and the first filename you see in the list that matches the criteria is your choice!
+
+When you have a comestible you want to add to the files, just go down this list, and the first
+filename you see in the list that matches the criteria is your choice!
### Liquid
`alcohol.json` -- "Drink" comestible with alcohol addiction
-`soup.json` -- "Drink" comestible which is a soup. This is more of a "food" than a drink - you primarily want the calories from this
+`soup.json` -- "Drink" comestible which is a soup. This is more of a "food" than a drink - you
+primarily want the calories from this
-`drink.json` -- "Drink" comestible. This is your drink of choice when you're thirsty! examples: tea, juice, water
+`drink.json` -- "Drink" comestible. This is your drink of choice when you're thirsty! examples: tea,
+juice, water
-`drink_other.json` -- "Drink" comestible that does not fit any other criteria. example: vinegar, mustard
+`drink_other.json` -- "Drink" comestible that does not fit any other criteria. example: vinegar,
+mustard
### Solid
`junkfood.json` -- comestible with the "junk" material. examples: cake, sugary cereal, nachos
-`sandwich.json` -- a "sandwich" is generally two slices of "bread" with something in between. examples: BLT, PB&J sandwich, fish sandwich
+`sandwich.json` -- a "sandwich" is generally two slices of "bread" with something in between.
+examples: BLT, PB&J sandwich, fish sandwich
-`offal_dishes.json` -- comestible made of various offals. the offal types are: liver, brain, kidney, sweetbread, stomach. This must be the primary part of the dish.
+`offal_dishes.json` -- comestible made of various offals. the offal types are: liver, brain, kidney,
+sweetbread, stomach. This must be the primary part of the dish.
`seed.json` -- seeds
@@ -68,4 +80,4 @@ When you have a comestible you want to add to the files, just go down this list,
`nuts.json` -- a nut or made of nuts
-`other.json` -- if you made it here, your comestible doesn't fit any other category!
\ No newline at end of file
+`other.json` -- if you made it here, your comestible doesn't fit any other category!
diff --git a/doc/IN_REPO_MODS.md b/doc/IN_REPO_MODS.md
index 04f8736f74e4..1a3738b5771e 100644
--- a/doc/IN_REPO_MODS.md
+++ b/doc/IN_REPO_MODS.md
@@ -1,59 +1,98 @@
# Getting Mods Into The Repository
-Cataclysm: Dark Days Ahead is not only moddable, but ships with a number of mods available for users to select from even without having to obtain mods from a third party source.
+Cataclysm: Dark Days Ahead is not only moddable, but ships with a number of mods available for users
+to select from even without having to obtain mods from a third party source.
## Why have in-repo mods?
The benefits of having mods in the repository include:
-* Visibility to a large number of users and potential contributors
-* The ability to use the CleverRaven issue tracker to report problems, as well as manage pull requests for adding to or fixing mods in the repository
-* Mods in the CleverRaven repository that demonstrate usage of a game feature which exists only for mods (i.e. not used by the core DDA game itself) help ensure that feature stays present in the game.
+- Visibility to a large number of users and potential contributors
+- The ability to use the CleverRaven issue tracker to report problems, as well as manage pull
+ requests for adding to or fixing mods in the repository
+- Mods in the CleverRaven repository that demonstrate usage of a game feature which exists only for
+ mods (i.e. not used by the core DDA game itself) help ensure that feature stays present in the
+ game.
There are however things that might be expected but are not guaranteed:
-* Guarantee of maintenance of content -- core devs are mostly focused on the main game itself; although some might pick up mods as a side task, people who continue working on the core game tend to like the feel of the core game.
-* Guarantee of maintenance of used features -- over the course of development it may become impossible to keep a feature, due to performance needs or maintenance burden.
-* Guarantee of sole authorship -- once in the repository, the mod is considered community content. While the developers might defer to an active curator of a mod, if they become inactive anyone else can step forward.
+- Guarantee of maintenance of content -- core devs are mostly focused on the main game itself;
+ although some might pick up mods as a side task, people who continue working on the core game tend
+ to like the feel of the core game.
+- Guarantee of maintenance of used features -- over the course of development it may become
+ impossible to keep a feature, due to performance needs or maintenance burden.
+- Guarantee of sole authorship -- once in the repository, the mod is considered community content.
+ While the developers might defer to an active curator of a mod, if they become inactive anyone
+ else can step forward.
## What kind of mods can be in the CleverRaven repository?
There are three primary categories of mods:
-* Content mods, which provide some kind of distinct experience from the core Dark Days Ahead game. This could be a change to lore, a change to a specific part of gameplay.
-* User Experience (UX) mods, which alter the look and feel of the game interface itself.
-* Development mods, which aren't "mods" in the typical sense but are instead there to ease the transition between "incomplete feature" and "complete feature", when a feature in the core game is sufficiently incomplete that the developers believe it needs to be optional to minimize disruption to players.
+- Content mods, which provide some kind of distinct experience from the core Dark Days Ahead game.
+ This could be a change to lore, a change to a specific part of gameplay.
+- User Experience (UX) mods, which alter the look and feel of the game interface itself.
+- Development mods, which aren't "mods" in the typical sense but are instead there to ease the
+ transition between "incomplete feature" and "complete feature", when a feature in the core game is
+ sufficiently incomplete that the developers believe it needs to be optional to minimize disruption
+ to players.
## What is necessary for a mod to be included in the repository?
-The most crucial criteria for a mod to be in the CleverRaven repository is that it has someone acting as a curator. This ensures that there is someone who is keeping an eye out for possible problems with the mod, and helping steer its development so that it continues to develop in accordance with its design purpose.
+The most crucial criteria for a mod to be in the CleverRaven repository is that it has someone
+acting as a curator. This ensures that there is someone who is keeping an eye out for possible
+problems with the mod, and helping steer its development so that it continues to develop in
+accordance with its design purpose.
Furthermore, there are additional criteria:
-* Content mods need to be providing some kind of curated experience. Either they introduce a set of locations, encounters, equipment, or progression mechanisms that do not fit the style of the core game, or they in some fashion modify the game according to some well-defined concept.
-* * Mods which do not qualify as curated experiences: Grab bag mods with no defined purpose (i.e. adding a bunch of random guns), "settings" mods that just turn off a possibly undesired but working feature (acid ants).
-* Development mods need to have an associated work-in-progress feature that they exist to mitigate.
+- Content mods need to be providing some kind of curated experience. Either they introduce a set of
+ locations, encounters, equipment, or progression mechanisms that do not fit the style of the core
+ game, or they in some fashion modify the game according to some well-defined concept.
+-
+ - Mods which do not qualify as curated experiences: Grab bag mods with no defined purpose (i.e.
+ adding a bunch of random guns), "settings" mods that just turn off a possibly undesired but
+ working feature (acid ants).
+- Development mods need to have an associated work-in-progress feature that they exist to mitigate.
## What are the responsibilities of a mod curator?
-A mod curator's role is to curate additions to mods and ensure they fit the concept of the mod. Curators are also responsible for at least acknowledging bug reports related to the mod: while curators don't have to personally manage every bugfix, if a bug is reported, the curator should be around to comment and help to find someone to fix it.
+A mod curator's role is to curate additions to mods and ensure they fit the concept of the mod.
+Curators are also responsible for at least acknowledging bug reports related to the mod: while
+curators don't have to personally manage every bugfix, if a bug is reported, the curator should be
+around to comment and help to find someone to fix it.
-It's important to understand that if a mod is in the repository, it is expected that people other than the curator are going to contribute to it, as with anything. On github, mod curators should Approve or Request Changes on PRs to the mod. If the curator has approved it, developers with merge permissions will treat that as a request to merge the changes.
+It's important to understand that if a mod is in the repository, it is expected that people other
+than the curator are going to contribute to it, as with anything. On github, mod curators should
+Approve or Request Changes on PRs to the mod. If the curator has approved it, developers with merge
+permissions will treat that as a request to merge the changes.
## When do mods get removed?
-Mods that have no curator or have proven to be consistently some type of maintenance burden are potentially subject to removal. Additionally, mods which had a purpose which is no longer relevant (development mods whose accompanying feature has been finished) will generally be removed once they're no longer deemed necessary. If a curator can't be reached for comment on a bugfix or review on a PR'd change for about a month, and hasn't left any indication of when they'll be back, developers will assume they've left the project, and the mod is orphaned.
+Mods that have no curator or have proven to be consistently some type of maintenance burden are
+potentially subject to removal. Additionally, mods which had a purpose which is no longer relevant
+(development mods whose accompanying feature has been finished) will generally be removed once
+they're no longer deemed necessary. If a curator can't be reached for comment on a bugfix or review
+on a PR'd change for about a month, and hasn't left any indication of when they'll be back,
+developers will assume they've left the project, and the mod is orphaned.
-Whenever a stable release occurs, mods which no longer qualify for inclusion will be marked 'obsolete'. This doesn't remove them from the repository, but it does prevent it from being used in new worlds (barring additional effort on the part of the player).
+Whenever a stable release occurs, mods which no longer qualify for inclusion will be marked
+'obsolete'. This doesn't remove them from the repository, but it does prevent it from being used in
+new worlds (barring additional effort on the part of the player).
## Why do mods get removed?
-While it may be counterintuitive, simply having a mod in the repository increases maintenance burden for the team, even for the people who don't use the mods itself.
+While it may be counterintuitive, simply having a mod in the repository increases maintenance burden
+for the team, even for the people who don't use the mods itself.
-Furthermore, mods which are shipped with the game but are not working correctly for whatever reason reflect poorly on the project on the whole. While it might seem 'easy' to fix a mod, the development team is mostly focused on the core game itself. This is why mod curators are necessary.
+Furthermore, mods which are shipped with the game but are not working correctly for whatever reason
+reflect poorly on the project on the whole. While it might seem 'easy' to fix a mod, the development
+team is mostly focused on the core game itself. This is why mod curators are necessary.
## How can mods be rescued from removal?
-If the mod otherwise meets inclusion criteria but lacks a curator (ie. has been declared orphaned), it's as simple as having someone else step forward as the new curator.
+If the mod otherwise meets inclusion criteria but lacks a curator (ie. has been declared orphaned),
+it's as simple as having someone else step forward as the new curator.
-Otherwise, it needs to either be made to meet the criteria, or it simply isn't going to be staying in the CleverRaven repository.
+Otherwise, it needs to either be made to meet the criteria, or it simply isn't going to be staying
+in the CleverRaven repository.
diff --git a/doc/ITEM_SPAWN.md b/doc/ITEM_SPAWN.md
index c50db7144fd5..a38cc53e337e 100644
--- a/doc/ITEM_SPAWN.md
+++ b/doc/ITEM_SPAWN.md
@@ -2,17 +2,23 @@
## Collection or Distribution
-In a Collection each entry is chosen independently from the other entries. Therefore the probability associated with each entry is absolute, in the range of 0...1. In the json files it is implemented as percentage with values from 0 to 100.
+In a Collection each entry is chosen independently from the other entries. Therefore the probability
+associated with each entry is absolute, in the range of 0...1. In the json files it is implemented
+as percentage with values from 0 to 100.
-A probability of 0 (or negative) means the entry is never chosen, a probability of 100% means its always chosen. The default is 100, because it's the most useful value. Default 0 would mean the entry can be removed anyway.
+A probability of 0 (or negative) means the entry is never chosen, a probability of 100% means its
+always chosen. The default is 100, because it's the most useful value. Default 0 would mean the
+entry can be removed anyway.
-A Distribution is a weighted list like the current system. Exactly one entry is chosen from it. The probability of each entry is relative to the probability of the other entries. A probability of 0 (or negative) means it is never chosen.
-
-An example: Suppose item A has a probability of 30 and item B has a probability of 20. Then the probabilities of the 4 combinations of A and B are:
+A Distribution is a weighted list like the current system. Exactly one entry is chosen from it. The
+probability of each entry is relative to the probability of the other entries. A probability of 0
+(or negative) means it is never chosen.
+An example: Suppose item A has a probability of 30 and item B has a probability of 20. Then the
+probabilities of the 4 combinations of A and B are:
| Combination | Collection | Distribution |
-| ----------------|------------|------------- |
+| --------------- | ---------- | ------------ |
| Neither A nor B | 56% | 0% |
| Only A | 24% | 60% |
| Only B | 14% | 40% |
@@ -33,7 +39,8 @@ The format is this:
}
```
-`subtype` is optional. It can be `collection` or `distribution`. If unspecified, it defaults to `old`, which denotes that this item group uses the old format (which is technically a distribution).
+`subtype` is optional. It can be `collection` or `distribution`. If unspecified, it defaults to
+`old`, which denotes that this item group uses the old format (which is technically a distribution).
There are [some caveats](#ammo-and-magazines) to watch out for when using `ammo` or `magazine`.
@@ -42,32 +49,38 @@ There are [some caveats](#ammo-and-magazines) to watch out for when using `ammo`
The `entries` list contains entries, each of which can be one of the following:
1. Item
- ```json
- { "item": "", ... }
- ```
+ ```json
+ { "item": "", ... }
+ ```
2. Group
- ``` json
- { "group": "", ... }
- ```
+ ```json
+ { "group": "", ... }
+ ```
3. Distribution
- ```json
- { "distribution": [
- "An array of entries, each of which can match any of these 4 formats"
- ] }
- ```
+ ```json
+ {
+ "distribution": [
+ "An array of entries, each of which can match any of these 4 formats"
+ ]
+ }
+ ```
4. Collection
- ```json
- { "collection": [
- "An array of entries, each of which can match any of these 4 formats"
- ] }
- ```
+ ```json
+ {
+ "collection": [
+ "An array of entries, each of which can match any of these 4 formats"
+ ]
+ }
+ ```
-The game decides based on the existence of either the `item` or the `group` value if the entry denotes a item or a reference to another item group.
+The game decides based on the existence of either the `item` or the `group` value if the entry
+denotes a item or a reference to another item group.
-Each entry can have more values (shown above as `...`). They allow further properties of the item(s):
+Each entry can have more values (shown above as `...`). They allow further properties of the
+item(s):
```json
"damage": |,
@@ -87,11 +100,14 @@ Each entry can have more values (shown above as `...`). They allow further prope
"container-group": "",
```
-`contents` is added as contents of the created item. It is not checked if they can be put into the item. This allows water, that contains a book, that contains a steel frame, that contains a corpse.
+`contents` is added as contents of the created item. It is not checked if they can be put into the
+item. This allows water, that contains a book, that contains a steel frame, that contains a corpse.
`count` makes the item spawn repeat to create more items, each time creating a new item.
-`charges`: Setting only min, not max will make the game calculate the max charges based on container or ammo/magazine capacity. Setting max too high will decrease it to maximum capacity. Not setting min will set it to 0 when max is set.
+`charges`: Setting only min, not max will make the game calculate the max charges based on container
+or ammo/magazine capacity. Setting max too high will decrease it to maximum capacity. Not setting
+min will set it to 0 when max is set.
```json
"damage-min": 0,
@@ -100,25 +116,36 @@ Each entry can have more values (shown above as `...`). They allow further prope
"charges": [10, 100]
```
-This will create 4 items, they can have different damage levels as the damage value is rolled separately for each of these items. Each item has charges (AKA ammo) in the range of 10 to 100 (inclusive); if the item needs a magazine before it can have charges, that will be taken care of for you. Using an array (which must have 2 entries) for charges/count/damage is equivalent to writing explicit min and max values. In other words `"count": [a,b]` is the same as `"count-min": a, "count-max": b`.
+This will create 4 items, they can have different damage levels as the damage value is rolled
+separately for each of these items. Each item has charges (AKA ammo) in the range of 10 to 100
+(inclusive); if the item needs a magazine before it can have charges, that will be taken care of for
+you. Using an array (which must have 2 entries) for charges/count/damage is equivalent to writing
+explicit min and max values. In other words `"count": [a,b]` is the same as
+`"count-min": a, "count-max": b`.
-The container is checked and the item is put inside the container, and the charges of the item are capped/increased to match the size of the container.
+The container is checked and the item is put inside the container, and the charges of the item are
+capped/increased to match the size of the container.
### Ammo and Magazines
-Here are some ways to make items spawn with/without ammo/magazines (note that `ammo-item` can
-be specified for guns and magazines in the entries array to use a non-default ammo type):
+Here are some ways to make items spawn with/without ammo/magazines (note that `ammo-item` can be
+specified for guns and magazines in the entries array to use a non-default ammo type):
-* Specify an ammo/magazine chance (introduced in Section 2) for the entire item group.
- `ammo` specifies the percent chance that the entries will spawn fully loaded (if it needs a magazine, it will be added for you).
- `magazine` specifies the percent chance that the entries will spawn with a magazine.
- Both of these default to 0 if unspecified.
+- Specify an ammo/magazine chance (introduced in Section 2) for the entire item group. `ammo`
+ specifies the percent chance that the entries will spawn fully loaded (if it needs a magazine, it
+ will be added for you). `magazine` specifies the percent chance that the entries will spawn with a
+ magazine. Both of these default to 0 if unspecified.
- Note that `ammo` and `magazine` only apply to tools, guns, and magazines. Furthermore, they don't apply to tools whose entry explicitly specifies how much ammo (charges) to spawn with, or to tools whose JSON item definition specifies a random amount or a fixed, nonzero amount of initial charges.
+ Note that `ammo` and `magazine` only apply to tools, guns, and magazines. Furthermore, they don't
+ apply to tools whose entry explicitly specifies how much ammo (charges) to spawn with, or to tools
+ whose JSON item definition specifies a random amount or a fixed, nonzero amount of initial
+ charges.
- If any item groups are referenced from your item group, then their ammo/magazine chances are ignored, and yours are used instead.
+ If any item groups are referenced from your item group, then their ammo/magazine chances are
+ ignored, and yours are used instead.
-* Use `charges`, `charges-min`, or `charges-max` in the entries array. A default magazine will be added for you if needed.
+- Use `charges`, `charges-min`, or `charges-max` in the entries array. A default magazine will be
+ added for you if needed.
## Shortcuts
@@ -134,7 +161,8 @@ means the same as:
"entries": [ { "item": "" }, { "item": "", "prob": 10 } ]
```
-In other words: a single string denotes an item id, an array (which must contain a string and a number) denotes an item id and a probability.
+In other words: a single string denotes an item id, an array (which must contain a string and a
+number) denotes an item id and a probability.
This is true for groups as well:
@@ -142,18 +170,21 @@ This is true for groups as well:
"groups": [ "", [ "", 10 ] ]
```
-This format does not support further properties of the created items - the probability is only optional for entries of collections!
+This format does not support further properties of the created items - the probability is only
+optional for entries of collections!
-The content of "entries", "items" and "groups" are all added if those members exist. This will have the item `` appear twice in the item group:
+The content of "entries", "items" and "groups" are all added if those members exist. This will have
+the item `` appear twice in the item group:
```json
{
- "items": [ "" ],
- "entries": [ { "item": "" } ]
+ "items": [""],
+ "entries": [{ "item": "" }]
}
```
-Another example. The group "milk" spawns a container (taken from milk_containers) that contains milk (maximal amount that fits into the container, because there is no specific charges value defined).
+Another example. The group "milk" spawns a container (taken from milk_containers) that contains milk
+(maximal amount that fits into the container, because there is no specific charges value defined).
```json
{
@@ -177,9 +208,15 @@ Another example. The group "milk" spawns a container (taken from milk_containers
## Inlined item groups
-In some places one can define an item group directly instead of giving the id of a group. One can not refer to that group as it has no visible id (it has an unspecific/random id internally). This is most useful when the group is very specific to the place it is used and wont ever appear anywhere else.
+In some places one can define an item group directly instead of giving the id of a group. One can
+not refer to that group as it has no visible id (it has an unspecific/random id internally). This is
+most useful when the group is very specific to the place it is used and wont ever appear anywhere
+else.
-As an example: monster death drops (`death_drops` entry in the `MONSTER` object, see [JSON_INFO.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/JSON_INFO.md)) can do this. If the monster is very specific (e.g. a special robot, a unique endgame monster), the item spawned upon its death won't (in that form) appear in any other group.
+As an example: monster death drops (`death_drops` entry in the `MONSTER` object, see
+[JSON_INFO.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/JSON_INFO.md)) can do
+this. If the monster is very specific (e.g. a special robot, a unique endgame monster), the item
+spawned upon its death won't (in that form) appear in any other group.
Therefore, this snippet:
@@ -199,26 +236,30 @@ is equivalent to:
```json
{
- "death_drops": {
- "subtype": "distribution",
- "items": [ "a", "b" ]
- }
+ "death_drops": {
+ "subtype": "distribution",
+ "items": ["a", "b"]
+ }
}
```
-The inline group is read like any other group and one can use all the properties mentioned above. Its `type` and its `id` members are always ignored.
+The inline group is read like any other group and one can use all the properties mentioned above.
+Its `type` and its `id` members are always ignored.
-Instead of a full JSON object, one can also write a JSON array. The default subtype is used and the array is read like the "entries" array (see above). Each entry of that array must be a JSON object. Example:
+Instead of a full JSON object, one can also write a JSON array. The default subtype is used and the
+array is read like the "entries" array (see above). Each entry of that array must be a JSON object.
+Example:
```json
{
- "death_drops": [
- { "item": "rag", "damage": 2 }, { "item": "bowling_ball" }
- ]
+ "death_drops": [
+ { "item": "rag", "damage": 2 },
+ { "item": "bowling_ball" }
+ ]
}
```
-----
+---
### Notes
@@ -227,20 +268,27 @@ Instead of a full JSON object, one can also write a JSON array. The default subt
You can test your item groups in the game:
1. Load a game and call the debug menu
- > *TIP* (If a key isn't bound to the debug menu or you forgot it, use ESC > 1)
+ > _TIP_ (If a key isn't bound to the debug menu or you forgot it, use ESC >
+ > 1)
2. Choose "Test Item Group".
3. Select the item group you want to debug.
- The game will then spawn items in that group 100 times and count the spawned items. They'll then be displayed, sorted by their frequency.
+ The game will then spawn items in that group 100 times and count the spawned items. They'll then
+ be displayed, sorted by their frequency.
- > *TIP*: You can filter anything in the debug menu using /
+ > _TIP_: You can filter anything in the debug menu using /
-You should not add items to the item group `EMPTY_GROUP`. This group can be used when the game requires a group id, but you don't want to spawn any items there. The group will never spawn items.
+You should not add items to the item group `EMPTY_GROUP`. This group can be used when the game
+requires a group id, but you don't want to spawn any items there. The group will never spawn items.
#### SUS
-When adding items to item groups, attempt to locate or create **SUS item groups.** SUS item groups are collections that contain a reasonable realistic distribution of items that might spawn in a given piece of storage furniture. SUS stands for "specific use storage." One of the aims of organizing item groups into specific use storage is to promote reusable tables that can be maintained and extended.
+When adding items to item groups, attempt to locate or create **SUS item groups.** SUS item groups
+are collections that contain a reasonable realistic distribution of items that might spawn in a
+given piece of storage furniture. SUS stands for "specific use storage." One of the aims of
+organizing item groups into specific use storage is to promote reusable tables that can be
+maintained and extended.
You can find SUS item groups at /data/json/itemgroups/SUS
diff --git a/doc/JSON Mapping Guides/Guide for beginning mapgen.md b/doc/JSON Mapping Guides/Guide for beginning mapgen.md
index 209302930c70..a461773c02e0 100644
--- a/doc/JSON Mapping Guides/Guide for beginning mapgen.md
+++ b/doc/JSON Mapping Guides/Guide for beginning mapgen.md
@@ -1,224 +1,302 @@
### Guide for basic mapgen
-This guide will cover the basics of mapgen, which files you need to edit, the tags in each file and the differences in creating specials or regular city buildings. For full technical information about mapgen entries refer to: [doc/MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/MAPGEN.md).
+This guide will cover the basics of mapgen, which files you need to edit, the tags in each file and
+the differences in creating specials or regular city buildings. For full technical information about
+mapgen entries refer to:
+[doc/MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/MAPGEN.md).
First, lets cover some basic concepts and the files you'll add or edit.
#### General comments:
-CDDA mapgen is surprisingly powerful once you get used to working with it. You can use lots of tricks to add variability and interest to your maps. Most advanced mapgen techniques will go into a different tutorial. This one covers basic concepts and how to create a basic single OMT (overmap terrain tile)sized building. We will touch on palette usage and how to add a roof as well.
+CDDA mapgen is surprisingly powerful once you get used to working with it. You can use lots of
+tricks to add variability and interest to your maps. Most advanced mapgen techniques will go into a
+different tutorial. This one covers basic concepts and how to create a basic single OMT (overmap
+terrain tile)sized building. We will touch on palette usage and how to add a roof as well.
#### Specials vs. city buildings:
-A special is a building that spawns outside the city and requires additional information to spawn, like its distance from cities and valid OMT terrain types. They also used to be the only multi-tile buildings in the game until recent changes allowed special type buildings to spawn inside cities. Examples of specials are: farms, cabins, LMOE, etc.
+A special is a building that spawns outside the city and requires additional information to spawn,
+like its distance from cities and valid OMT terrain types. They also used to be the only multi-tile
+buildings in the game until recent changes allowed special type buildings to spawn inside cities.
+Examples of specials are: farms, cabins, LMOE, etc.
-City buildings can be single or multi-tile in size and have their spawns limited to the city boundaries. A building can be both a city building and a special but would require both sets of entries to spawn for both types. Some motels are examples of this (see: 2fmotel_city & 2fmotel).
+City buildings can be single or multi-tile in size and have their spawns limited to the city
+boundaries. A building can be both a city building and a special but would require both sets of
+entries to spawn for both types. Some motels are examples of this (see: 2fmotel_city & 2fmotel).
-Important policy: since the roof project, all buildings are now multi-tile across z levels. All new buildings should always get a JSON roof added. Soon, all basements will also be custom fit to the ground floor mapgen, so it is good practice to include dedicated downstairs if you want a basement.
+Important policy: since the roof project, all buildings are now multi-tile across z levels. All new
+buildings should always get a JSON roof added. Soon, all basements will also be custom fit to the
+ground floor mapgen, so it is good practice to include dedicated downstairs if you want a basement.
#### The Files & their purpose:
-1. You will add a new mapgen file in: [data/json/mapgen](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/data/json/mapgen) or one of its sub-folders. If you are using an existing foundation shape for the building, you may append it to that building's file.
- * This is the blueprint for the building. It can also hold all the building’s data for adding furniture and loot (see palette for an alternative).
-
-2. You will add entries for each z level you create in the appropriate overmap_terrain file ([data/json/overmap/overmap_terrain](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/data/json/overmap/overmap_terrain)).
- * These entries will define what your building looks like in the overmap, its symbol, color, and spawn requirements like adding sidewalks, it will also control flags for some mapgen functions.
-
-3. You will add an entry into either specials.json or multitile_city_buildings.json depending on if it is a special or a city building.
- * For multitile_city_buildings this will link the various z levels &/or multiple OMTs of your building.
- * For specials, this will link the various z levels or multiple OMTs of your building, define any needed road/subway/etc. connections, and define its spawning parameters.
-
-4. Add an entry into regional_settings.json for city buildings. This will allow them to spawn in the world.
-
-5. Optional but recommended for any large project: adding a new palette file into mapgen_palettes folder (you may use any existing palette too).
- * This is a file that can be shared among several maps that holds a portion of the mapgen file data. It is commonly used for defining terrain and furniture. You can also put in loot, vehicle, monster spawns and any other data that normally goes in the `"object"` tag of the mapgen file.
- * Please avoid editing existing mapgen palettes because you may affect existing maps using a combination of the palette and the mapgen file.
+1. You will add a new mapgen file in:
+ [data/json/mapgen](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/data/json/mapgen) or
+ one of its sub-folders. If you are using an existing foundation shape for the building, you may
+ append it to that building's file.
+ - This is the blueprint for the building. It can also hold all the building’s data for adding
+ furniture and loot (see palette for an alternative).
+
+2. You will add entries for each z level you create in the appropriate overmap_terrain file
+ ([data/json/overmap/overmap_terrain](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/data/json/overmap/overmap_terrain)).
+ - These entries will define what your building looks like in the overmap, its symbol, color, and
+ spawn requirements like adding sidewalks, it will also control flags for some mapgen functions.
+
+3. You will add an entry into either specials.json or multitile_city_buildings.json depending on if
+ it is a special or a city building.
+ - For multitile_city_buildings this will link the various z levels &/or multiple OMTs of your
+ building.
+ - For specials, this will link the various z levels or multiple OMTs of your building, define any
+ needed road/subway/etc. connections, and define its spawning parameters.
+
+4. Add an entry into regional_settings.json for city buildings. This will allow them to spawn in the
+ world.
+
+5. Optional but recommended for any large project: adding a new palette file into mapgen_palettes
+ folder (you may use any existing palette too).
+ - This is a file that can be shared among several maps that holds a portion of the mapgen file
+ data. It is commonly used for defining terrain and furniture. You can also put in loot,
+ vehicle, monster spawns and any other data that normally goes in the `"object"` tag of the
+ mapgen file.
+ - Please avoid editing existing mapgen palettes because you may affect existing maps using a
+ combination of the palette and the mapgen file.
#### Starting the mapgen entry:
-When I start a new map project, I generally will add in all the entries I need for it to spawn in game from the outset. This way I can test it as I work on it and adjust it as needed. So, I recommend setting up all the needed files when you begin your project.
+When I start a new map project, I generally will add in all the entries I need for it to spawn in
+game from the outset. This way I can test it as I work on it and adjust it as needed. So, I
+recommend setting up all the needed files when you begin your project.
Before beginning you’ll want to make some decisions:
1. What size will it be overall (how many OMTs?)
2. Where will it spawn?
3. Will I use a palette or put everything in the mapgen file.
- * If you use a palette, define as much of it as possible from the outset.
+ - If you use a palette, define as much of it as possible from the outset.
4. Advanced questions:
- * Will I use nested maps?
- * Do I want it to connect to the subways or roads?
- * Will I be using the mapgen object data in combination with a palette (see the mall 2nd floor if you want a master class in combined usage of both types)?
+ - Will I use nested maps?
+ - Do I want it to connect to the subways or roads?
+ - Will I be using the mapgen object data in combination with a palette (see the mall 2nd floor if
+ you want a master class in combined usage of both types)?
#### The mapgen map:
-This covers the mapgen file map flags and what they do in layman’s terms. You can get more extensive information from [doc/MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/MAPGEN.md).
+This covers the mapgen file map flags and what they do in layman’s terms. You can get more extensive
+information from
+[doc/MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/MAPGEN.md).
-the mapgen file has some meta data tags and the `"object"` data which defines everything to make the map.
+the mapgen file has some meta data tags and the `"object"` data which defines everything to make the
+map.
##### The metadata:
Sample:
+
```
- "type": "mapgen",
- "method": "json",
- "om_terrain": "s_restaurant_coffee",
- "weight": 250,
+"type": "mapgen",
+"method": "json",
+"om_terrain": "s_restaurant_coffee",
+"weight": 250,
```
-1. `"type"` will always be mapgen (we'll cover other map types in future tutorials), the `"method"` will always be JSON. This data tells the program how to treat this file.
-2. `"om_terrain"`: this is basically your internal name for the map (not the name that shows up on the overmap). It should usually be unique unless you plan on having multiple variants of the same map which **share the same building foundation shape** (note: I mean the actual shape of the building's foundation).
+1. `"type"` will always be mapgen (we'll cover other map types in future tutorials), the `"method"`
+ will always be JSON. This data tells the program how to treat this file.
+
+2. `"om_terrain"`: this is basically your internal name for the map (not the name that shows up on
+ the overmap). It should usually be unique unless you plan on having multiple variants of the same
+ map which **share the same building foundation shape** (note: I mean the actual shape of the
+ building's foundation).
-3. `"weight"`: This entry is the weight of this particular map compared to others **sharing the same om_terrain name**. So, say you have one version of a house then you make an identical house with different spawns (like a fully furnished house and an abandoned version). This weight will determine how often each spawns in relation to the other. Say the furnished house is at 100, and the abandoned one is at 20. So, it'll spawn 5x less than the furnished house.
+3. `"weight"`: This entry is the weight of this particular map compared to others **sharing the same
+ om_terrain name**. So, say you have one version of a house then you make an identical house with
+ different spawns (like a fully furnished house and an abandoned version). This weight will
+ determine how often each spawns in relation to the other. Say the furnished house is at 100, and
+ the abandoned one is at 20. So, it'll spawn 5x less than the furnished house.
##### The object data:
-This is the section of tags that defines your map, its terrains, furniture, and various spawn types. There are several ways to place items (and nested maps). These deserve their own tutorial. For this document we'll be using "explicit symbol" placement for loot spawns, the easiest to use. Everything in the object section can be placed in a mapgen_palette except rows and fill_ter.
+This is the section of tags that defines your map, its terrains, furniture, and various spawn types.
+There are several ways to place items (and nested maps). These deserve their own tutorial. For this
+document we'll be using "explicit symbol" placement for loot spawns, the easiest to use. Everything
+in the object section can be placed in a mapgen_palette except rows and fill_ter.
-Sample object segment: everything in the object needs to fall within the object's {} braces or it won't be included. If you misplace the end bracket, you probably won't get a loading error.
+Sample object segment: everything in the object needs to fall within the object's {} braces or it
+won't be included. If you misplace the end bracket, you probably won't get a loading error.
Sample:
+
```
- "object": {
- "fill_ter": "t_floor",
- "rows": [
- "S___________SSTzzzzzzzTS",
- "S_____,_____SSzMbMbMbMzS",
- "S_____,_____SSSSSSSS/MzS",
- "S_____,_____SSSSSSSS/MzS",
- "S_____,_____SSzzzSSS/MzS",
- "S_____,_____SS||V{{{V||S",
- "S_____,_____SS|D <|S",
- "SSSSSSSSSSSSSS|r OS",
- "SSSSSSSSSSSSSS|r |S",
- "SVVVVVVVz./Mzz| #W##|S",
- "SVD>>>>Vz./bMzV #ww%|S",
- "SV BBB>Vz.b/..{ xwwF|S",
- "SV B>Vz.Mb..{ flwl|S",
- "SV B>Vz....zV flwU|S",
- "SV B>Vzbbbzz|X #wwG|S",
- "S|B ^||||||||||^ ||I||S",
- "SO6 B|=;|;=|99 r|FwC|S",
- "S|B 6|=A|A=|9 r|Fwc|S",
- "S| B|+|||+|9 D|!ww|S",
- "S|B |Lwl|S",
- "SO6 6 6 6 B66B|Lwl|S",
- "S|B ^???????^ B66B|Lwl|S",
- "S|||||||||||||||||||3||S",
- "S4SSSSSSSSSSSSSSSSSSSSSS"
- ],
- "terrain": {
- " ": "t_floor",
- "!": "t_linoleum_white",
- "#": "t_linoleum_white",
- "%": "t_linoleum_white",
- "+": "t_door_c",
- ",": "t_pavement_y",
- ".": "t_grass",
- "/": "t_dirt",
- "3": [ "t_door_locked", "t_door_locked_alarm" ],
- ";": "t_linoleum_white",
- "=": "t_linoleum_white",
- "A": "t_linoleum_white",
- "C": "t_linoleum_white",
- "F": "t_linoleum_white",
- "G": "t_linoleum_white",
- "I": "t_door_locked_interior",
- "L": "t_linoleum_white",
- "M": "t_dirt",
- "O": "t_window",
- "S": "t_sidewalk",
- "U": "t_linoleum_white",
- "V": "t_wall_glass",
- "W": "t_fencegate_c",
- "_": "t_pavement",
- "b": "t_dirt",
- "c": "t_linoleum_white",
- "f": "t_linoleum_white",
- "l": "t_linoleum_white",
- "w": "t_linoleum_white",
- "x": "t_console_broken",
- "z": "t_shrub",
- "{": "t_door_glass_c",
- "|": "t_wall_b",
- "<": "t_stairs_up",
- "4": "t_gutter_downspout",
- "T": "t_tree_coffee"
- },
- "furniture": {
- "#": "f_counter",
- "%": "f_trashcan",
- "/": "f_bluebell",
- "6": "f_table",
- "9": "f_rack",
- ">": "f_counter",
- "?": "f_sofa",
- "A": "f_sink",
- "B": "f_chair",
- "C": "f_desk",
- "D": "f_trashcan",
- "F": "f_fridge",
- "G": "f_oven",
- "L": "f_locker",
- "M": "f_dahlia",
- "U": "f_sink",
- "X": "f_rack",
- "^": "f_indoor_plant",
- "b": "f_dandelion",
- "f": "f_glass_fridge",
- "l": "f_rack",
- "r": "f_rack"
- },
- "toilets": { ";": { } },
- "items": {
- "#": { "item": "coffee_counter", "chance": 10 },
- "6": { "item": "coffee_table", "chance": 35 },
- "9": { "item": "coffee_display_2", "chance": 85, "repeat": [ 1, 8 ] },
- ";": { "item": "coffee_bathroom", "chance": 15 },
- "=": { "item": "coffee_bathroom", "chance": 35 },
- ">": { "item": "coffee_table", "chance": 25 },
- "A": { "item": "coffee_bathroom", "chance": 35 },
- "C": { "item": "office", "chance": 70 },
- "D": { "item": "coffee_trash", "chance": 75 },
- "F": { "item": "coffee_fridge", "chance": 80, "repeat": [ 1, 8 ] },
- "G": { "item": "oven", "chance": 35 },
- "L": { "item": "coffee_locker", "chance": 75 },
- "U": { "item": "coffee_dishes", "chance": 75 },
- "X": { "item": "coffee_newsstand", "chance": 90, "repeat": [ 1, 8 ] },
- "f": { "item": "coffee_freezer", "chance": 85, "repeat": [ 1, 8 ] },
- "l": { "item": "coffee_prep", "chance": 50 },
- "r": { "item": "coffee_condiments", "chance": 80, "repeat": [ 1, 8 ] }
- },
- "monsters": { "!": { "monster": "GROUP_COFFEE_SHOP_ZOMBIE", "chance": 1 } },
- "place_monsters": [ { "monster": "GROUP_ZOMBIE", "x": [ 3, 17 ], "y": [ 13, 18 ], "chance": 1 } ],
- "vehicles": { "c": { "vehicle": "swivel_chair", "chance": 100, "status": 1 } }
- }
+"object": {
+ "fill_ter": "t_floor",
+ "rows": [
+ "S___________SSTzzzzzzzTS",
+ "S_____,_____SSzMbMbMbMzS",
+ "S_____,_____SSSSSSSS/MzS",
+ "S_____,_____SSSSSSSS/MzS",
+ "S_____,_____SSzzzSSS/MzS",
+ "S_____,_____SS||V{{{V||S",
+ "S_____,_____SS|D <|S",
+ "SSSSSSSSSSSSSS|r OS",
+ "SSSSSSSSSSSSSS|r |S",
+ "SVVVVVVVz./Mzz| #W##|S",
+ "SVD>>>>Vz./bMzV #ww%|S",
+ "SV BBB>Vz.b/..{ xwwF|S",
+ "SV B>Vz.Mb..{ flwl|S",
+ "SV B>Vz....zV flwU|S",
+ "SV B>Vzbbbzz|X #wwG|S",
+ "S|B ^||||||||||^ ||I||S",
+ "SO6 B|=;|;=|99 r|FwC|S",
+ "S|B 6|=A|A=|9 r|Fwc|S",
+ "S| B|+|||+|9 D|!ww|S",
+ "S|B |Lwl|S",
+ "SO6 6 6 6 B66B|Lwl|S",
+ "S|B ^???????^ B66B|Lwl|S",
+ "S|||||||||||||||||||3||S",
+ "S4SSSSSSSSSSSSSSSSSSSSSS"
+ ],
+ "terrain": {
+ " ": "t_floor",
+ "!": "t_linoleum_white",
+ "#": "t_linoleum_white",
+ "%": "t_linoleum_white",
+ "+": "t_door_c",
+ ",": "t_pavement_y",
+ ".": "t_grass",
+ "/": "t_dirt",
+ "3": [ "t_door_locked", "t_door_locked_alarm" ],
+ ";": "t_linoleum_white",
+ "=": "t_linoleum_white",
+ "A": "t_linoleum_white",
+ "C": "t_linoleum_white",
+ "F": "t_linoleum_white",
+ "G": "t_linoleum_white",
+ "I": "t_door_locked_interior",
+ "L": "t_linoleum_white",
+ "M": "t_dirt",
+ "O": "t_window",
+ "S": "t_sidewalk",
+ "U": "t_linoleum_white",
+ "V": "t_wall_glass",
+ "W": "t_fencegate_c",
+ "_": "t_pavement",
+ "b": "t_dirt",
+ "c": "t_linoleum_white",
+ "f": "t_linoleum_white",
+ "l": "t_linoleum_white",
+ "w": "t_linoleum_white",
+ "x": "t_console_broken",
+ "z": "t_shrub",
+ "{": "t_door_glass_c",
+ "|": "t_wall_b",
+ "<": "t_stairs_up",
+ "4": "t_gutter_downspout",
+ "T": "t_tree_coffee"
+ },
+ "furniture": {
+ "#": "f_counter",
+ "%": "f_trashcan",
+ "/": "f_bluebell",
+ "6": "f_table",
+ "9": "f_rack",
+ ">": "f_counter",
+ "?": "f_sofa",
+ "A": "f_sink",
+ "B": "f_chair",
+ "C": "f_desk",
+ "D": "f_trashcan",
+ "F": "f_fridge",
+ "G": "f_oven",
+ "L": "f_locker",
+ "M": "f_dahlia",
+ "U": "f_sink",
+ "X": "f_rack",
+ "^": "f_indoor_plant",
+ "b": "f_dandelion",
+ "f": "f_glass_fridge",
+ "l": "f_rack",
+ "r": "f_rack"
+ },
+ "toilets": { ";": { } },
+ "items": {
+ "#": { "item": "coffee_counter", "chance": 10 },
+ "6": { "item": "coffee_table", "chance": 35 },
+ "9": { "item": "coffee_display_2", "chance": 85, "repeat": [ 1, 8 ] },
+ ";": { "item": "coffee_bathroom", "chance": 15 },
+ "=": { "item": "coffee_bathroom", "chance": 35 },
+ ">": { "item": "coffee_table", "chance": 25 },
+ "A": { "item": "coffee_bathroom", "chance": 35 },
+ "C": { "item": "office", "chance": 70 },
+ "D": { "item": "coffee_trash", "chance": 75 },
+ "F": { "item": "coffee_fridge", "chance": 80, "repeat": [ 1, 8 ] },
+ "G": { "item": "oven", "chance": 35 },
+ "L": { "item": "coffee_locker", "chance": 75 },
+ "U": { "item": "coffee_dishes", "chance": 75 },
+ "X": { "item": "coffee_newsstand", "chance": 90, "repeat": [ 1, 8 ] },
+ "f": { "item": "coffee_freezer", "chance": 85, "repeat": [ 1, 8 ] },
+ "l": { "item": "coffee_prep", "chance": 50 },
+ "r": { "item": "coffee_condiments", "chance": 80, "repeat": [ 1, 8 ] }
+ },
+ "monsters": { "!": { "monster": "GROUP_COFFEE_SHOP_ZOMBIE", "chance": 1 } },
+ "place_monsters": [ { "monster": "GROUP_ZOMBIE", "x": [ 3, 17 ], "y": [ 13, 18 ], "chance": 1 } ],
+ "vehicles": { "c": { "vehicle": "swivel_chair", "chance": 100, "status": 1 } }
+}
```
-1. The `"fill_ter"`: this tag defines the default terrain/flooring for use under furniture and for undefined symbols in your rows. Generally, pick the terrain that has the most furniture associated with it.
+1. The `"fill_ter"`: this tag defines the default terrain/flooring for use under furniture and for
+ undefined symbols in your rows. Generally, pick the terrain that has the most furniture
+ associated with it.
-2. The `"rows"`: this is the actual blueprint for your building. A standard OMT tile (overmap tile) is 24x24 tiles. Rows begin their x,y coordinates at 0,0 which is the upper left corner of your map.
- * tip: if you cross stitch or are familiar with cross stitch patterns, this should all look very familiar. You have the map and the "legend" areas.
+2. The `"rows"`: this is the actual blueprint for your building. A standard OMT tile (overmap tile)
+ is 24x24 tiles. Rows begin their x,y coordinates at 0,0 which is the upper left corner of your
+ map.
+ - tip: if you cross stitch or are familiar with cross stitch patterns, this should all look very
+ familiar. You have the map and the "legend" areas.
-3. `"terrain"`: this defines what all those letters in the rows mean when they are terrains. A symbol can return a single terrain, or, it can offer a chance to spawn from a selection of terrains. Here are some quick examples:
+3. `"terrain"`: this defines what all those letters in the rows mean when they are terrains. A
+ symbol can return a single terrain, or, it can offer a chance to spawn from a selection of
+ terrains. Here are some quick examples:
`"*": "t_door_c",`: this will make every * in the rows a closed wooden door.
-`"*": [ [ "t_door_locked_peep", 2 ], "t_door_locked_alarm", [ "t_door_locked", 10 ], "t_door_c" ],`: this array will randomly choose from a selection of doors. Some are weighted to have a higher chance to spawn then others. Locked doors will be most common, then peephole doors. Finally closed & locked/alarmed have the same basic weight and will spawn the least.
-
-*Note: in my example, the fill_ter is for floor. So if you have furniture that you want to spawn with a different floor, you must use that same symbol that you've given the furniture and also define it as a terrain for your new flooring. In the above example, several furniture symbols are using white linoleum for their flooring. If you don't do this step, your furniture will end up having the wrong flooring which will be especially noticeable if you smash it. I often do a final map check where I go around in game and smash furniture to check their terrains before submitting my maps. This can be quite cathartic.*
-
-4. `"furniture"` tag: Like terrain, this is a list of the furniture ID's and their map symbols. It can handle the same sort of arrays as terrain.
-
-5. `"toilets"` and other specially defined furniture: you'll run into some specially defined common furniture which allows for some easier placement. In our sample map the entry: `"toilets": { ";": { } },` defines the symbol entry and will also auto-place water in your toilets. There are a few other specialty furniture entries.
-
-The other most common one is: `"vendingmachines": { "D": { "item_group": "vending_drink" }, "V": { "item_group": "vending_food" } }` this assigns two symbols for vending machines and makes one for food & one for drinks. *note: you can put any item_group into the machines, like those bullet ones*.
-
-6. Item spawns: There are many ways to place items. This tutorial will only cover explicit symbol placement which is the easiest. There is documentation all about loot spawns you can read for further information. See: [doc/ITEM_SPAWN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/ITEM_SPAWN.md).
-
-our sample uses "items": for its tag. others include: "place_item", "place_items", "place_loot". Some of these allow for individual item placement and others groups, or both. This will be covered in another tutorial.
+`"*": [ [ "t_door_locked_peep", 2 ], "t_door_locked_alarm", [ "t_door_locked", 10 ], "t_door_c" ],`:
+this array will randomly choose from a selection of doors. Some are weighted to have a higher chance
+to spawn then others. Locked doors will be most common, then peephole doors. Finally closed &
+locked/alarmed have the same basic weight and will spawn the least.
+
+_Note: in my example, the fill_ter is for floor. So if you have furniture that you want to spawn
+with a different floor, you must use that same symbol that you've given the furniture and also
+define it as a terrain for your new flooring. In the above example, several furniture symbols are
+using white linoleum for their flooring. If you don't do this step, your furniture will end up
+having the wrong flooring which will be especially noticeable if you smash it. I often do a final
+map check where I go around in game and smash furniture to check their terrains before submitting my
+maps. This can be quite cathartic._
+
+4. `"furniture"` tag: Like terrain, this is a list of the furniture ID's and their map symbols. It
+ can handle the same sort of arrays as terrain.
+
+5. `"toilets"` and other specially defined furniture: you'll run into some specially defined common
+ furniture which allows for some easier placement. In our sample map the entry:
+ `"toilets": { ";": { } },` defines the symbol entry and will also auto-place water in your
+ toilets. There are a few other specialty furniture entries.
+
+The other most common one is:
+`"vendingmachines": { "D": { "item_group": "vending_drink" }, "V": { "item_group": "vending_food" } }`
+this assigns two symbols for vending machines and makes one for food & one for drinks. _note: you
+can put any item_group into the machines, like those bullet ones_.
+
+6. Item spawns: There are many ways to place items. This tutorial will only cover explicit symbol
+ placement which is the easiest. There is documentation all about loot spawns you can read for
+ further information. See:
+ [doc/ITEM_SPAWN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/ITEM_SPAWN.md).
+
+our sample uses "items": for its tag. others include: "place_item", "place_items", "place_loot".
+Some of these allow for individual item placement and others groups, or both. This will be covered
+in another tutorial.
For now lets break this one apart:
+
```
"items": {
"a": { "item": "stash_wood", "chance": 30, "repeat": [ 2, 5 ] },
@@ -229,11 +307,20 @@ For now lets break this one apart:
}
```
-"a" in this case is a fireplace as defined in its map. So, in the fireplace I want to add the stash_wood item_group. By defining the group under "a", every fireplace in my map will get this group. This is particularly powerful for common item_groups like in houses where every fridge is going to get the same item_group. The chance is its spawn chance and repeat means that it will repeat the roll for that chance 2-5 times, so this fireplace can be extra stocked or have a little bit or nothing if it fails its chance rolls.
+"a" in this case is a fireplace as defined in its map. So, in the fireplace I want to add the
+stash_wood item_group. By defining the group under "a", every fireplace in my map will get this
+group. This is particularly powerful for common item_groups like in houses where every fridge is
+going to get the same item_group. The chance is its spawn chance and repeat means that it will
+repeat the roll for that chance 2-5 times, so this fireplace can be extra stocked or have a little
+bit or nothing if it fails its chance rolls.
+
+the "d" entry is a dresser. I wanted the dressers to pull from two possible item groups, one a man's
+selection and the other women’s. So, there is an array [ ... ] which encompasses all possible
+item_groups for this symbol.
-the "d" entry is a dresser. I wanted the dressers to pull from two possible item groups, one a man's selection and the other women’s. So, there is an array [ ... ] which encompasses all possible item_groups for this symbol.
+You can add as many item_groups to the array as you'd like. This is one of my racks in the generic
+house palette:
-You can add as many item_groups to the array as you'd like. This is one of my racks in the generic house palette:
```
"q": [
{ "item": "tools_home", "chance": 40 },
@@ -244,23 +331,37 @@ You can add as many item_groups to the array as you'd like. This is one of my r
],
```
-*Note: When using explicit symbol placements, remember your group has a chance to spawn in every furniture using that symbol, so it can end up being quite generous. If you want two bookcases with different item spawns, give each bookcase its own symbol, or, use an alternate item spawn format, like the one using x,y coordinates for placement.*
+_Note: When using explicit symbol placements, remember your group has a chance to spawn in every
+furniture using that symbol, so it can end up being quite generous. If you want two bookcases with
+different item spawns, give each bookcase its own symbol, or, use an alternate item spawn format,
+like the one using x,y coordinates for placement._
+
+7. Monster spawns: our example has two types of monster spawns listed.
-7. Monster spawns: our example has two types of monster spawns listed.
```
"monsters": { "!": { "monster": "GROUP_COFFEE_SHOP_ZOMBIE", "chance": 1 } },
"place_monsters": [ { "monster": "GROUP_ZOMBIE", "x": [ 3, 17 ], "y": [ 13, 18 ], "chance": 1 } ],
```
-The first entry is using that explicit symbol placement technique. The end entry is using a range x,y coordinates to place monsters. *Note: We are moving away from putting monsters in the mapgen file in favor of overmap_terrain entries, so only use this if you want a specific monster group to spawn for specific reasons. See overmap_terrain section for more information.*
+The first entry is using that explicit symbol placement technique. The end entry is using a range
+x,y coordinates to place monsters. _Note: We are moving away from putting monsters in the mapgen
+file in favor of overmap_terrain entries, so only use this if you want a specific monster group to
+spawn for specific reasons. See overmap_terrain section for more information._
8. Vehicle spawns:
+
```
"vehicles": { "c": { "vehicle": "swivel_chair", "chance": 100, "status": 1 } }
```
+
Our vehicle happens to be a swivel chair using explicit symbol placement.
-This next example uses vehicle_groups and x,y placement. It also includes rotation and status. The rotation is which direction the vehicle will spawn on the map, and that status is its overall condition. Fuel is pretty self explanatory. Always test your vehcile spawns in game, they can be rather picky in their placement and the rotation doesn't really match what you'd expect the numbers to mean. The 0,0 point of vehicles can vary so you'll have to experiment to get the spawns in the right spots, especially in tight spaces.
+This next example uses vehicle_groups and x,y placement. It also includes rotation and status. The
+rotation is which direction the vehicle will spawn on the map, and that status is its overall
+condition. Fuel is pretty self explanatory. Always test your vehcile spawns in game, they can be
+rather picky in their placement and the rotation doesn't really match what you'd expect the numbers
+to mean. The 0,0 point of vehicles can vary so you'll have to experiment to get the spawns in the
+right spots, especially in tight spaces.
```
"place_vehicles": [
@@ -273,17 +374,21 @@ This next example uses vehicle_groups and x,y placement. It also includes rotat
{ "chance": 75, "fuel": 0, "rotation": 270, "status": 1, "vehicle": "junkyard_vehicles", "x": 28, "y": 18 },
{ "chance": 75, "fuel": 0, "rotation": 270, "status": 1, "vehicle": "junkyard_vehicles", "x": 35, "y": 18 }
]
- ```
+```
-9. Liquids in furniture: this entry is for a standing tank. I've defined the tank in the furniture entry as "g".
+9. Liquids in furniture: this entry is for a standing tank. I've defined the tank in the furniture
+ entry as "g".
`"liquids": { "g": { "liquid": "water_clean", "amount": [ 0, 100 ] } },`
This places clean water in the tank, and a range of amount to spawn.
-10. There are other specialty placement techniques that you'll pick up as you look at more maps. One of my favorites is for the new planters:
+10. There are other specialty placement techniques that you'll pick up as you look at more maps. One
+ of my favorites is for the new planters:
-Since the planter is a "sealed item" you define what's going into that container. This example places seeds (ready to harvest) in the planters. Note the first one will place a seedling, the others are harvest ready. I've given each planter type an explicit symbol for quicker placement.
+Since the planter is a "sealed item" you define what's going into that container. This example
+places seeds (ready to harvest) in the planters. Note the first one will place a seedling, the
+others are harvest ready. I've given each planter type an explicit symbol for quicker placement.
```
"sealed_item": {
@@ -297,111 +402,139 @@ Since the planter is a "sealed item" you define what's going into that container
"8": { "item": { "item": "soybean_seed" }, "furniture": "f_planter_harvest" },
"9": { "item": { "item": "seed_zucchini" }, "furniture": "f_planter_harvest" }
}
- ```
+```
-11. Best practices:
- * If you are making a new house please use this palette: "standard_domestic_palette". The loots are already assigned and it covers a wide range of domestic furniture. This will keep your house in sync with all the other houses for loot spawns.
- * All buildings should also get roof entries.
- * While entry placement for json doesn't really matter, try to keep your mapgen files ordered like the majority existing maps. Be kind to future contributors.
- Add meta data at the top of the file.
- Object entry comes second. Within the object entry it is generally: palette, set_points, terrain, furniture, random special entries like toilets, item spawns, vehicle spawns, monster spawns.
- * All buildings should at least get one "t_gutter_downspout" for roof access. Players would love at least 2.
- if you're adding gutters to a multi-z level building, don't forget the intermediate floors. You'll need to stagger the downspouts to players can climb up (like ladders).
- * If you put things on your roof that would be difficult to get up there, make sure to provide better roof access via ladders and stairs.
- * For your basic grass cover outside please use: `"t_region_groundcover_urban",` to maintain consistency across map boundaries. Here are my standard flora entries for grass, shrubs & trees:
+11. Best practices:
+
+- If you are making a new house please use this palette: "standard_domestic_palette". The loots are
+ already assigned and it covers a wide range of domestic furniture. This will keep your house in
+ sync with all the other houses for loot spawns.
+- All buildings should also get roof entries.
+- While entry placement for json doesn't really matter, try to keep your mapgen files ordered like
+ the majority existing maps. Be kind to future contributors. Add meta data at the top of the file.
+ Object entry comes second. Within the object entry it is generally: palette, set_points, terrain,
+ furniture, random special entries like toilets, item spawns, vehicle spawns, monster spawns.
+- All buildings should at least get one "t_gutter_downspout" for roof access. Players would love at
+ least 2. if you're adding gutters to a multi-z level building, don't forget the intermediate
+ floors. You'll need to stagger the downspouts to players can climb up (like ladders).
+- If you put things on your roof that would be difficult to get up there, make sure to provide
+ better roof access via ladders and stairs.
+- For your basic grass cover outside please use: `"t_region_groundcover_urban",` to maintain
+ consistency across map boundaries. Here are my standard flora entries for grass, shrubs & trees:
```
- ".": "t_region_groundcover_urban",
- "A": [ "t_region_shrub", "t_region_shrub_fruit", "t_region_shrub_decorative" ],
- "Z": [ [ "t_region_tree_fruit", 2 ], [ "t_region_tree_nut", 2 ], "t_region_tree_shade" ],
+".": "t_region_groundcover_urban",
+"A": [ "t_region_shrub", "t_region_shrub_fruit", "t_region_shrub_decorative" ],
+"Z": [ [ "t_region_tree_fruit", 2 ], [ "t_region_tree_nut", 2 ], "t_region_tree_shade" ],
```
finally for flowers (which are furniture):
+
```
"p": "f_region_flower"
```
#### Adding the roof!
-Almost all CDDA buildings are now roof-capable and we'd love to keep it that way. Make sure to submit a roof map with your building. This can go into the same file as your ground floor and any other floors that share the same building shape/foundation.
+Almost all CDDA buildings are now roof-capable and we'd love to keep it that way. Make sure to
+submit a roof map with your building. This can go into the same file as your ground floor and any
+other floors that share the same building shape/foundation.
-So, this is super easy compared to the building we just went over. It has all the same basic components. I recommend you start by using the rows from your ground floor map and converting it to the `"roof_palette"` symbol set. Basically your just going to trace the outline in gutters, add a t_gutter_drop next to your t_gutter_spout below and toss some infrastructure up there. I used nests extensively in commercial building roofs and we'll cover that in advanced mapgen.
+So, this is super easy compared to the building we just went over. It has all the same basic
+components. I recommend you start by using the rows from your ground floor map and converting it to
+the `"roof_palette"` symbol set. Basically your just going to trace the outline in gutters, add a
+t_gutter_drop next to your t_gutter_spout below and toss some infrastructure up there. I used nests
+extensively in commercial building roofs and we'll cover that in advanced mapgen.
sample roof:
+
```
- {
- "type": "mapgen",
- "method": "json",
- "om_terrain": "house_01_roof",
- "object": {
- "fill_ter": "t_shingle_flat_roof",
- "rows": [
- " ",
- " ",
- " ",
- " |222222222222222223 ",
- " |.................3 ",
- " |.................3 ",
- " |.............N...3 ",
- " 5.................3 ",
- " |.................3 ",
- " |.................3 ",
- " |........&........3 ",
- " |.................3 ",
- " |......=..........3 ",
- " |.................3 ",
- " |----------------53 ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " "
- ],
- "palettes": [ "roof_palette" ],
- "terrain": { ".": "t_shingle_flat_roof" }
- }
+{
+ "type": "mapgen",
+ "method": "json",
+ "om_terrain": "house_01_roof",
+ "object": {
+ "fill_ter": "t_shingle_flat_roof",
+ "rows": [
+ " ",
+ " ",
+ " ",
+ " |222222222222222223 ",
+ " |.................3 ",
+ " |.................3 ",
+ " |.............N...3 ",
+ " 5.................3 ",
+ " |.................3 ",
+ " |.................3 ",
+ " |........&........3 ",
+ " |.................3 ",
+ " |......=..........3 ",
+ " |.................3 ",
+ " |----------------53 ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " "
+ ],
+ "palettes": [ "roof_palette" ],
+ "terrain": { ".": "t_shingle_flat_roof" }
}
- ```
+}
+```
- 1. I always just append `_roof` to the buildings ID.
- 2. See how the palette takes the place of all that data from our earlier example. So clean and easy.
- 3. There is no `"weight"` entry because this will only spawn with its building (once linked).
- 4. My palette uses "t_flat_roof" as its default roof. For houses, I wanted shingles. So, I added the "t_shingle_flat_roof" in this mapgen which will override the palettes entry for `".": "t_flat_roof"`. (more on this in advanced mapgen).
+1. I always just append `_roof` to the buildings ID.
+2. See how the palette takes the place of all that data from our earlier example. So clean and easy.
+3. There is no `"weight"` entry because this will only spawn with its building (once linked).
+4. My palette uses "t_flat_roof" as its default roof. For houses, I wanted shingles. So, I added the
+ "t_shingle_flat_roof" in this mapgen which will override the palettes entry for
+ `".": "t_flat_roof"`. (more on this in advanced mapgen).
-I have a separate roof document at: [doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/JSON%20Mapping%20Guides/JSON_ROOF_MAPGEN.md).
+I have a separate roof document at:
+[doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/JSON%20Mapping%20Guides/JSON_ROOF_MAPGEN.md).
#### Linking various mapgen maps using multitile_city_buildings.json
- This file is found at: [data/json/overmap/multitile_city_buildings.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/overmap/multitile_city_buildings.json).
+This file is found at:
+[data/json/overmap/multitile_city_buildings.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/overmap/multitile_city_buildings.json).
- *Remember this file is for city buildings only, not specials*
+_Remember this file is for city buildings only, not specials_
- A standard entry:
- ```
- {
- "type": "city_building",
- "id": "house_dogs",
- "locations": [ "land" ],
- "overmaps": [
- { "point": [ 0, 0, 0 ], "overmap": "house_dogs_north" },
- { "point": [ 0, 0, 1 ], "overmap": "house_dogs_roof_north" },
- { "point": [ 0, 0, -1 ], "overmap": "basement" }
- ]
- },
- ```
+A standard entry:
+
+```
+{
+ "type": "city_building",
+ "id": "house_dogs",
+ "locations": [ "land" ],
+ "overmaps": [
+ { "point": [ 0, 0, 0 ], "overmap": "house_dogs_north" },
+ { "point": [ 0, 0, 1 ], "overmap": "house_dogs_roof_north" },
+ { "point": [ 0, 0, -1 ], "overmap": "basement" }
+ ]
+},
+```
- 1. The `"type"` won't change. It should always be "city_building".
- 2. `"id"`: This ID is often the same as your mapgen ID but it doesn't have to be the same. We could use something more generic like "house". This ID will be used in regional settings for spawns, so keep in mind how many buildings are using the ID. I prefer distinct ID's because it makes debug spawning much, much easier.
- 3. `"locations"`: defines where this building can be place by overmap terrain type. Land is the default.
- 4. `"overmaps"`: this is the bit where you define how the maps fit together, so lets break it up:
- `{ "point": [ 0, 0, 0 ], "overmap": "house_dogs_north" },`
- point: its point in relation to the other mapgen files for your building. The coordinates are [ x, y, z ]. In this example, x,y are 0 because we only have one map per z level. Zero for y means this is the ground level. Note the roof above is at 1 and the basement is -1.
- 5. appending `_north` to the ID's:
- * If your building rotates you need this compass point so the floors can match up correctly. This is the generic basement mapgen group and thus doesn't get `_north` (this will change as we add dedicated stairs to our houses).
+1. The `"type"` won't change. It should always be "city_building".
+2. `"id"`: This ID is often the same as your mapgen ID but it doesn't have to be the same. We could
+ use something more generic like "house". This ID will be used in regional settings for spawns, so
+ keep in mind how many buildings are using the ID. I prefer distinct ID's because it makes debug
+ spawning much, much easier.
+3. `"locations"`: defines where this building can be place by overmap terrain type. Land is the
+ default.
+4. `"overmaps"`: this is the bit where you define how the maps fit together, so lets break it up:
+ `{ "point": [ 0, 0, 0 ], "overmap": "house_dogs_north" },` point: its point in relation to the
+ other mapgen files for your building. The coordinates are [ x, y, z ]. In this example, x,y are 0
+ because we only have one map per z level. Zero for y means this is the ground level. Note the
+ roof above is at 1 and the basement is -1.
+5. appending `_north` to the ID's:
+ - If your building rotates you need this compass point so the floors can match up correctly. This
+ is the generic basement mapgen group and thus doesn't get `_north` (this will change as we add
+ dedicated stairs to our houses).
#### Setting overmap spawns using regional_map_settings.json
@@ -409,16 +542,20 @@ I have a separate roof document at: [doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md
1. For city buildings and houses you'll scroll down to the `"city":` flag.
2. Find your appropriate subtag, `"houses"` or `"shops"` usually.
-3. Add your ID from the multitile_city_buildings entry. This can also accept the mapgen ID and not complain which is why the are often the same name.
+3. Add your ID from the multitile_city_buildings entry. This can also accept the mapgen ID and not
+ complain which is why the are often the same name.
4. Choose a good weight for your building.
#### Linking and spawning specials:
-Put the entry in: [data/json/overmap/overmap_special/specials.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/overmap/overmap_special/specials.json).
+Put the entry in:
+[data/json/overmap/overmap_special/specials.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/overmap/overmap_special/specials.json).
-This entry does the job of both the regional_map_settings and multitile_city_buildings plus other fun overmap stuff.
+This entry does the job of both the regional_map_settings and multitile_city_buildings plus other
+fun overmap stuff.
Example:
+
```
{
"type": "overmap_special",
@@ -444,146 +581,176 @@ Example:
"occurrences": [ 0, 1 ],
"flags": [ "CLASSIC" ]
}
- ```
-
- 1. `"type"`: is overmap_special.
- 2. `"id"` is your buildings ID for the overmap. It also displays on the overmap in game.
- 3. `"overmaps"` this works the same way as it does in the city building entries. Note that the pump station is bigger then 1 OMT on the ground, so the y coordinate changes as well.
- 4. `"connections"`: this places road, sewer, subway connections for your map.
- 5. `"locations"`: valid OMT types this building can be placed on.
- 6. `"city_distance"`, `"city_sizes"` both are parameters for where this spawns in relation to cities.
- 7. `"occurrences": [ 0, 1 ],`: Ok so occurrences can mean two things depending on if it uses the "UNIQUE" flag or not. When the flag is absent, this simply translates to how many times this special can spawn PER overmap. So 0 to 1 in this case.
- If you use the UNIQUE flag, this becomes a percentage so [ 1, 10 ] wouldn't be 1 to 10 times per overmap but a 1 in 10% chance to spawn on the overmap. So 10% chance to spawn once per overmap.
- 8. `"flags"`: These are flags you can use to further define the special. For a list of flags see: [doc/JSON_FLAGS.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/JSON_FLAGS.md).
+```
-Read: [doc/OVERMAP.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/OVERMAP.md) for more details.
+1. `"type"`: is overmap_special.
+2. `"id"` is your buildings ID for the overmap. It also displays on the overmap in game.
+3. `"overmaps"` this works the same way as it does in the city building entries. Note that the pump
+ station is bigger then 1 OMT on the ground, so the y coordinate changes as well.
+4. `"connections"`: this places road, sewer, subway connections for your map.
+5. `"locations"`: valid OMT types this building can be placed on.
+6. `"city_distance"`, `"city_sizes"` both are parameters for where this spawns in relation to
+ cities.
+7. `"occurrences": [ 0, 1 ],`: Ok so occurrences can mean two things depending on if it uses the
+ "UNIQUE" flag or not. When the flag is absent, this simply translates to how many times this
+ special can spawn PER overmap. So 0 to 1 in this case. If you use the UNIQUE flag, this becomes a
+ percentage so [ 1, 10 ] wouldn't be 1 to 10 times per overmap but a 1 in 10% chance to spawn on
+ the overmap. So 10% chance to spawn once per overmap.
+8. `"flags"`: These are flags you can use to further define the special. For a list of flags see:
+ [doc/JSON_FLAGS.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/JSON_FLAGS.md).
+
+Read: [doc/OVERMAP.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/OVERMAP.md) for
+more details.
#### Overmap_terrain entries:
-Choose a file for your building type at: [data/json/overmap/overmap_terrain](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/data/json/overmap/overmap_terrain).
+Choose a file for your building type at:
+[data/json/overmap/overmap_terrain](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/data/json/overmap/overmap_terrain).
-This set of entries defines how your building will look on the overmap. It supports copy-from.
+This set of entries defines how your building will look on the overmap. It supports copy-from.
Example:
+
+```
+{
+ "type": "overmap_terrain",
+ "id": "s_music",
+ "copy-from": "generic_city_building",
+ "name": "music store",
+ "sym": "m",
+ "color": "brown",
+ "mondensity": 2,
+ "extend": { "flags": [ "SOURCE_LUXURY" ] }
+},
+{
+ "type": "overmap_terrain",
+ "id": "s_music_roof",
+ "copy-from": "generic_city_building",
+ "name": "music store roof",
+ "sym": "m",
+ "color": "brown"
+}
```
- {
- "type": "overmap_terrain",
- "id": "s_music",
- "copy-from": "generic_city_building",
- "name": "music store",
- "sym": "m",
- "color": "brown",
- "mondensity": 2,
- "extend": { "flags": [ "SOURCE_LUXURY" ] }
- },
- {
- "type": "overmap_terrain",
- "id": "s_music_roof",
- "copy-from": "generic_city_building",
- "name": "music store roof",
- "sym": "m",
- "color": "brown"
- }
- ```
You need one entry per mapgen ID:
+
1. `"type"` will always be overmap_terrain.
2. `"id"` will be the same ID you used in your mapgen file.
3. `"copy-from"` this will copy any data from another entry, excepting what you define here.
4. `"name"` how the name displays on the overmap.
5. `"sym"` the symbol displayed on the overmap. If left out, the carrots will be used `v<>^`
6. `"color"` color for overmap symbol.
-7. `"mondesntiy"` sets the default monster density for this overmap tile. You'll use this for general zombie spawns and reserve the mapgen monster entries for special spanwns for that location (e.g. a pet store's pets).
-8. `"extend"` many of these flags will be used by NPCs in the future for their AI, try to add flags appropriate for your location. Others further define the mapgen, like having sidewalks generate.
+7. `"mondesntiy"` sets the default monster density for this overmap tile. You'll use this for
+ general zombie spawns and reserve the mapgen monster entries for special spanwns for that
+ location (e.g. a pet store's pets).
+8. `"extend"` many of these flags will be used by NPCs in the future for their AI, try to add flags
+ appropriate for your location. Others further define the mapgen, like having sidewalks generate.
-For further information see: [Overmap Terrain section of doc/OVERMAP.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/OVERMAP.md#overmap-terrain).
+For further information see:
+[Overmap Terrain section of doc/OVERMAP.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/OVERMAP.md#overmap-terrain).
#### Palettes:
-As mentioned earlier, palettes can hold almost all the information that the object entry contains, except for `rows` and `fill_ter`. Their main purpose is to reduce the need to add the same basic data to related maps and to maintain symbol consistency.
+As mentioned earlier, palettes can hold almost all the information that the object entry contains,
+except for `rows` and `fill_ter`. Their main purpose is to reduce the need to add the same basic
+data to related maps and to maintain symbol consistency.
-Entries that are added to the mapgen file's object will over ride the same symbol used in the palette. In this way, you can use a palette and make select alterations to each mapgen as needed. This is especially useful for multiple ground terrains like carpets, concrete, etc.
+Entries that are added to the mapgen file's object will over ride the same symbol used in the
+palette. In this way, you can use a palette and make select alterations to each mapgen as needed.
+This is especially useful for multiple ground terrains like carpets, concrete, etc.
-Terrain works very well when you substitute it via the mapgen file. I've had less success overriding furniture but need to test it more to clarify when it works as intended. I have recently noticed that if your palette symbol uses an array of values, the mapgen entry can't override it.
+Terrain works very well when you substitute it via the mapgen file. I've had less success overriding
+furniture but need to test it more to clarify when it works as intended. I have recently noticed
+that if your palette symbol uses an array of values, the mapgen entry can't override it.
-Example:
-Entry for the mapgen file object: `"palettes": [ "roof_palette" ],`
+Example: Entry for the mapgen file object: `"palettes": [ "roof_palette" ],`
The palette metadata:
+
```
- "type": "palette",
- "id": "roof_palette",
+"type": "palette",
+"id": "roof_palette",
```
Everything else will look like a series of object entries, for example the roof_palette:
```
- {
- "type": "palette",
- "id": "roof_palette",
- "terrain": {
- ".": "t_flat_roof",
- " ": "t_open_air",
- "o": "t_glass_roof",
- "_": "t_floor",
- "2": "t_gutter_north",
- "-": "t_gutter_south",
- "3": "t_gutter_east",
- "4": "t_gutter_downspout",
- "|": "t_gutter_west",
- "5": "t_gutter_drop",
- "#": "t_grate",
- "&": "t_null",
- ":": "t_null",
- "X": "t_null",
- "=": "t_null",
- "A": "t_null",
- "b": "t_null",
- "c": "t_null",
- "t": "t_null",
- "r": "t_null",
- "L": "t_null",
- "C": "t_null",
- "Y": "t_null",
- "y": "t_null"
- },
- "furniture": {
- "&": "f_roof_turbine_vent",
- "N": "f_TV_antenna",
- ":": "f_cellphone_booster",
- "X": "f_small_satelitte_dish",
- "~": "f_chimney",
- "=": "f_vent_pipe",
- "A": "f_air_conditioner",
- "b": "f_bench",
- "c": "f_counter",
- "t": "f_table",
- "r": "f_rack",
- "L": "f_locker",
- "C": [ "f_crate_c", "f_cardboard_box" ],
- "Y": "f_stool",
- "s": "f_sofa",
- "S": "f_sink",
- "e": "f_oven",
- "F": "f_fridge",
- "y": [ "f_indoor_plant_y", "f_indoor_plant" ]
- },
- "toilets": { "T": { } }
- }
- ```
+{
+ "type": "palette",
+ "id": "roof_palette",
+ "terrain": {
+ ".": "t_flat_roof",
+ " ": "t_open_air",
+ "o": "t_glass_roof",
+ "_": "t_floor",
+ "2": "t_gutter_north",
+ "-": "t_gutter_south",
+ "3": "t_gutter_east",
+ "4": "t_gutter_downspout",
+ "|": "t_gutter_west",
+ "5": "t_gutter_drop",
+ "#": "t_grate",
+ "&": "t_null",
+ ":": "t_null",
+ "X": "t_null",
+ "=": "t_null",
+ "A": "t_null",
+ "b": "t_null",
+ "c": "t_null",
+ "t": "t_null",
+ "r": "t_null",
+ "L": "t_null",
+ "C": "t_null",
+ "Y": "t_null",
+ "y": "t_null"
+ },
+ "furniture": {
+ "&": "f_roof_turbine_vent",
+ "N": "f_TV_antenna",
+ ":": "f_cellphone_booster",
+ "X": "f_small_satelitte_dish",
+ "~": "f_chimney",
+ "=": "f_vent_pipe",
+ "A": "f_air_conditioner",
+ "b": "f_bench",
+ "c": "f_counter",
+ "t": "f_table",
+ "r": "f_rack",
+ "L": "f_locker",
+ "C": [ "f_crate_c", "f_cardboard_box" ],
+ "Y": "f_stool",
+ "s": "f_sofa",
+ "S": "f_sink",
+ "e": "f_oven",
+ "F": "f_fridge",
+ "y": [ "f_indoor_plant_y", "f_indoor_plant" ]
+ },
+ "toilets": { "T": { } }
+}
+```
-If you want to look at more complex palettes, the standard_domestic_palette in [data/json/mapgen_palettes/house_general_palette.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/mapgen_palettes/house_general_palette.json) is a good look at a palette designed to work across all CDDA houses. It includes the loot spawns and accounts for most furniture that will be used in a house. I also left a list of symbols open to be used in the mapgen file for specific location needs.
+If you want to look at more complex palettes, the standard_domestic_palette in
+[data/json/mapgen_palettes/house_general_palette.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/mapgen_palettes/house_general_palette.json)
+is a good look at a palette designed to work across all CDDA houses. It includes the loot spawns and
+accounts for most furniture that will be used in a house. I also left a list of symbols open to be
+used in the mapgen file for specific location needs.
-Finally, the series of house_w palettes at [data/json/mapgen_palettes/house_w_palette.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/mapgen_palettes/house_w_palette.json) are designed to work together for houses using nested mapgen. There is a palette devoted to the foundation, another for the nests, and finally another one I've designed for domestic outdoor nested chunks.
+Finally, the series of house_w palettes at
+[data/json/mapgen_palettes/house_w_palette.json](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/data/json/mapgen_palettes/house_w_palette.json)
+are designed to work together for houses using nested mapgen. There is a palette devoted to the
+foundation, another for the nests, and finally another one I've designed for domestic outdoor nested
+chunks.
#### Final comments:
-The information here should be enough for you got get around mapgen and start making maps but there are a lot of variations that will be covered in focused.
+The information here should be enough for you got get around mapgen and start making maps but there
+are a lot of variations that will be covered in focused.
Not covered in this document:
- * Nested maps and their placement.
- * NPC spawns.
- * Advanced terrain tricks for complex floor options.
- * traps, terrain and you.
- * update_mapgen (NPC and player triggered map updates).
- * faction camp expansion maps.
- * field emitting furniture.
+
+- Nested maps and their placement.
+- NPC spawns.
+- Advanced terrain tricks for complex floor options.
+- traps, terrain and you.
+- update_mapgen (NPC and player triggered map updates).
+- faction camp expansion maps.
+- field emitting furniture.
diff --git a/doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md b/doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md
index 883e0fd25530..efd6d87f04fb 100644
--- a/doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md
+++ b/doc/JSON Mapping Guides/JSON_ROOF_MAPGEN.md
@@ -1,6 +1,7 @@
# Adding Json Roof Guide
-Adding json roofs to a building involves using a few more files to link the roof and building together during mapgen.
+Adding json roofs to a building involves using a few more files to link the roof and building
+together during mapgen.
Files that will be edited:
@@ -14,140 +15,155 @@ Files that will be edited:
## Making the Roof Map
-Refer to [MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/MAPGEN.md) for creating the map if you are new to map creation.
+Refer to [MAPGEN.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/MAPGEN.md) for
+creating the map if you are new to map creation.
-Open the file that contains the map for the building `data/json/mapgen/[name of building].json`
-Add a new entry for the roof. You can copy the building entry since you want the same foundation footprint for the roof.
+Open the file that contains the map for the building `data/json/mapgen/[name of building].json` Add
+a new entry for the roof. You can copy the building entry since you want the same foundation
+footprint for the roof.
-Give the roof a unique om_terrain ID. Below is an example of the main floor and roof om_terrain IDs
+Give the roof a unique om_terrain ID. Below is an example of the main floor and roof om_terrain IDs
-````json
+```json
"om_terrain": [ "abstorefront" ],
"om_terrain": [ "abstorefront_roof" ],
-````
+```
-Note: If you are adding a roof to an existing building that shares a common om_terrain ID with other maps, you will need to change the om_terrain ID for the existing floor to be unique.
+Note: If you are adding a roof to an existing building that shares a common om_terrain ID with other
+maps, you will need to change the om_terrain ID for the existing floor to be unique.
-Keep the outline of the walls from the original floor,
-Add `t_open_air` outside the building and a `t_flat_roof` over the building's footprint.
-There are a few flat roof terrains to choose from in terrain.json.
+Keep the outline of the walls from the original floor, Add `t_open_air` outside the building and a
+`t_flat_roof` over the building's footprint. There are a few flat roof terrains to choose from in
+terrain.json.
-There are a number of terrains and furniture structures for roofs including gutters, chimneys, and roof turbine vents.
-Browse json/terrain.json and furniture.json for ideas. Consider roof access, you can use ladders, stairs and gutters. Some furniture is also climbable.
+There are a number of terrains and furniture structures for roofs including gutters, chimneys, and
+roof turbine vents. Browse json/terrain.json and furniture.json for ideas. Consider roof access, you
+can use ladders, stairs and gutters. Some furniture is also climbable.
-There is a set of optional nested map chunks at `data/json/mapgen/nested_chunks_roof.json` if you'd like to incorporate them. Add any new nested chunks for roofs here as well.
+There is a set of optional nested map chunks at `data/json/mapgen/nested_chunks_roof.json` if you'd
+like to incorporate them. Add any new nested chunks for roofs here as well.
Sample roof entry:
-````json
- {
- "type": "mapgen",
- "method": "json",
- "om_terrain": "abstorefront_roof",
- "weight": 200,
- "object": {
- "fill_ter": "t_flat_roof",
- "rows": [
- " ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |....................3 ",
- " |......&.............3 ",
- " |....................3 ",
- " |....................3 ",
- " |-----------------5--3 ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " ",
- " "
- ],
- "terrain": {
- ".": "t_flat_roof",
- " ": "t_open_air",
- "|": "t_gutter_west",
- "-": "t_gutter_south",
- "3": "t_gutter_east",
- "5": "t_gutter_drop"
- },
- "furniture": { "&": "f_roof_turbine_vent" },
- "place_items": [ { "item": "roof_trash", "x": [ 2, 21 ], "y": [ 3, 14 ], "chance": 50, "repeat": [ 1, 3 ] } ],
- "place_nested": [
- {
- "chunks": [
- [ "null", 50 ],
- [ "roof_4x4_party", 15 ],
- [ "roof_4x4_holdout", 5 ],
- [ "roof_4x4_utility", 40 ],
- [ "roof_5x5_coop", 5 ]
- ],
- "x": [ 3, 15 ],
- "y": [ 3, 7 ]
- }
- ]
- }
+
+```json
+{
+ "type": "mapgen",
+ "method": "json",
+ "om_terrain": "abstorefront_roof",
+ "weight": 200,
+ "object": {
+ "fill_ter": "t_flat_roof",
+ "rows": [
+ " ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |......&.............3 ",
+ " |....................3 ",
+ " |....................3 ",
+ " |-----------------5--3 ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " ",
+ " "
+ ],
+ "terrain": {
+ ".": "t_flat_roof",
+ " ": "t_open_air",
+ "|": "t_gutter_west",
+ "-": "t_gutter_south",
+ "3": "t_gutter_east",
+ "5": "t_gutter_drop"
+ },
+ "furniture": { "&": "f_roof_turbine_vent" },
+ "place_items": [
+ { "item": "roof_trash", "x": [2, 21], "y": [3, 14], "chance": 50, "repeat": [1, 3] }
+ ],
+ "place_nested": [
+ {
+ "chunks": [
+ ["null", 50],
+ ["roof_4x4_party", 15],
+ ["roof_4x4_holdout", 5],
+ ["roof_4x4_utility", 40],
+ ["roof_5x5_coop", 5]
+ ],
+ "x": [3, 15],
+ "y": [3, 7]
+ }
+ ]
+ }
}
-````
+```
## Linking the main floor and the roof
-Navigate to `json/overmap/multitile_city_buildings.json` or `json/overmap/multitile_buildings_terrain.json` for buildings taking up more then one overmap tile per z level (schools, mansions).
-Add an entry for the main floor. The `point` coordinates define to the x, y, z positions of the building. The 1 places the roof one z level above the ground floor.
+
+Navigate to `json/overmap/multitile_city_buildings.json` or
+`json/overmap/multitile_buildings_terrain.json` for buildings taking up more then one overmap tile
+per z level (schools, mansions). Add an entry for the main floor. The `point` coordinates define to
+the x, y, z positions of the building. The 1 places the roof one z level above the ground floor.
Append north for rotating buildings to orient the z levels.
-````json
- {
- "type": "city_building",
- "id": "abstorefront",
- "locations": [ "land" ],
- "overmaps": [
- { "point": [ 0, 0, 0 ], "overmap": "abstorefront_north" },
- { "point": [ 0, 0, 1 ], "overmap": "abstorefront_roof_north" }
- ]
- }
-````
+```json
+{
+ "type": "city_building",
+ "id": "abstorefront",
+ "locations": ["land"],
+ "overmaps": [
+ { "point": [0, 0, 0], "overmap": "abstorefront_north" },
+ { "point": [0, 0, 1], "overmap": "abstorefront_roof_north" }
+ ]
+}
+```
+
## Overmap Specials
-Overmap specials are handled a little differently. They use `json/overmap/specials.json` for both linking z levels and overmap spawning. A special won't need an entry in `data/json/regional_map_settings.json`
+Overmap specials are handled a little differently. They use `json/overmap/specials.json` for both
+linking z levels and overmap spawning. A special won't need an entry in
+`data/json/regional_map_settings.json`
Overmap special example:
-````json
- {
- "type" : "overmap_special",
- "id" : "Evac Shelter",
- "overmaps" : [
- { "point":[0,0,0], "overmap": "shelter"},
- { "point":[0,0,-1], "overmap": "shelter_under"},
- { "point":[0,0,1], "overmap": "shelter_roof"}
- ],
- "connections" : [
- { "point" : [0,-1,0], "terrain" : "road" }
- ],
- "locations" : [ "wilderness" ],
- "city_distance" : [5, 10],
- "city_sizes" : [4, 12],
- "occurrences" : [1, 3],
- "rotate" : false,
- "flags" : [ "CLASSIC" ]
- }
-````
+
+```json
+{
+ "type": "overmap_special",
+ "id": "Evac Shelter",
+ "overmaps": [
+ { "point": [0, 0, 0], "overmap": "shelter" },
+ { "point": [0, 0, -1], "overmap": "shelter_under" },
+ { "point": [0, 0, 1], "overmap": "shelter_roof" }
+ ],
+ "connections": [
+ { "point": [0, -1, 0], "terrain": "road" }
+ ],
+ "locations": ["wilderness"],
+ "city_distance": [5, 10],
+ "city_sizes": [4, 12],
+ "occurrences": [1, 3],
+ "rotate": false,
+ "flags": ["CLASSIC"]
+}
+```
+
## Adding the overmap_terrain entry
-Navigate to `data/json/overmap_terrain.json`
-Every z level gets an entry that defines how it appears on the overmap.
-The name field will determine what is displayed in the in-game overmap.
-The entries should share the same color and symbol.
+Navigate to `data/json/overmap_terrain.json` Every z level gets an entry that defines how it appears
+on the overmap. The name field will determine what is displayed in the in-game overmap. The entries
+should share the same color and symbol.
-````json
+```json
{
"type": "overmap_terrain",
"id": "abandonedwarehouse",
@@ -164,16 +180,20 @@ The entries should share the same color and symbol.
"sym": 119,
"color": "brown"
}
-````
+```
+
## Adding to regional_map_settings
+
Navigate to `data/json/regional_map_settings.json`
-This determines the spawn frequency and location of non-special buildings.
-Find the appropriate category for your building and add either the overmap_special ID or the city_building ID and include a spawn weight.
+This determines the spawn frequency and location of non-special buildings. Find the appropriate
+category for your building and add either the overmap_special ID or the city_building ID and include
+a spawn weight.
-````json
+```json
""abandonedwarehouse": 200,
-````
+```
+
When testing you can increase the spawn rate if you want to survey your work using natural spawns.
Finally, always [lint](http://dev.narc.ro/cataclysm/format.html) your additions before submitting.
diff --git a/doc/JSON_FLAGS.md b/doc/JSON_FLAGS.md
index 2ed65f71b5d2..719955c97bbc 100644
--- a/doc/JSON_FLAGS.md
+++ b/doc/JSON_FLAGS.md
@@ -47,7 +47,7 @@
- [Sizes](#sizes)
- [Special attacks](#special-attacks)
- [Mutations](#mutations)
- - [Flags](#flags-7)
+ - [Flags](#flags-7)
- [Categories](#categories-1)
- [Overmap](#overmap)
- [Overmap connections](#overmap-connections)
@@ -73,431 +73,489 @@
- [Vehicle parts requiring other vehicle parts](#vehicle-parts-requiring-other-vehicle-parts)
- [Fuel types](#fuel-types)
-
## Notes
-- Some flags (items, effects, vehicle parts) have to be defined in `flags.json` or `vp_flags.json` (with type: `json_flag`) to work correctly.
-- Many of the flags intended for one category or item type, can be used in other categories or item types. Experiment to see where else flags can be used.
+- Some flags (items, effects, vehicle parts) have to be defined in `flags.json` or `vp_flags.json`
+ (with type: `json_flag`) to work correctly.
+- Many of the flags intended for one category or item type, can be used in other categories or item
+ types. Experiment to see where else flags can be used.
- Offensive and defensive flags can be used on any item type that can be wielded.
-
## Inheritance
-When an item is crafted, it can inherit flags from the components that were used to craft it. This requires that the flag to be inherited has the `"craft_inherit": true` entry. If you don't want a particular item to inherit flags when crafted, specify the member delete_flags, which is an array of strings. Flags specified there will be removed from the resultant item upon crafting. This will override flag inheritance, but will not delete flags that are part of the item type itself.
-
+When an item is crafted, it can inherit flags from the components that were used to craft it. This
+requires that the flag to be inherited has the `"craft_inherit": true` entry. If you don't want a
+particular item to inherit flags when crafted, specify the member delete_flags, which is an array of
+strings. Flags specified there will be removed from the resultant item upon crafting. This will
+override flag inheritance, but will not delete flags that are part of the item type itself.
## TODO
-- Descriptions for `Special attacks` under `Monsters` could stand to be more descriptive of exactly what the attack does.
-- `Ammo effects` under `Ammo` need more descriptive details, and some need to be double-checked for accuracy.
-
+- Descriptions for `Special attacks` under `Monsters` could stand to be more descriptive of exactly
+ what the attack does.
+- `Ammo effects` under `Ammo` need more descriptive details, and some need to be double-checked for
+ accuracy.
## Ammo
### Ammo type
-These are handled through `ammo_types.json`. You can tag a weapon with these to have it chamber existing ammo, or make your own ammo there. The first column in this list is the tag's "id", the internal identifier DDA uses to track the tag, and the second is a brief description of the ammo tagged. Use the id to search for ammo listings, as ids are constant throughout DDA's code. Happy chambering! :-)
-
-- ```120mm``` 120mm HEAT
-- ```12mm``` 12mm
-- ```20x66mm``` 20x66mm Shot (and relatives)
-- ```223``` .223 Remington (and 5.56 NATO)
-- ```22``` .22LR (and relatives)
-- ```3006``` 30.06
-- ```300``` .300 WinMag
-- ```308``` .308 Winchester (and relatives)
-- ```32``` .32 ACP
-- ```36paper``` .36 cap & ball
-- ```38``` .38 Special
-- ```40``` 10mm
-- ```40mm``` 40mm Grenade
-- ```44``` .44 Magnum
-- ```44paper``` .44 cap & ball
-- ```454``` .454 Casull
-- ```45``` .45 ACP (and relatives)
-- ```46``` 46mm
-- ```500``` .500 Magnum
-- ```50``` .50 BMG
-- ```57``` 57mm
-- ```5x50``` 5x50 Dart
-- ```66mm``` 66mm HEAT
-- ```700nx``` .700 Nitro Express
-- ```762R``` 7.62x54mm
-- ```762``` 7.62x39mm
-- ```762x25``` 7.62x25mm
-- ```84x246mm``` 84x246mm HE
-- ```8x40mm``` 8mm Caseless
-- ```9mm``` 9x19mm Luger (and relatives)
-- ```9x18``` 9x18mm
-- ```BB``` BB
-- ```RPG-7``` RPG-7
-- ```UPS``` UPS charges
-- ```ammo_flintlock``` Flintlock ammo
-- ```ampoule``` Ampoule
-- ```arrow``` Arrow
-- ```battery``` Battery
-- ```blunderbuss``` Blunderbuss
-- ```bolt``` Bolt
-- ```charcoal``` Charcoal
-- ```components``` Components
-- ```dart``` Dart
-- ```diesel``` Diesel
-- ```fish_bait``` Fish bait
-- ```fishspear``` Speargun spear
-- ```fusion``` Laser Pack
-- ```gasoline``` Gasoline
-- ```homebrew_rocket``` homebrew rocket
-- ```lamp_oil``` Lamp oil
-- ```laser_capacitor``` Charge
-- ```m235``` M235 TPA (66mm Incendiary Rocket)
-- ```metal_rail``` Rebar Rail
-- ```money``` Cents
-- ```muscle``` Muscle
-- ```nail``` Nail
-- ```pebble``` Pebble
-- ```plasma``` Plasma
-- ```plutonium``` Plutonium Cell
-- ```rebreather_filter``` Rebreather filter
-- ```shot``` Shotshell
-- ```signal_flare``` Signal Flare
-- ```tape``` Duct tape
-- ```thread``` Thread
-- ```thrown``` Thrown
-- ```unfinished_char``` Semi-charred fuel
-- ```water``` Water
+These are handled through `ammo_types.json`. You can tag a weapon with these to have it chamber
+existing ammo, or make your own ammo there. The first column in this list is the tag's "id", the
+internal identifier DDA uses to track the tag, and the second is a brief description of the ammo
+tagged. Use the id to search for ammo listings, as ids are constant throughout DDA's code. Happy
+chambering! :-)
+
+- `120mm` 120mm HEAT
+- `12mm` 12mm
+- `20x66mm` 20x66mm Shot (and relatives)
+- `223` .223 Remington (and 5.56 NATO)
+- `22` .22LR (and relatives)
+- `3006` 30.06
+- `300` .300 WinMag
+- `308` .308 Winchester (and relatives)
+- `32` .32 ACP
+- `36paper` .36 cap & ball
+- `38` .38 Special
+- `40` 10mm
+- `40mm` 40mm Grenade
+- `44` .44 Magnum
+- `44paper` .44 cap & ball
+- `454` .454 Casull
+- `45` .45 ACP (and relatives)
+- `46` 46mm
+- `500` .500 Magnum
+- `50` .50 BMG
+- `57` 57mm
+- `5x50` 5x50 Dart
+- `66mm` 66mm HEAT
+- `700nx` .700 Nitro Express
+- `762R` 7.62x54mm
+- `762` 7.62x39mm
+- `762x25` 7.62x25mm
+- `84x246mm` 84x246mm HE
+- `8x40mm` 8mm Caseless
+- `9mm` 9x19mm Luger (and relatives)
+- `9x18` 9x18mm
+- `BB` BB
+- `RPG-7` RPG-7
+- `UPS` UPS charges
+- `ammo_flintlock` Flintlock ammo
+- `ampoule` Ampoule
+- `arrow` Arrow
+- `battery` Battery
+- `blunderbuss` Blunderbuss
+- `bolt` Bolt
+- `charcoal` Charcoal
+- `components` Components
+- `dart` Dart
+- `diesel` Diesel
+- `fish_bait` Fish bait
+- `fishspear` Speargun spear
+- `fusion` Laser Pack
+- `gasoline` Gasoline
+- `homebrew_rocket` homebrew rocket
+- `lamp_oil` Lamp oil
+- `laser_capacitor` Charge
+- `m235` M235 TPA (66mm Incendiary Rocket)
+- `metal_rail` Rebar Rail
+- `money` Cents
+- `muscle` Muscle
+- `nail` Nail
+- `pebble` Pebble
+- `plasma` Plasma
+- `plutonium` Plutonium Cell
+- `rebreather_filter` Rebreather filter
+- `shot` Shotshell
+- `signal_flare` Signal Flare
+- `tape` Duct tape
+- `thread` Thread
+- `thrown` Thrown
+- `unfinished_char` Semi-charred fuel
+- `water` Water
### Effects
-- ```ACIDBOMB``` Leaves a pool of acid on detonation.
-- ```BEANBAG``` Stuns the target.
-- ```BLACKPOWDER``` May clog up the gun with blackpowder fouling, which will also cause rust.
-- ```BLINDS_EYES``` Blinds the target if it hits the head (ranged projectiles can't actually hit the eyes at the moment).
-- ```BOUNCE``` Inflicts target with `bounced` effect and rebounds to a nearby target without this effect.
-- ```COOKOFF``` Explodes when lit on fire.
-- ```CUSTOM_EXPLOSION``` Explosion as specified in ```"explosion"``` field of used ammo. See `JSON_INFO.md`.
-- ```DRAW_AS_LINE``` Doesn't go through regular bullet animation, instead draws a line and the bullet on its end for one frame.
-- ```EXPLOSIVE_BIG``` Large explosion without any shrapnel.
-- ```EXPLOSIVE_HUGE``` Huge explosion without any shrapnel.
-- ```EXPLOSIVE``` Explodes without any shrapnel.
-- ```FLAME``` Very small explosion that lights fires.
-- ```FLARE``` Lights the target on fire.
-- ```FLASHBANG``` Blinds and deafens nearby targets.
-- ```FRAG``` Small explosion that spreads shrapnel.
-- ```INCENDIARY``` Lights target on fire.
-- ```LARGE_BEANBAG``` Heavily stuns the target.
-- ```LASER``` Creates a trail of laser (the field type)
-- ```LIGHTNING``` Creates a trail of lightning.
-- ```MININUKE_MOD``` Small thermo-nuclear detonation that leaves behind radioactive fallout.
-- ```MUZZLE_SMOKE``` Generate a small cloud of smoke at the source.
-- ```NAPALM``` Explosion that spreads fire.
-- ```NEVER_MISFIRES``` Firing ammo without this flag may trigger a misfiring, this is independent of the weapon flags.
-- ```NOGIB``` Prevents overkill damage on the target (target won't explode into gibs, see also the monster flag NO_GIBS).
-- ```NO_PENETRATE_OBSTACLES``` Prevents a projectile from going through a tile with obstacles, such as chainlink fences or dressers.
-- ```TANGLE``` When this projectile hits a target, it has a chance to tangle them up and immobilise them.
-- ```NO_EMBED``` When an item would be spawned from the projectile, it will always be spawned on the ground rather than in monster's inventory. Implied for active thrown items. Doesn't do anything on projectiles that do not drop items.
-- ```NO_ITEM_DAMAGE``` Will not damage items on the map even when it otherwise would try to.
-- ```PLASMA``` Creates a trail of superheated plasma.
-- ```RECOVER_[X]``` Has a (X-1/X) chance to create a single charge of the used ammo at the point of impact.
-- ```RECYCLED``` (For handmade ammo) causes the gun to misfire sometimes, this independent of the weapon flags.
-- ```SHOT``` Multiple smaller pellets; less effective against armor but increases chance to hit and no point-blank penalty
-- ```SMOKE_BIG``` Generates a large cloud of smoke at the target.
-- ```SMOKE``` Generates a cloud of smoke at the target.
-- ```STREAM_BIG``` Leaves a trail of intense fire fields.
-- ```STREAM``` Leaves a trail of fire fields.
-- ```TRAIL``` Creates a trail of smoke.
-- ```WIDE``` Prevents `HARDTOSHOOT` monster flag from having any effect. Implied by ```SHOT``` or liquid ammo.
-
+- `ACIDBOMB` Leaves a pool of acid on detonation.
+- `BEANBAG` Stuns the target.
+- `BLACKPOWDER` May clog up the gun with blackpowder fouling, which will also cause rust.
+- `BLINDS_EYES` Blinds the target if it hits the head (ranged projectiles can't actually hit the
+ eyes at the moment).
+- `BOUNCE` Inflicts target with `bounced` effect and rebounds to a nearby target without this
+ effect.
+- `COOKOFF` Explodes when lit on fire.
+- `CUSTOM_EXPLOSION` Explosion as specified in `"explosion"` field of used ammo. See `JSON_INFO.md`.
+- `DRAW_AS_LINE` Doesn't go through regular bullet animation, instead draws a line and the bullet on
+ its end for one frame.
+- `EXPLOSIVE_BIG` Large explosion without any shrapnel.
+- `EXPLOSIVE_HUGE` Huge explosion without any shrapnel.
+- `EXPLOSIVE` Explodes without any shrapnel.
+- `FLAME` Very small explosion that lights fires.
+- `FLARE` Lights the target on fire.
+- `FLASHBANG` Blinds and deafens nearby targets.
+- `FRAG` Small explosion that spreads shrapnel.
+- `INCENDIARY` Lights target on fire.
+- `LARGE_BEANBAG` Heavily stuns the target.
+- `LASER` Creates a trail of laser (the field type)
+- `LIGHTNING` Creates a trail of lightning.
+- `MININUKE_MOD` Small thermo-nuclear detonation that leaves behind radioactive fallout.
+- `MUZZLE_SMOKE` Generate a small cloud of smoke at the source.
+- `NAPALM` Explosion that spreads fire.
+- `NEVER_MISFIRES` Firing ammo without this flag may trigger a misfiring, this is independent of the
+ weapon flags.
+- `NOGIB` Prevents overkill damage on the target (target won't explode into gibs, see also the
+ monster flag NO_GIBS).
+- `NO_PENETRATE_OBSTACLES` Prevents a projectile from going through a tile with obstacles, such as
+ chainlink fences or dressers.
+- `TANGLE` When this projectile hits a target, it has a chance to tangle them up and immobilise
+ them.
+- `NO_EMBED` When an item would be spawned from the projectile, it will always be spawned on the
+ ground rather than in monster's inventory. Implied for active thrown items. Doesn't do anything on
+ projectiles that do not drop items.
+- `NO_ITEM_DAMAGE` Will not damage items on the map even when it otherwise would try to.
+- `PLASMA` Creates a trail of superheated plasma.
+- `RECOVER_[X]` Has a (X-1/X) chance to create a single charge of the used ammo at the point of
+ impact.
+- `RECYCLED` (For handmade ammo) causes the gun to misfire sometimes, this independent of the weapon
+ flags.
+- `SHOT` Multiple smaller pellets; less effective against armor but increases chance to hit and no
+ point-blank penalty
+- `SMOKE_BIG` Generates a large cloud of smoke at the target.
+- `SMOKE` Generates a cloud of smoke at the target.
+- `STREAM_BIG` Leaves a trail of intense fire fields.
+- `STREAM` Leaves a trail of fire fields.
+- `TRAIL` Creates a trail of smoke.
+- `WIDE` Prevents `HARDTOSHOOT` monster flag from having any effect. Implied by `SHOT` or liquid
+ ammo.
## Armor
### Covers
-- ```ARMS``` ... same ```ARM_L``` and ```ARM_R```
-- ```ARM_L```
-- ```ARM_R```
-- ```EYES```
-- ```FEET``` ... same ```FOOT_L``` and ```FOOT_R```
-- ```FOOT_L```
-- ```FOOT_R```
-- ```HANDS``` ... same ```HAND_L``` and ```HAND_R```
-- ```HAND_L```
-- ```HAND_R```
-- ```HEAD```
-- ```LEGS``` ... same ```LEG_L``` and ```LEG_R```
-- ```LEG_L```
-- ```LEG_R```
-- ```MOUTH```
-- ```TORSO```
+- `ARMS` ... same `ARM_L` and `ARM_R`
+- `ARM_L`
+- `ARM_R`
+- `EYES`
+- `FEET` ... same `FOOT_L` and `FOOT_R`
+- `FOOT_L`
+- `FOOT_R`
+- `HANDS` ... same `HAND_L` and `HAND_R`
+- `HAND_L`
+- `HAND_R`
+- `HEAD`
+- `LEGS` ... same `LEG_L` and `LEG_R`
+- `LEG_L`
+- `LEG_R`
+- `MOUTH`
+- `TORSO`
### Flags
-Some armor flags, such as `WATCH` and `ALARMCLOCK` are compatible with other item types. Experiment to find which flags work elsewhere.
-
-- ```ACTIVE_CLOAKING``` While active, drains UPS to provide invisibility.
-- ```ALARMCLOCK``` Has an alarm-clock feature.
-- ```ALLOWS_NATURAL_ATTACKS``` Doesn't prevent any natural attacks or similar benefits from mutations, fingertip razors, etc., like most items covering the relevant body part would.
-- ```AURA``` This item goes in the outer aura layer, intended for metaphysical effects.
-- ```BAROMETER``` This gear is equipped with an accurate barometer (which is used to measure atmospheric pressure).
-- ```BELTED``` Layer for backpacks and things worn over outerwear.
-- ```BLIND``` Blinds the wearer while worn, and provides nominal protection v. flashbang flashes.
-- ```BLOCK_WHILE_WORN``` Allows worn armor or shields to be used for blocking attacks. See also the `Techniques` section.
-- ```BULLET_IMMNUE``` Wearing an item with this flag makes you immune to bullet damage
-- ```CLIMATE_CONTROL``` This piece of clothing has climate control of some sort, keeping you warmer or cooler depending on ambient and bodily temperature.
-- ```COLLAR``` This piece of clothing has a wide collar that can keep your mouth warm.
-- ```DEAF``` Makes the player deaf.
-- ```ELECTRIC_IMMUNE``` This gear completely protects you from electric discharges.
-- ```FANCY``` Wearing this clothing gives a morale bonus if the player has the `Stylish` trait.
-- ```FIX_FARSIGHT``` This gear corrects farsightedness.
-- ```FIX_NEARSIGHT``` This gear corrects nearsightedness.
-- ```FLOTATION``` Prevents the player from drowning in deep water. Also prevents diving underwater.
-- ```FRAGILE``` This gear is less resistant to damage than normal.
-- ```HELMET_COMPAT``` Items that are not SKINTIGHT or OVERSIZE but can be worn with a helmet.
-- ```HOOD``` Allow this clothing to conditionally cover the head, for additional warmth or water protection., if the player's head isn't encumbered
-- ```HYGROMETER``` This gear is equipped with an accurate hygrometer (which is used to measure humidity).
-- ```NO_TAKEOFF``` Item with that flag can't be taken off.
-- ```NO_QUICKDRAW``` Don't offer to draw items from this holster when the fire key is pressed whilst the players hands are empty
-- ```ONLY_ONE``` You can wear only one.
-- ```OUTER``` Outer garment layer.
-- ```OVERSIZE``` Can always be worn no matter encumbrance/mutations/bionics/etc., but prevents any other clothing being worn over this.
-- ```PARTIAL_DEAF``` Reduces the volume of sounds to a safe level.
-- ```PERSONAL``` This item goes in the personal aura layer, intended for metaphysical effects.
-- ```POCKETS``` Increases warmth for hands if the player's hands are cold and the player is wielding nothing.
-- ```POWERARMOR_EXO``` Marks the item as the main exoskeleton for power armor.
-- ```POWERARMOR_EXTERNAL``` Marks the item as external pieces that cover body parts the exoskeleton doesn't.
-- ```POWERARMOR_MOD``` Marks the item as a power armour mod that is worn onto an exoskeleton/external piece.
-- ```POWERARMOR_COMPATIBLE``` Makes item compatible with power armor despite other parameters causing failure.
-- ```PSYSHIELD_PARTIAL``` 25% chance to protect against fear_paralyze monster attack when worn.
-- ```RAD_PROOF``` This piece of clothing completely protects you from radiation.
-- ```RAD_RESIST``` This piece of clothing partially protects you from radiation.
-- ```RAINPROOF``` Prevents the covered body-part(s) from getting wet in the rain.
-- ```REQUIRES_BALANCE``` Gear that requires a certain balance to be steady with. If the player is hit while wearing, they have a chance to be downed.
-- ```RESTRICT_HANDS``` Prevents the player from wielding a weapon two-handed, forcing one-handed use if the weapon permits it.
-- ```ROLLER_ONE``` A less stable and slower version of ROLLER_QUAD, still allows the player to move faster than walking speed.
-- ```ROLLER_QUAD```The medium choice between ROLLER_INLINE and ROLLER_ONE, while it is more stable, and moves faster, it also has a harsher non-flat terrain penalty then ROLLER_ONE.
-- ```ROLLER_INLINE``` Faster, but less stable overall, the penalty for non-flat terrain is even harsher.
-- ```SEMITANGIBLE``` Prevents the item from participating in the encumbrance system when worn.
-- ```COMPACT``` Prevents the item from participating in the layering encumbrance system when worn.
-- ```SKINTIGHT``` Undergarment layer.
-- ```SLOWS_MOVEMENT``` This piece of clothing multiplies move cost by 1.1.
-- ```SLOWS_THIRST``` This piece of clothing multiplies the rate at which the player grows thirsty by 0.70.
-- ```STURDY``` This clothing is a lot more resistant to damage than normal.
-- ```SUN_GLASSES``` Prevents glaring when in sunlight.
-- ```SUPER_FANCY``` Gives an additional moral bonus over `FANCY` if the player has the `Stylish` trait.
-- ```SWIM_GOGGLES``` Allows you to see much further under water.
-- ```THERMOMETER``` This gear is equipped with an accurate thermometer (which is used to measure temperature).
-- ```VARSIZE``` Can be made to fit via tailoring.
-- ```WAIST``` Layer for belts other things worn on the waist.
-- ```WATCH``` Acts as a watch and allows the player to see actual time.
-- ```WATERPROOF``` Prevents the covered body-part(s) from getting wet in any circumstance.
-- ```WATER_FRIENDLY``` Prevents the item from making the body part count as unfriendly to water and thus causing negative morale from being wet.
-
+Some armor flags, such as `WATCH` and `ALARMCLOCK` are compatible with other item types. Experiment
+to find which flags work elsewhere.
+
+- `ACTIVE_CLOAKING` While active, drains UPS to provide invisibility.
+- `ALARMCLOCK` Has an alarm-clock feature.
+- `ALLOWS_NATURAL_ATTACKS` Doesn't prevent any natural attacks or similar benefits from mutations,
+ fingertip razors, etc., like most items covering the relevant body part would.
+- `AURA` This item goes in the outer aura layer, intended for metaphysical effects.
+- `BAROMETER` This gear is equipped with an accurate barometer (which is used to measure atmospheric
+ pressure).
+- `BELTED` Layer for backpacks and things worn over outerwear.
+- `BLIND` Blinds the wearer while worn, and provides nominal protection v. flashbang flashes.
+- `BLOCK_WHILE_WORN` Allows worn armor or shields to be used for blocking attacks. See also the
+ `Techniques` section.
+- `BULLET_IMMNUE` Wearing an item with this flag makes you immune to bullet damage
+- `CLIMATE_CONTROL` This piece of clothing has climate control of some sort, keeping you warmer or
+ cooler depending on ambient and bodily temperature.
+- `COLLAR` This piece of clothing has a wide collar that can keep your mouth warm.
+- `DEAF` Makes the player deaf.
+- `ELECTRIC_IMMUNE` This gear completely protects you from electric discharges.
+- `FANCY` Wearing this clothing gives a morale bonus if the player has the `Stylish` trait.
+- `FIX_FARSIGHT` This gear corrects farsightedness.
+- `FIX_NEARSIGHT` This gear corrects nearsightedness.
+- `FLOTATION` Prevents the player from drowning in deep water. Also prevents diving underwater.
+- `FRAGILE` This gear is less resistant to damage than normal.
+- `HELMET_COMPAT` Items that are not SKINTIGHT or OVERSIZE but can be worn with a helmet.
+- `HOOD` Allow this clothing to conditionally cover the head, for additional warmth or water
+ protection., if the player's head isn't encumbered
+- `HYGROMETER` This gear is equipped with an accurate hygrometer (which is used to measure
+ humidity).
+- `NO_TAKEOFF` Item with that flag can't be taken off.
+- `NO_QUICKDRAW` Don't offer to draw items from this holster when the fire key is pressed whilst the
+ players hands are empty
+- `ONLY_ONE` You can wear only one.
+- `OUTER` Outer garment layer.
+- `OVERSIZE` Can always be worn no matter encumbrance/mutations/bionics/etc., but prevents any other
+ clothing being worn over this.
+- `PARTIAL_DEAF` Reduces the volume of sounds to a safe level.
+- `PERSONAL` This item goes in the personal aura layer, intended for metaphysical effects.
+- `POCKETS` Increases warmth for hands if the player's hands are cold and the player is wielding
+ nothing.
+- `POWERARMOR_EXO` Marks the item as the main exoskeleton for power armor.
+- `POWERARMOR_EXTERNAL` Marks the item as external pieces that cover body parts the exoskeleton
+ doesn't.
+- `POWERARMOR_MOD` Marks the item as a power armour mod that is worn onto an exoskeleton/external
+ piece.
+- `POWERARMOR_COMPATIBLE` Makes item compatible with power armor despite other parameters causing
+ failure.
+- `PSYSHIELD_PARTIAL` 25% chance to protect against fear_paralyze monster attack when worn.
+- `RAD_PROOF` This piece of clothing completely protects you from radiation.
+- `RAD_RESIST` This piece of clothing partially protects you from radiation.
+- `RAINPROOF` Prevents the covered body-part(s) from getting wet in the rain.
+- `REQUIRES_BALANCE` Gear that requires a certain balance to be steady with. If the player is hit
+ while wearing, they have a chance to be downed.
+- `RESTRICT_HANDS` Prevents the player from wielding a weapon two-handed, forcing one-handed use if
+ the weapon permits it.
+- `ROLLER_ONE` A less stable and slower version of ROLLER_QUAD, still allows the player to move
+ faster than walking speed.
+- `ROLLER_QUAD`The medium choice between ROLLER_INLINE and ROLLER_ONE, while it is more stable, and
+ moves faster, it also has a harsher non-flat terrain penalty then ROLLER_ONE.
+- `ROLLER_INLINE` Faster, but less stable overall, the penalty for non-flat terrain is even harsher.
+- `SEMITANGIBLE` Prevents the item from participating in the encumbrance system when worn.
+- `COMPACT` Prevents the item from participating in the layering encumbrance system when worn.
+- `SKINTIGHT` Undergarment layer.
+- `SLOWS_MOVEMENT` This piece of clothing multiplies move cost by 1.1.
+- `SLOWS_THIRST` This piece of clothing multiplies the rate at which the player grows thirsty by
+ 0.70.
+- `STURDY` This clothing is a lot more resistant to damage than normal.
+- `SUN_GLASSES` Prevents glaring when in sunlight.
+- `SUPER_FANCY` Gives an additional moral bonus over `FANCY` if the player has the `Stylish` trait.
+- `SWIM_GOGGLES` Allows you to see much further under water.
+- `THERMOMETER` This gear is equipped with an accurate thermometer (which is used to measure
+ temperature).
+- `VARSIZE` Can be made to fit via tailoring.
+- `WAIST` Layer for belts other things worn on the waist.
+- `WATCH` Acts as a watch and allows the player to see actual time.
+- `WATERPROOF` Prevents the covered body-part(s) from getting wet in any circumstance.
+- `WATER_FRIENDLY` Prevents the item from making the body part count as unfriendly to water and thus
+ causing negative morale from being wet.
## Bionics
-- ```BIONIC_ARMOR_INTERFACE``` This bionic can provide power to powered armor.
-- ```BIONIC_FAULTY``` This bionic is a "faulty" bionic.
-- ```BIONIC_GUN``` This bionic is a gun bionic and activating it will fire it. Prevents all other activation effects.
-- ```BIONIC_NPC_USABLE``` The NPC AI knows how to use this CBM and it can be installed on an NPC.
-- ```BIONIC_POWER_SOURCE``` This bionic is a power source bionic.
-- ```BIONIC_SLEEP_FRIENDLY``` This bionic won't prompt the user to turn it off if they try to sleep while it's active.
-- ```BIONIC_TOGGLED``` This bionic only has a function when activated, else it causes it's effect every turn.
-- ```BIONIC_WEAPON``` This bionic is a weapon bionic and activating it will create (or destroy) bionic's fake_item in user's hands. Prevents all other activation effects.
-- ```BIONIC_SHOCKPROOF``` This bionic can't be incapacitated by electrical attacks.
-
+- `BIONIC_ARMOR_INTERFACE` This bionic can provide power to powered armor.
+- `BIONIC_FAULTY` This bionic is a "faulty" bionic.
+- `BIONIC_GUN` This bionic is a gun bionic and activating it will fire it. Prevents all other
+ activation effects.
+- `BIONIC_NPC_USABLE` The NPC AI knows how to use this CBM and it can be installed on an NPC.
+- `BIONIC_POWER_SOURCE` This bionic is a power source bionic.
+- `BIONIC_SLEEP_FRIENDLY` This bionic won't prompt the user to turn it off if they try to sleep
+ while it's active.
+- `BIONIC_TOGGLED` This bionic only has a function when activated, else it causes it's effect every
+ turn.
+- `BIONIC_WEAPON` This bionic is a weapon bionic and activating it will create (or destroy) bionic's
+ fake_item in user's hands. Prevents all other activation effects.
+- `BIONIC_SHOCKPROOF` This bionic can't be incapacitated by electrical attacks.
## Books
-- ```INSPIRATIONAL``` Reading this book grants bonus morale to characters with the SPIRITUAL trait.
+- `INSPIRATIONAL` Reading this book grants bonus morale to characters with the SPIRITUAL trait.
### Use actions
-- ```ACIDBOMB_ACT``` Get rid of it or you'll end up like that guy in Robocop.
-- ```ACIDBOMB``` Pull the pin on an acid bomb.
-- ```AUTOCLAVE``` Sterilize one CBM by autoclaving it.
-- ```ARROW_FLAMABLE``` Light your arrow and let fly.
-- ```BELL``` Ring the bell.
-- ```BOLTCUTTERS``` Use your town key to gain access anywhere.
-- ```BREAK_STICK``` Breaks long stick into two.
-- ```C4``` Arm the C4.
-- ```CABLE_ATTACH``` This item is a cable spool. Use it to try to attach to a vehicle.
-- ```CAN_GOO``` Release a little blob buddy.
-- ```CAPTURE_MONSTER_ACT``` Capture and encapsulate a monster. The associated action is also used for releasing it.
-- ```CARVER_OFF``` Turn the carver on.
-- ```CARVER_ON``` Turn the carver off.
-- ```CHAINSAW_OFF``` Turn the chainsaw on.
-- ```CHAINSAW_ON``` Turn the chainsaw off.
-- ```COMBATSAW_OFF``` Turn the combat-saw on.
-- ```COMBATSAW_ON``` Turn the combat-saw off
-- ```CROWBAR``` Pry open doors, windows, man-hole covers and many other things that need prying.
-- ```DIG``` Clear rubble.
-- ```DIRECTIONAL_ANTENNA``` Find the source of a signal with your radio.
-- ```DIVE_TANK``` Use compressed air tank to breathe.
-- ```DOG_WHISTLE``` Dogs hate this thing; your dog seems pretty cool with it though.
-- ```DOLLCHAT``` That creepy doll just keeps on talking.
-- ```ELEC_CHAINSAW_OFF``` Turn the electric chainsaw on.
-- ```ELEC_CHAINSAW_ON``` Turn the electric chainsaw off.
-- ```EXTINGUISHER``` Put out fires.
-- ```FIRECRACKER_ACT``` The saddest Fourth of July.
-- ```FIRECRACKER_PACK_ACT``` Keep the change you filthy animal.
-- ```FIRECRACKER_PACK``` Light an entire packet of firecrackers.
-- ```FIRECRACKER``` Light a singular firecracker.
-- ```FLASHBANG``` Pull the pin on a flashbang.
-- ```GEIGER``` Detect local radiation levels.
-- ```GRANADE_ACT``` Assaults enemies with source code fixes?
-- ```GRANADE``` Pull the pin on Granade.
-- ```GRENADE``` Pull the pin on a grenade.
-- ```HACKSAW``` Cut metal into chunks.
-- ```HAMMER``` Pry boards off of windows, doors and fences.
-- ```HEATPACK``` Activate the heatpack and get warm.
-- ```HEAT_FOOD``` Heat food around fires.
-- ```HOTPLATE``` Use the hotplate.
-- ```JACKHAMMER``` Bust down walls and other constructions.
-- ```JET_INJECTOR``` Inject some jet drugs right into your veins.
-- ```LAW``` Unpack the LAW for firing.
-- ```LIGHTSTRIP``` Activates the lightstrip.
-- ```LUMBER``` Cut logs into planks.
-- ```MAKEMOUND``` Make a mound of dirt.
-- ```MANHACK``` Activate a manhack.
-- ```MATCHBOMB``` Light the matchbomb.
-- ```MILITARYMAP``` Learn of local military installations, and show roads.
-- ```MININUKE``` Set the timer and run. Or hit with a hammer (not really).
-- ```MOLOTOV_LIT``` Throw it, but don't drop it.
-- ```MOLOTOV``` Light the Molotov cocktail.
-- ```MOP``` Mop up the mess.
-- ```MP3_ON``` Turn the mp3 player off.
-- ```MP3``` Turn the mp3 player on.
-- ```NOISE_EMITTER_OFF``` Turn the noise emitter on.
-- ```NOISE_EMITTER_ON``` Turn the noise emitter off.
-- ```NONE``` Do nothing.
-- ```PACK_CBM``` Put CBM in special autoclave pouch so that they stay sterile once sterilized.
-- ```PHEROMONE``` Makes zombies ignore you.
-- ```PICKAXE``` Does nothing but berate you for having it (I'm serious).
-- ```PLACE_RANDOMLY``` This is very much like the flag in the manhack iuse, it prevents the item from querying the player as to where they want the monster unloaded to, and instead choses randomly.
-- ```PORTABLE_GAME``` Play games.
-- ```PORTAL``` Create portal traps.
-- ```RADIO_OFF``` Turn the radio on.
-- ```RADIO_ON``` Turn the radio off.
-- ```RAG``` Stop the bleeding.
-- ```RESTAURANTMAP``` Learn of local eateries, and show roads.
-- ```ROADMAP``` Learn of local common points-of-interest and show roads.
-- ```SCISSORS``` Cut up clothing.
-- ```SEED``` Asks if you are sure that you want to eat the seed. As it is better to plant seeds.
-- ```SEW``` Sew clothing.
-- ```SHELTER``` Put up a full-blown shelter.
-- ```SHOCKTONFA_OFF``` Turn the shocktonfa on.
-- ```SHOCKTONFA_ON``` Turn the shocktonfa off.
-- ```SIPHON``` Siphon liquids out of vehicle.
-- ```SMOKEBOMB_ACT``` This may be a good way to hide as a smoker.
-- ```SMOKEBOMB``` Pull the pin on a smoke bomb.
-- ```SOLARPACK_OFF``` Fold solar backpack array.
-- ```SOLARPACK``` Unfold solar backpack array.
-- ```SOLDER_WELD``` Solder or weld items, or cauterize wounds.
-- ```SPRAY_CAN``` Graffiti the town.
-- ```SURVIVORMAP``` Learn of local points-of-interest that can help you survive, and show roads.
-- ```TAZER``` Shock someone or something.
-- ```TELEPORT``` Teleport.
-- ```TORCH``` Light a torch.
-- ```TOURISTMAP``` Learn of local points-of-interest that a tourist would like to visit, and show roads.
-- ```TOWEL``` Dry your character using the item as towel.
-- ```TOW_ATTACH``` This is a tow cable, activate it to attach it to a vehicle.
-- ```TURRET``` Activate a turret.
-- ```WASH_ALL_ITEMS``` Wash items with FILTHY flag.
-- ```WASH_HARD_ITEMS``` Wash hard items with FILTHY flag.
-- ```WASH_SOFT_ITEMS``` Wash soft items with FILTHY flag.
-- ```WATER_PURIFIER``` Purify water.
-
+- `ACIDBOMB_ACT` Get rid of it or you'll end up like that guy in Robocop.
+- `ACIDBOMB` Pull the pin on an acid bomb.
+- `AUTOCLAVE` Sterilize one CBM by autoclaving it.
+- `ARROW_FLAMABLE` Light your arrow and let fly.
+- `BELL` Ring the bell.
+- `BOLTCUTTERS` Use your town key to gain access anywhere.
+- `BREAK_STICK` Breaks long stick into two.
+- `C4` Arm the C4.
+- `CABLE_ATTACH` This item is a cable spool. Use it to try to attach to a vehicle.
+- `CAN_GOO` Release a little blob buddy.
+- `CAPTURE_MONSTER_ACT` Capture and encapsulate a monster. The associated action is also used for
+ releasing it.
+- `CARVER_OFF` Turn the carver on.
+- `CARVER_ON` Turn the carver off.
+- `CHAINSAW_OFF` Turn the chainsaw on.
+- `CHAINSAW_ON` Turn the chainsaw off.
+- `COMBATSAW_OFF` Turn the combat-saw on.
+- `COMBATSAW_ON` Turn the combat-saw off
+- `CROWBAR` Pry open doors, windows, man-hole covers and many other things that need prying.
+- `DIG` Clear rubble.
+- `DIRECTIONAL_ANTENNA` Find the source of a signal with your radio.
+- `DIVE_TANK` Use compressed air tank to breathe.
+- `DOG_WHISTLE` Dogs hate this thing; your dog seems pretty cool with it though.
+- `DOLLCHAT` That creepy doll just keeps on talking.
+- `ELEC_CHAINSAW_OFF` Turn the electric chainsaw on.
+- `ELEC_CHAINSAW_ON` Turn the electric chainsaw off.
+- `EXTINGUISHER` Put out fires.
+- `FIRECRACKER_ACT` The saddest Fourth of July.
+- `FIRECRACKER_PACK_ACT` Keep the change you filthy animal.
+- `FIRECRACKER_PACK` Light an entire packet of firecrackers.
+- `FIRECRACKER` Light a singular firecracker.
+- `FLASHBANG` Pull the pin on a flashbang.
+- `GEIGER` Detect local radiation levels.
+- `GRANADE_ACT` Assaults enemies with source code fixes?
+- `GRANADE` Pull the pin on Granade.
+- `GRENADE` Pull the pin on a grenade.
+- `HACKSAW` Cut metal into chunks.
+- `HAMMER` Pry boards off of windows, doors and fences.
+- `HEATPACK` Activate the heatpack and get warm.
+- `HEAT_FOOD` Heat food around fires.
+- `HOTPLATE` Use the hotplate.
+- `JACKHAMMER` Bust down walls and other constructions.
+- `JET_INJECTOR` Inject some jet drugs right into your veins.
+- `LAW` Unpack the LAW for firing.
+- `LIGHTSTRIP` Activates the lightstrip.
+- `LUMBER` Cut logs into planks.
+- `MAKEMOUND` Make a mound of dirt.
+- `MANHACK` Activate a manhack.
+- `MATCHBOMB` Light the matchbomb.
+- `MILITARYMAP` Learn of local military installations, and show roads.
+- `MININUKE` Set the timer and run. Or hit with a hammer (not really).
+- `MOLOTOV_LIT` Throw it, but don't drop it.
+- `MOLOTOV` Light the Molotov cocktail.
+- `MOP` Mop up the mess.
+- `MP3_ON` Turn the mp3 player off.
+- `MP3` Turn the mp3 player on.
+- `NOISE_EMITTER_OFF` Turn the noise emitter on.
+- `NOISE_EMITTER_ON` Turn the noise emitter off.
+- `NONE` Do nothing.
+- `PACK_CBM` Put CBM in special autoclave pouch so that they stay sterile once sterilized.
+- `PHEROMONE` Makes zombies ignore you.
+- `PICKAXE` Does nothing but berate you for having it (I'm serious).
+- `PLACE_RANDOMLY` This is very much like the flag in the manhack iuse, it prevents the item from
+ querying the player as to where they want the monster unloaded to, and instead choses randomly.
+- `PORTABLE_GAME` Play games.
+- `PORTAL` Create portal traps.
+- `RADIO_OFF` Turn the radio on.
+- `RADIO_ON` Turn the radio off.
+- `RAG` Stop the bleeding.
+- `RESTAURANTMAP` Learn of local eateries, and show roads.
+- `ROADMAP` Learn of local common points-of-interest and show roads.
+- `SCISSORS` Cut up clothing.
+- `SEED` Asks if you are sure that you want to eat the seed. As it is better to plant seeds.
+- `SEW` Sew clothing.
+- `SHELTER` Put up a full-blown shelter.
+- `SHOCKTONFA_OFF` Turn the shocktonfa on.
+- `SHOCKTONFA_ON` Turn the shocktonfa off.
+- `SIPHON` Siphon liquids out of vehicle.
+- `SMOKEBOMB_ACT` This may be a good way to hide as a smoker.
+- `SMOKEBOMB` Pull the pin on a smoke bomb.
+- `SOLARPACK_OFF` Fold solar backpack array.
+- `SOLARPACK` Unfold solar backpack array.
+- `SOLDER_WELD` Solder or weld items, or cauterize wounds.
+- `SPRAY_CAN` Graffiti the town.
+- `SURVIVORMAP` Learn of local points-of-interest that can help you survive, and show roads.
+- `TAZER` Shock someone or something.
+- `TELEPORT` Teleport.
+- `TORCH` Light a torch.
+- `TOURISTMAP` Learn of local points-of-interest that a tourist would like to visit, and show roads.
+- `TOWEL` Dry your character using the item as towel.
+- `TOW_ATTACH` This is a tow cable, activate it to attach it to a vehicle.
+- `TURRET` Activate a turret.
+- `WASH_ALL_ITEMS` Wash items with FILTHY flag.
+- `WASH_HARD_ITEMS` Wash hard items with FILTHY flag.
+- `WASH_SOFT_ITEMS` Wash soft items with FILTHY flag.
+- `WATER_PURIFIER` Purify water.
## Comestibles
### Comestible type
-- ```DRINK```
-- ```FOOD```
-- ```MED```
+- `DRINK`
+- `FOOD`
+- `MED`
### Addiction type
-- ```alcohol```
-- ```amphetamine```
-- ```caffeine```
-- ```cocaine```
-- ```crack```
-- ```nicotine```
-- ```opiate```
-- ```sleeping pill```
+- `alcohol`
+- `amphetamine`
+- `caffeine`
+- `cocaine`
+- `crack`
+- `nicotine`
+- `opiate`
+- `sleeping pill`
### Use action
-- ```ALCOHOL_STRONG``` Greatly increases drunkenness. Adds disease `drunk`.
-- ```ALCOHOL_WEAK``` Slightly increases drunkenness. Adds disease `drunk`
-- ```ALCOHOL``` Increases drunkenness. Adds disease `drunk`.
-- ```ANTIBIOTIC``` Helps fight infections. Removes disease `infected` and adds disease `recover`.
-- ```BANDAGE``` Stop bleeding.
-- ```BIRDFOOD``` Makes a small bird friendly.
-- ```BLECH``` Causes vomiting.
-- ```BLECH_BECAUSE_UNCLEAN``` Causes warning.
-- ```CATFOOD``` Makes a cat friendly.
-- ```CATTLEFODDER``` Makes a large herbivore friendly.
-- ```CHEW``` Displays message "You chew your %s", but otherwise does nothing.
-- ```CIG``` Alleviates nicotine cravings. Adds disease `cig`.
-- ```COKE``` Decreases hunger. Adds disease `high`.
-- ```CRACK``` Decreases hunger. Adds disease `high`.
-- ```DISINFECTANT``` Prevents infections.
-- ```DOGFOOD``` Makes a dog friendly.
-- ```FIRSTAID``` Heal.
-- ```FLUMED``` Adds disease `took_flumed`.
-- ```FLUSLEEP``` Adds disease `took_flumed` and increases fatigue.
-- ```FUNGICIDE``` Kills fungus and spores. Removes diseases `fungus` and `spores`.
-- ```HALLU``` Adds disease `hallu`.
-- ```HONEYCOMB``` Spawns wax.
-- ```INHALER``` Removes disease `asthma`.
-- ```IODINE``` Adds disease `iodine`.
-- ```MARLOSS``` "As you eat the berry, you have a near-religious experience, feeling at one with your surroundings..."
-- ```METH``` Adds disease `meth`
-- ```NONE``` "You can't do anything of interest with your [x]."
-- ```PKILL``` Reduces pain. Adds disease `pkill[n]` where `[n]` is the level of flag `PKILL_[n]` used on this comestible.
-- ```PLANTBLECH``` Causes vomiting if player does not contain plant mutations
-- ```POISON``` Adds diseases `poison` and `foodpoison`.
-- ```PROZAC``` Adds disease `took_prozac` if not currently present, otherwise acts as a minor stimulant. Rarely has the `took_prozac_bad` adverse effect.
-- ```PURIFIER``` Removes negative mutations.
-- ```ROYAL_JELLY``` Alleviates many negative conditions and diseases.
-- ```SEWAGE``` Causes vomiting and a chance to mutate.
-- ```SLEEP``` Greatly increases fatigue.
-- ```THORAZINE``` Removes diseases `hallu`, `visuals`, `high`. Additionally removes disease `formication` if disease `dermatik` isn't also present. Has a chance of a negative reaction which increases fatigue.
-- ```VACCINE``` Greatly increases health.
-- ```VITAMINS``` Increases healthiness (not to be confused with HP)
-- ```WEED``` Makes you roll with Cheech & Chong. Adds disease `weed_high`.
-- ```XANAX``` Alleviates anxiety. Adds disease `took_xanax`.
+- `ALCOHOL_STRONG` Greatly increases drunkenness. Adds disease `drunk`.
+- `ALCOHOL_WEAK` Slightly increases drunkenness. Adds disease `drunk`
+- `ALCOHOL` Increases drunkenness. Adds disease `drunk`.
+- `ANTIBIOTIC` Helps fight infections. Removes disease `infected` and adds disease `recover`.
+- `BANDAGE` Stop bleeding.
+- `BIRDFOOD` Makes a small bird friendly.
+- `BLECH` Causes vomiting.
+- `BLECH_BECAUSE_UNCLEAN` Causes warning.
+- `CATFOOD` Makes a cat friendly.
+- `CATTLEFODDER` Makes a large herbivore friendly.
+- `CHEW` Displays message "You chew your %s", but otherwise does nothing.
+- `CIG` Alleviates nicotine cravings. Adds disease `cig`.
+- `COKE` Decreases hunger. Adds disease `high`.
+- `CRACK` Decreases hunger. Adds disease `high`.
+- `DISINFECTANT` Prevents infections.
+- `DOGFOOD` Makes a dog friendly.
+- `FIRSTAID` Heal.
+- `FLUMED` Adds disease `took_flumed`.
+- `FLUSLEEP` Adds disease `took_flumed` and increases fatigue.
+- `FUNGICIDE` Kills fungus and spores. Removes diseases `fungus` and `spores`.
+- `HALLU` Adds disease `hallu`.
+- `HONEYCOMB` Spawns wax.
+- `INHALER` Removes disease `asthma`.
+- `IODINE` Adds disease `iodine`.
+- `MARLOSS` "As you eat the berry, you have a near-religious experience, feeling at one with your
+ surroundings..."
+- `METH` Adds disease `meth`
+- `NONE` "You can't do anything of interest with your [x]."
+- `PKILL` Reduces pain. Adds disease `pkill[n]` where `[n]` is the level of flag `PKILL_[n]` used on
+ this comestible.
+- `PLANTBLECH` Causes vomiting if player does not contain plant mutations
+- `POISON` Adds diseases `poison` and `foodpoison`.
+- `PROZAC` Adds disease `took_prozac` if not currently present, otherwise acts as a minor stimulant.
+ Rarely has the `took_prozac_bad` adverse effect.
+- `PURIFIER` Removes negative mutations.
+- `ROYAL_JELLY` Alleviates many negative conditions and diseases.
+- `SEWAGE` Causes vomiting and a chance to mutate.
+- `SLEEP` Greatly increases fatigue.
+- `THORAZINE` Removes diseases `hallu`, `visuals`, `high`. Additionally removes disease
+ `formication` if disease `dermatik` isn't also present. Has a chance of a negative reaction which
+ increases fatigue.
+- `VACCINE` Greatly increases health.
+- `VITAMINS` Increases healthiness (not to be confused with HP)
+- `WEED` Makes you roll with Cheech & Chong. Adds disease `weed_high`.
+- `XANAX` Alleviates anxiety. Adds disease `took_xanax`.
### Flags
-- ```ACID``` when consumed using the BLECH function, penalties are reduced if acidproof.
-- ```CARNIVORE_OK``` Can be eaten by characters with the Carnivore mutation.
-- ```CANT_HEAL_EVERYONE``` This med can't be used by everyone, it require a special mutation. See `can_heal_with` in mutation.
-- ```EATEN_COLD``` Morale bonus for eating cold.
-- ```EATEN_HOT``` Morale bonus for eating hot.
-- ```INEDIBLE``` Inedible by default, enabled to eat when in conjunction with (mutation threshold) flags: BIRD, CATTLE.
-- ```FERTILIZER``` Works as fertilizer for farming, of if this consumed with the PLANTBLECH function penalties will be reversed for plants.
-- ```FREEZERBURN``` First thaw is MUSHY, second is rotten
-- ```FUNGAL_VECTOR``` Will give a fungal infection when consumed.
-- ```HIDDEN_HALLU``` ... Food causes hallucinations, visible only with a certain survival skill level.
-- ```HIDDEN_POISON``` ... Food displays as poisonous with a certain survival skill level. Note that this doesn't make items poisonous on its own, consider adding `"use_action": "POISON"` as well, or using `FORAGE_POISON` instead.
-- ```MELTS``` Provides half fun unless frozen. Edible when frozen.
-- ```MILLABLE``` Can be placed inside a mill, to turn into flour.
-- ```MYCUS_OK``` Can be eaten by post-threshold Mycus characters. Only applies to mycus fruits by default.
-- ```NEGATIVE_MONOTONY_OK``` Allows ```negative_monotony``` property to lower comestible fun to negative values.
-- ```NO_INGEST``` Administered by some means other than oral intake.
-- ```PKILL_1``` Minor painkiller.
-- ```PKILL_2``` Moderate painkiller.
-- ```PKILL_3``` Heavy painkiller.
-- ```PKILL_4``` "You shoot up."
-- ```PKILL_L``` Slow-release painkiller.
-- ```RAW``` Reduces kcal by 25%, until cooked (that is, used in a recipe that requires a heat source). Should be added to *all* uncooked food, unless that food derives more than 50% of its calories from sugars (i.e. many fruits, some veggies) or fats (i.e. butchered fat, coconut). TODO: Make a unit test for these criteria after fat/protein/carbs are added.
-- ```SMOKABLE``` Accepted by smoking rack.
-- ```SMOKED``` Not accepted by smoking rack (product of smoking).
-- ```USE_EAT_VERB``` "You drink your %s." or "You eat your %s."
-- ```USE_ON_NPC``` Can be used on NPCs (not necessarily by them).
-- ```ZOOM``` Zoom items can increase your overmap sight range.
+- `ACID` when consumed using the BLECH function, penalties are reduced if acidproof.
+- `CARNIVORE_OK` Can be eaten by characters with the Carnivore mutation.
+- `CANT_HEAL_EVERYONE` This med can't be used by everyone, it require a special mutation. See
+ `can_heal_with` in mutation.
+- `EATEN_COLD` Morale bonus for eating cold.
+- `EATEN_HOT` Morale bonus for eating hot.
+- `INEDIBLE` Inedible by default, enabled to eat when in conjunction with (mutation threshold)
+ flags: BIRD, CATTLE.
+- `FERTILIZER` Works as fertilizer for farming, of if this consumed with the PLANTBLECH function
+ penalties will be reversed for plants.
+- `FREEZERBURN` First thaw is MUSHY, second is rotten
+- `FUNGAL_VECTOR` Will give a fungal infection when consumed.
+- `HIDDEN_HALLU` ... Food causes hallucinations, visible only with a certain survival skill level.
+- `HIDDEN_POISON` ... Food displays as poisonous with a certain survival skill level. Note that this
+ doesn't make items poisonous on its own, consider adding `"use_action": "POISON"` as well, or
+ using `FORAGE_POISON` instead.
+- `MELTS` Provides half fun unless frozen. Edible when frozen.
+- `MILLABLE` Can be placed inside a mill, to turn into flour.
+- `MYCUS_OK` Can be eaten by post-threshold Mycus characters. Only applies to mycus fruits by
+ default.
+- `NEGATIVE_MONOTONY_OK` Allows `negative_monotony` property to lower comestible fun to negative
+ values.
+- `NO_INGEST` Administered by some means other than oral intake.
+- `PKILL_1` Minor painkiller.
+- `PKILL_2` Moderate painkiller.
+- `PKILL_3` Heavy painkiller.
+- `PKILL_4` "You shoot up."
+- `PKILL_L` Slow-release painkiller.
+- `RAW` Reduces kcal by 25%, until cooked (that is, used in a recipe that requires a heat source).
+ Should be added to _all_ uncooked food, unless that food derives more than 50% of its calories
+ from sugars (i.e. many fruits, some veggies) or fats (i.e. butchered fat, coconut). TODO: Make a
+ unit test for these criteria after fat/protein/carbs are added.
+- `SMOKABLE` Accepted by smoking rack.
+- `SMOKED` Not accepted by smoking rack (product of smoking).
+- `USE_EAT_VERB` "You drink your %s." or "You eat your %s."
+- `USE_ON_NPC` Can be used on NPCs (not necessarily by them).
+- `ZOOM` Zoom items can increase your overmap sight range.
## Furniture and Terrain
@@ -505,325 +563,388 @@ List of known flags, used in both `terrain.json` and `furniture.json`.
### Flags
-- ```ALARMED``` Sets off an alarm if smashed.
-- ```ALLOW_FIELD_EFFECT``` Apply field effects to items inside ```SEALED``` terrain/furniture.
-- ```AUTO_WALL_SYMBOL``` (only for terrain) The symbol of this terrain will be one of the line drawings (corner, T-intersection, straight line etc.) depending on the adjacent terrains.
-
- Example: `-` and `|` are both terrain with the `CONNECT_TO_WALL` flag. `O` does not have the flag, while `X` and `Y` have the `AUTO_WALL_SYMBOL` flag.
-
- `X` terrain will be drawn as a T-intersection (connected to west, south and east), `Y` will be drawn as horizontal line (going from west to east, no connection to south).
- ```
- -X- -Y-
- | O
- ```
-
-- ```BARRICADABLE_DOOR_DAMAGED```
-- ```BARRICADABLE_DOOR_REINFORCED_DAMAGED```
-- ```BARRICADABLE_DOOR_REINFORCED```
-- ```BARRICADABLE_DOOR``` Door that can be barricaded.
-- ```BARRICADABLE_WINDOW_CURTAINS```
-- ```BARRICADABLE_WINDOW``` Window that can be barricaded.
-- ```BASHABLE``` Players + Monsters can bash this.
-- ```BLOCK_WIND``` This terrain will block the effects of wind.
-- ```BURROWABLE``` Burrowing monsters can travel under this terrain, while most others can't (e.g. graboid will traverse under the chain link fence, while ordinary zombie will be stopped by it).
-- ```BUTCHER_EQ``` Butcher's equipment - required for full butchery of corpses.
-- ```CAN_SIT``` Furniture the player can sit on. Player sitting near furniture with the "FLAT_SURF" tag will get mood bonus for eating.
-- ```CHIP``` Used in construction menu to determine if wall can have paint chipped off.
-- ```COLLAPSES``` Has a roof that can collapse.
-- ```CONNECT_TO_WALL``` (only for terrain) This flag has been superseded by the JSON entry `connects_to`, but is retained for backward compatibility.
-- ```CONSOLE``` Used as a computer.
-- ```CONTAINER``` Items on this square are hidden until looted by the player.
-- ```DECONSTRUCT``` Can be deconstructed.
-- ```DEEP_WATER```
-- ```DESTROY_ITEM``` Items that land here are destroyed. See also `NOITEM`
-- ```DIFFICULT_Z``` Most zombies will not be able to follow you up this terrain ( i.e a ladder )
-- ```DIGGABLE_CAN_DEEPEN``` Diggable location can be dug again to make deeper (e.g. shallow pit to deep pit).
-- ```DIGGABLE``` Digging monsters, seeding monster, digging with shovel, etc.
-- ```DOOR``` Can be opened (used for NPC path-finding).
-- ```EASY_DECONSTRUCT``` Player can deconstruct this without tools.
-- ```ELEVATOR``` Terrain with this flag will move player, NPCs, monsters, and items up and down when player activates nearby `elevator controls`.
-- ```EMITTER``` This furniture will emit fields automatically as defined by its emissions entry
-- ```FIRE_CONTAINER``` Stops fire from spreading (brazier, wood stove, etc.)
-- ```FLAMMABLE_ASH``` Burns to ash rather than rubble.
-- ```FLAMMABLE_HARD``` Harder to light on fire, but still possible.
-- ```FLAMMABLE``` Can be lit on fire.
-- ```FLAT_SURF``` Furniture or terrain with a flat hard surface (e.g. table, but not chair; tree stump, etc.).
-- ```FLAT``` Player can build and move furniture on.
-- ```FORAGE_HALLU``` This item can be found with the `HIDDEN_HALLU` flag when found through foraging.
-- ```FORAGE_POISION``` This item can be found with the `HIDDEN_POISON` flag when found through foraging.
-- ```FRIDGE``` This item refrigerates items inside. Preserving them. Unlike vehicle parts, any furniture with this flag constantly functions.
-- ```FREEZER``` This item freezes items inside. Preserving them extremely well. Unlike vehicle parts, any furniture with this flag constantly functions.
-- ```GOES_DOWN``` Can use > to go down a level.
-- ```GOES_UP``` Can use < to go up a level.
-- ```GROWTH_SEED``` This plant is in its seed stage of growth.
-- ```GROWTH_SEEDLING``` This plant is in its seedling stage of growth.
-- ```GROWTH_MATURE``` This plant is in a mature stage of a growth.
-- ```GROWTH_HARVEST``` This plant is ready for harvest.
-- ```HARVESTED``` Marks the harvested version of a terrain type (e.g. harvesting an apple tree turns it into a harvested tree, which later becomes an apple tree again).
-- ```HIDE_PLACE``` Creatures on this tile can't be seen by creatures not standing on adjacent tiles
-- ```INDOORS``` Has a roof over it; blocks rain, sunlight, etc.
-- ```LADDER``` This piece of furniture that makes climbing easy (works only with z-level mode).
-- ```LIQUIDCONT``` Furniture that contains liquid, allows for contents to be accessed in some checks even if `SEALED`.
-- ```LIQUID``` Blocks movement, but isn't a wall (lava, water, etc.)
-- ```MINEABLE``` Can be mined with a pickaxe/jackhammer.
-- ```MOUNTABLE``` Suitable for guns with the `MOUNTED_GUN` flag.
-- ```NOCOLLIDE``` Feature that simply doesn't collide with vehicles at all.
-- ```NOITEM``` Items cannot be added here but may overflow to adjacent tiles. See also `DESTROY_ITEM`
-- ```NO_FLOOR``` Things should fall when placed on this tile
-- ```NO_SIGHT``` Creature on this tile have their sight reduced to one tile
-- ```NO_SCENT``` This tile cannot have scent values, which prevents scent diffusion through this tile
-- ```OPENCLOSE_INSIDE``` If it's a door (with an 'open' or 'close' field), it can only be opened or closed if you're inside.
-- ```PAINFUL``` May cause a small amount of pain.
-- ```PERMEABLE``` Permeable for gases.
-- ```PLACE_ITEM``` Valid terrain for `place_item()` to put items on.
-- ```PLANT``` A 'furniture' that grows and fruits.
-- ```PLANTABLE``` This terrain or furniture can have seeds planted in it.
-- ```PLOWABLE``` Terrain can be plowed.
-- ```RAMP_END```
-- ```RAMP``` Can be used to move up a z-level
-- ```REDUCE_SCENT``` Reduces scent diffusion (not total amount of scent in area); only works if also bashable.
-- ```ROAD``` Flat and hard enough to drive or skate (with rollerblades) on.
-- ```ROUGH``` May hurt the player's feet.
-- ```RUG``` Enables the `Remove Carpet` Construction entry.
-- ```SALT_WATER``` Source of salt water (works for terrains with examine action "water_source").
-- ```SEALED``` Can't use e to retrieve items; must smash them open first.
-- ```SEEN_FROM_ABOVE``` Visible from a higher level (provided the tile above has no floor)
-- ```SHARP``` May do minor damage to players/monsters passing through it.
-- ```SHORT``` Feature too short to collide with vehicle protrusions. (mirrors, blades).
-- ```SIGN``` Show written message on examine.
-- ```SMALL_PASSAGE``` This terrain or furniture is too small for large or huge creatures to pass through.
-- ```SUN_ROOF_ABOVE``` This furniture (terrain is not supported currently) has a "fake roof" above, that blocks sunlight. Special hack for #44421, to be removed later.
-- ```SUPPORTS_ROOF``` Used as a boundary for roof construction.
-- ```SUPPRESS_SMOKE``` Prevents smoke from fires; used by ventilated wood stoves, etc.
-- ```SWIMMABLE``` Player and monsters can swim through it.
-- ```THIN_OBSTACLE``` Passable by players and monsters; vehicles destroy it.
-- ```TINY``` Feature too short to collide with vehicle undercarriage. Vehicles drive over them with no damage, unless a wheel hits them.
-- ```TRANSPARENT``` Players and monsters can see through/past it. Also sets ter_t.transparent.
-- ```UNSTABLE``` Walking here cause the bouldering effect on the character.
-- ```USABLE_FIRE``` This terrain or furniture counts as a nearby fire for crafting.
-- ```VEH_TREAT_AS_BASH_BELOW``` Vehicles will not collide with this even if it counts as rough terrain, like floor with bash_below does. Used for terrain meant to be turned into other terrain when smashed instead of destroying the tile beneath it.
-- ```WALL``` This terrain is an upright obstacle. Used for fungal conversion, and also implies `CONNECT_TO_WALL`.
+- `ALARMED` Sets off an alarm if smashed.
+- `ALLOW_FIELD_EFFECT` Apply field effects to items inside `SEALED` terrain/furniture.
+- `AUTO_WALL_SYMBOL` (only for terrain) The symbol of this terrain will be one of the line drawings
+ (corner, T-intersection, straight line etc.) depending on the adjacent terrains.
+
+ Example: `-` and `|` are both terrain with the `CONNECT_TO_WALL` flag. `O` does not have the flag,
+ while `X` and `Y` have the `AUTO_WALL_SYMBOL` flag.
+
+ `X` terrain will be drawn as a T-intersection (connected to west, south and east), `Y` will be
+ drawn as horizontal line (going from west to east, no connection to south).
+ ```
+ -X- -Y-
+ | O
+ ```
+
+- `BARRICADABLE_DOOR_DAMAGED`
+- `BARRICADABLE_DOOR_REINFORCED_DAMAGED`
+- `BARRICADABLE_DOOR_REINFORCED`
+- `BARRICADABLE_DOOR` Door that can be barricaded.
+- `BARRICADABLE_WINDOW_CURTAINS`
+- `BARRICADABLE_WINDOW` Window that can be barricaded.
+- `BASHABLE` Players + Monsters can bash this.
+- `BLOCK_WIND` This terrain will block the effects of wind.
+- `BURROWABLE` Burrowing monsters can travel under this terrain, while most others can't (e.g.
+ graboid will traverse under the chain link fence, while ordinary zombie will be stopped by it).
+- `BUTCHER_EQ` Butcher's equipment - required for full butchery of corpses.
+- `CAN_SIT` Furniture the player can sit on. Player sitting near furniture with the "FLAT_SURF" tag
+ will get mood bonus for eating.
+- `CHIP` Used in construction menu to determine if wall can have paint chipped off.
+- `COLLAPSES` Has a roof that can collapse.
+- `CONNECT_TO_WALL` (only for terrain) This flag has been superseded by the JSON entry
+ `connects_to`, but is retained for backward compatibility.
+- `CONSOLE` Used as a computer.
+- `CONTAINER` Items on this square are hidden until looted by the player.
+- `DECONSTRUCT` Can be deconstructed.
+- `DEEP_WATER`
+- `DESTROY_ITEM` Items that land here are destroyed. See also `NOITEM`
+- `DIFFICULT_Z` Most zombies will not be able to follow you up this terrain ( i.e a ladder )
+- `DIGGABLE_CAN_DEEPEN` Diggable location can be dug again to make deeper (e.g. shallow pit to deep
+ pit).
+- `DIGGABLE` Digging monsters, seeding monster, digging with shovel, etc.
+- `DOOR` Can be opened (used for NPC path-finding).
+- `EASY_DECONSTRUCT` Player can deconstruct this without tools.
+- `ELEVATOR` Terrain with this flag will move player, NPCs, monsters, and items up and down when
+ player activates nearby `elevator controls`.
+- `EMITTER` This furniture will emit fields automatically as defined by its emissions entry
+- `FIRE_CONTAINER` Stops fire from spreading (brazier, wood stove, etc.)
+- `FLAMMABLE_ASH` Burns to ash rather than rubble.
+- `FLAMMABLE_HARD` Harder to light on fire, but still possible.
+- `FLAMMABLE` Can be lit on fire.
+- `FLAT_SURF` Furniture or terrain with a flat hard surface (e.g. table, but not chair; tree stump,
+ etc.).
+- `FLAT` Player can build and move furniture on.
+- `FORAGE_HALLU` This item can be found with the `HIDDEN_HALLU` flag when found through foraging.
+- `FORAGE_POISION` This item can be found with the `HIDDEN_POISON` flag when found through foraging.
+- `FRIDGE` This item refrigerates items inside. Preserving them. Unlike vehicle parts, any furniture
+ with this flag constantly functions.
+- `FREEZER` This item freezes items inside. Preserving them extremely well. Unlike vehicle parts,
+ any furniture with this flag constantly functions.
+- `GOES_DOWN` Can use > to go down a level.
+- `GOES_UP` Can use < to go up a level.
+- `GROWTH_SEED` This plant is in its seed stage of growth.
+- `GROWTH_SEEDLING` This plant is in its seedling stage of growth.
+- `GROWTH_MATURE` This plant is in a mature stage of a growth.
+- `GROWTH_HARVEST` This plant is ready for harvest.
+- `HARVESTED` Marks the harvested version of a terrain type (e.g. harvesting an apple tree turns it
+ into a harvested tree, which later becomes an apple tree again).
+- `HIDE_PLACE` Creatures on this tile can't be seen by creatures not standing on adjacent tiles
+- `INDOORS` Has a roof over it; blocks rain, sunlight, etc.
+- `LADDER` This piece of furniture that makes climbing easy (works only with z-level mode).
+- `LIQUIDCONT` Furniture that contains liquid, allows for contents to be accessed in some checks
+ even if `SEALED`.
+- `LIQUID` Blocks movement, but isn't a wall (lava, water, etc.)
+- `MINEABLE` Can be mined with a pickaxe/jackhammer.
+- `MOUNTABLE` Suitable for guns with the `MOUNTED_GUN` flag.
+- `NOCOLLIDE` Feature that simply doesn't collide with vehicles at all.
+- `NOITEM` Items cannot be added here but may overflow to adjacent tiles. See also `DESTROY_ITEM`
+- `NO_FLOOR` Things should fall when placed on this tile
+- `NO_SIGHT` Creature on this tile have their sight reduced to one tile
+- `NO_SCENT` This tile cannot have scent values, which prevents scent diffusion through this tile
+- `OPENCLOSE_INSIDE` If it's a door (with an 'open' or 'close' field), it can only be opened or
+ closed if you're inside.
+- `PAINFUL` May cause a small amount of pain.
+- `PERMEABLE` Permeable for gases.
+- `PLACE_ITEM` Valid terrain for `place_item()` to put items on.
+- `PLANT` A 'furniture' that grows and fruits.
+- `PLANTABLE` This terrain or furniture can have seeds planted in it.
+- `PLOWABLE` Terrain can be plowed.
+- `RAMP_END`
+- `RAMP` Can be used to move up a z-level
+- `REDUCE_SCENT` Reduces scent diffusion (not total amount of scent in area); only works if also
+ bashable.
+- `ROAD` Flat and hard enough to drive or skate (with rollerblades) on.
+- `ROUGH` May hurt the player's feet.
+- `RUG` Enables the `Remove Carpet` Construction entry.
+- `SALT_WATER` Source of salt water (works for terrains with examine action "water_source").
+- `SEALED` Can't use e to retrieve items; must smash them open first.
+- `SEEN_FROM_ABOVE` Visible from a higher level (provided the tile above has no floor)
+- `SHARP` May do minor damage to players/monsters passing through it.
+- `SHORT` Feature too short to collide with vehicle protrusions. (mirrors, blades).
+- `SIGN` Show written message on examine.
+- `SMALL_PASSAGE` This terrain or furniture is too small for large or huge creatures to pass
+ through.
+- `SUN_ROOF_ABOVE` This furniture (terrain is not supported currently) has a "fake roof" above, that
+ blocks sunlight. Special hack for #44421, to be removed later.
+- `SUPPORTS_ROOF` Used as a boundary for roof construction.
+- `SUPPRESS_SMOKE` Prevents smoke from fires; used by ventilated wood stoves, etc.
+- `SWIMMABLE` Player and monsters can swim through it.
+- `THIN_OBSTACLE` Passable by players and monsters; vehicles destroy it.
+- `TINY` Feature too short to collide with vehicle undercarriage. Vehicles drive over them with no
+ damage, unless a wheel hits them.
+- `TRANSPARENT` Players and monsters can see through/past it. Also sets ter_t.transparent.
+- `UNSTABLE` Walking here cause the bouldering effect on the character.
+- `USABLE_FIRE` This terrain or furniture counts as a nearby fire for crafting.
+- `VEH_TREAT_AS_BASH_BELOW` Vehicles will not collide with this even if it counts as rough terrain,
+ like floor with bash_below does. Used for terrain meant to be turned into other terrain when
+ smashed instead of destroying the tile beneath it.
+- `WALL` This terrain is an upright obstacle. Used for fungal conversion, and also implies
+ `CONNECT_TO_WALL`.
### Examine Actions
-- ```aggie_plant``` Harvest plants.
-- ```autodoc``` Brings the autodoc consoles menu. Needs the ```AUTODOC``` flag to function properly and an adjacent furniture with the ```AUTODOC_COUCH``` flag.
-- ```autoclave_empty``` Start the autoclave cycle if it contains filthy CBM, and the player has enough water.
-- ```autoclave_full``` Check on the progress of the cycle, and collect sterile CBM once cycle is completed.
-- ```bars``` Take advantage of AMORPHOUS and slip through the bars.
-- ```bulletin_board``` Use this to arrange tasks for your faction camp.
-- ```cardreader``` Use the cardreader with a valid card, or attempt to hack.
-- ```chainfence``` Hop over the chain fence.
-- ```controls_gate``` Controls the attached gate.
-- ```dirtmound``` Plant seeds and plants.
-- ```elevator``` Use the elevator to change floors.
-- ```fault``` Displays descriptive message, but otherwise unused.
-- ```flower_poppy``` Pick the mutated poppy.
-- ```fswitch``` Flip the switch and the rocks will shift.
-- ```fungus``` Release spores as the terrain crumbles away.
-- ```gaspump``` Use the gas-pump.
-- ```none``` None
-- ```pedestal_temple``` Opens the temple if you have a petrified eye.
-- ```pedestal_wyrm``` Spawn wyrms.
-- ```pit_covered``` Uncover the pit.
-- ```pit``` Cover the pit if you have some planks of wood.
-- ```portable_structure``` Take down a tent or similar portable structure.
-- ```recycle_compactor``` Compress pure metal objects into basic shapes.
-- ```rubble``` Clear up the rubble if you have a shovel.
-- ```safe``` Attempt to crack the safe.
-- ```shelter``` Take down the shelter.
-- ```shrub_marloss``` Pick a marloss bush.
-- ```shrub_wildveggies``` Pick a wild veggies shrub.
-- ```slot_machine``` Gamble.
-- ```toilet``` Either drink or get water out of the toilet.
-- ```trap``` Interact with a trap.
-- ```water_source``` Drink or get water from a water source.
+- `aggie_plant` Harvest plants.
+- `autodoc` Brings the autodoc consoles menu. Needs the `AUTODOC` flag to function properly and an
+ adjacent furniture with the `AUTODOC_COUCH` flag.
+- `autoclave_empty` Start the autoclave cycle if it contains filthy CBM, and the player has enough
+ water.
+- `autoclave_full` Check on the progress of the cycle, and collect sterile CBM once cycle is
+ completed.
+- `bars` Take advantage of AMORPHOUS and slip through the bars.
+- `bulletin_board` Use this to arrange tasks for your faction camp.
+- `cardreader` Use the cardreader with a valid card, or attempt to hack.
+- `chainfence` Hop over the chain fence.
+- `controls_gate` Controls the attached gate.
+- `dirtmound` Plant seeds and plants.
+- `elevator` Use the elevator to change floors.
+- `fault` Displays descriptive message, but otherwise unused.
+- `flower_poppy` Pick the mutated poppy.
+- `fswitch` Flip the switch and the rocks will shift.
+- `fungus` Release spores as the terrain crumbles away.
+- `gaspump` Use the gas-pump.
+- `none` None
+- `pedestal_temple` Opens the temple if you have a petrified eye.
+- `pedestal_wyrm` Spawn wyrms.
+- `pit_covered` Uncover the pit.
+- `pit` Cover the pit if you have some planks of wood.
+- `portable_structure` Take down a tent or similar portable structure.
+- `recycle_compactor` Compress pure metal objects into basic shapes.
+- `rubble` Clear up the rubble if you have a shovel.
+- `safe` Attempt to crack the safe.
+- `shelter` Take down the shelter.
+- `shrub_marloss` Pick a marloss bush.
+- `shrub_wildveggies` Pick a wild veggies shrub.
+- `slot_machine` Gamble.
+- `toilet` Either drink or get water out of the toilet.
+- `trap` Interact with a trap.
+- `water_source` Drink or get water from a water source.
### Fungal Conversions Only
-- ```FLOWER``` This furniture is a flower.
-- ```FUNGUS``` Fungal covered.
-- ```ORGANIC``` This furniture is partly organic.
-- ```SHRUB``` This terrain is a shrub.
-- ```TREE``` This terrain is a tree.
-- ```YOUNG``` This terrain is a young tree.
+- `FLOWER` This furniture is a flower.
+- `FUNGUS` Fungal covered.
+- `ORGANIC` This furniture is partly organic.
+- `SHRUB` This terrain is a shrub.
+- `TREE` This terrain is a tree.
+- `YOUNG` This terrain is a young tree.
### Furniture Only
-- ```AUTODOC``` This furniture can be an autodoc console, it also needs the ```autodoc``` examine action.
-- ```AUTODOC_COUCH``` This furniture can be a couch for a furniture with the ```autodoc``` examine action.
-- ```BLOCKSDOOR``` This will boost map terrain's resistance to bashing if `str_*_blocked` is set (see `map_bash_info`)
-
+- `AUTODOC` This furniture can be an autodoc console, it also needs the `autodoc` examine action.
+- `AUTODOC_COUCH` This furniture can be a couch for a furniture with the `autodoc` examine action.
+- `BLOCKSDOOR` This will boost map terrain's resistance to bashing if `str_*_blocked` is set (see
+ `map_bash_info`)
## Generic
### Flags
-- ```BIONIC_NPC_USABLE``` ... Safe CBMs that NPCs can use without extensive NPC rewrites to utilize toggle CBMs.
-- ```BIONIC_TOGGLED``` ... This bionic only has a function when activated, instead of causing its effect every turn.
-- ```BIONIC_POWER_SOURCE``` ... This bionic is a source of bionic power.
-- ```BIONIC_SHOCKPROOF``` ... This bionic can't be incapacitated by electrical attacks.
-- ```BIONIC_FAULTY``` ... This bionic is a "faulty" bionic.
-- ```BIONIC_WEAPON``` ... This bionic is a weapon bionic and activating it will create (or destroy) bionic's fake_item in user's hands. Prevents all other activation effects.
-- ```BIONIC_ARMOR_INTERFACE``` ... This bionic can provide power to powered armor.
-- ```BIONIC_SLEEP_FRIENDLY``` ... This bionic won't provide a warning if the player tries to sleep while it's active.
-- ```BIONIC_GUN``` ... This bionic is a gun bionic and activating it will fire it. Prevents all other activation effects.
-- ```CORPSE``` ... Flag used to spawn various human corpses during the mapgen.
-- ```DANGEROUS``` ... NPCs will not accept this item. Explosion iuse actor implies this flag. Implies "NPC_THROW_NOW".
-- ```DETERGENT``` ... This item can be used as a detergent in a washing machine.
-- ```DURABLE_MELEE``` ... Item is made to hit stuff and it does it well, so it's considered to be a lot tougher than other weapons made of the same materials.
-- ```FAKE_MILL``` ... Item is a fake item, to denote a partially milled product by @ref Item::process_fake_mill, where conditions for its removal are set.
-- ```FAKE_SMOKE``` ... Item is a fake item generating smoke, recognizable by @ref item::process_fake_smoke, where conditions for its removal are set.
-- ```FIREWOOD``` ... This item can serve as a firewood. Items with this flag are sorted out to "Loot: Wood" zone
-- ```FRAGILE_MELEE``` ... Fragile items that fall apart easily when used as a weapon due to poor construction quality and will break into components when broken.
-- ```GAS_DISCOUNT``` ... Discount cards for the automated gas stations.
-- ```IS_PET_ARMOR``` ... Is armor for a pet monster, not armor for a person
-- ```LEAK_ALWAYS``` ... Leaks (may be combined with "RADIOACTIVE").
-- ```LEAK_DAM``` ... Leaks when damaged (may be combined with "RADIOACTIVE").
-- ```NEEDS_UNFOLD``` ... Has an additional time penalty upon wielding. For melee weapons and guns this is offset by the relevant skill. Stacks with "SLOW_WIELD".
-- ```NO_PACKED``` ... This item is not protected against contamination and won't stay sterile. Only applies to CBMs.
-- ```NO_REPAIR``` ... Prevents repairing of this item even if otherwise suitable tools exist.
-- ```NO_SALVAGE``` ... Item cannot be broken down through a salvage process. Best used when something should not be able to be broken down (i.e. base components like leather patches).
-- ```NO_STERILE``` ... This item is not sterile. Only applies to CBMs.
-- ```NPC_ACTIVATE``` ... NPCs can activate this item as an alternative attack. Currently by throwing it right after activation. Implied by "BOMB".
-- ```NPC_ALT_ATTACK``` ... Shouldn't be set directly. Implied by "NPC_ACTIVATE" and "NPC_THROWN".
-- ```NPC_THROWN``` ... NPCs will throw this item (without activating it first) as an alternative attack.
-- ```NPC_THROW_NOW``` ... NPCs will try to throw this item away, preferably at enemies. Implies "TRADER_AVOID" and "NPC_THROWN".
-- ```PSEUDO``` ... Used internally to mark items that are referred to in the crafting inventory but are not actually items. They can be used as tools, but not as components. Implies "TRADER_AVOID".
-- ```RADIOACTIVE``` ... Is radioactive (can be used with LEAK_*).
-- ```RAIN_PROTECT``` ... Protects from sunlight and from rain, when wielded.
-- ```REDUCED_BASHING``` ... Gunmod flag; reduces the item's bashing damage by 50%.
-- ```REDUCED_WEIGHT``` ... Gunmod flag; reduces the item's base weight by 25%.
-- ```REQUIRES_TINDER``` ... Requires tinder to be present on the tile this item tries to start a fire on.
-- ```SLEEP_AID``` ... This item helps in sleeping.
-- ```SLEEP_IGNORE``` ... This item is not shown as before-sleep warning.
-- ```SLOW_WIELD``` ... Has an additional time penalty upon wielding. For melee weapons and guns this is offset by the relevant skill. Stacks with "NEEDS_UNFOLD".
-- ```TACK``` ... Item can be used as tack for a mount.
-- ```TIE_UP``` ... Item can be used to tie up a creature.
-- ```TINDER``` ... This item can be used as tinder for lighting a fire with a REQUIRES_TINDER flagged firestarter.
-- ```TRADER_AVOID``` ... NPCs will not start with this item. Use this for active items (e.g. flashlight (on)), dangerous items (e.g. active bomb), fake item or unusual items (e.g. unique quest item).
-- ```UNBREAKABLE_MELEE``` ... Does never get damaged when used as melee weapon.
-- ```UNRECOVERABLE``` ... Cannot be recovered from a disassembly.
-
+- `BIONIC_NPC_USABLE` ... Safe CBMs that NPCs can use without extensive NPC rewrites to utilize
+ toggle CBMs.
+- `BIONIC_TOGGLED` ... This bionic only has a function when activated, instead of causing its effect
+ every turn.
+- `BIONIC_POWER_SOURCE` ... This bionic is a source of bionic power.
+- `BIONIC_SHOCKPROOF` ... This bionic can't be incapacitated by electrical attacks.
+- `BIONIC_FAULTY` ... This bionic is a "faulty" bionic.
+- `BIONIC_WEAPON` ... This bionic is a weapon bionic and activating it will create (or destroy)
+ bionic's fake_item in user's hands. Prevents all other activation effects.
+- `BIONIC_ARMOR_INTERFACE` ... This bionic can provide power to powered armor.
+- `BIONIC_SLEEP_FRIENDLY` ... This bionic won't provide a warning if the player tries to sleep while
+ it's active.
+- `BIONIC_GUN` ... This bionic is a gun bionic and activating it will fire it. Prevents all other
+ activation effects.
+- `CORPSE` ... Flag used to spawn various human corpses during the mapgen.
+- `DANGEROUS` ... NPCs will not accept this item. Explosion iuse actor implies this flag. Implies
+ "NPC_THROW_NOW".
+- `DETERGENT` ... This item can be used as a detergent in a washing machine.
+- `DURABLE_MELEE` ... Item is made to hit stuff and it does it well, so it's considered to be a lot
+ tougher than other weapons made of the same materials.
+- `FAKE_MILL` ... Item is a fake item, to denote a partially milled product by @ref
+ Item::process_fake_mill, where conditions for its removal are set.
+- `FAKE_SMOKE` ... Item is a fake item generating smoke, recognizable by @ref
+ item::process_fake_smoke, where conditions for its removal are set.
+- `FIREWOOD` ... This item can serve as a firewood. Items with this flag are sorted out to "Loot:
+ Wood" zone
+- `FRAGILE_MELEE` ... Fragile items that fall apart easily when used as a weapon due to poor
+ construction quality and will break into components when broken.
+- `GAS_DISCOUNT` ... Discount cards for the automated gas stations.
+- `IS_PET_ARMOR` ... Is armor for a pet monster, not armor for a person
+- `LEAK_ALWAYS` ... Leaks (may be combined with "RADIOACTIVE").
+- `LEAK_DAM` ... Leaks when damaged (may be combined with "RADIOACTIVE").
+- `NEEDS_UNFOLD` ... Has an additional time penalty upon wielding. For melee weapons and guns this
+ is offset by the relevant skill. Stacks with "SLOW_WIELD".
+- `NO_PACKED` ... This item is not protected against contamination and won't stay sterile. Only
+ applies to CBMs.
+- `NO_REPAIR` ... Prevents repairing of this item even if otherwise suitable tools exist.
+- `NO_SALVAGE` ... Item cannot be broken down through a salvage process. Best used when something
+ should not be able to be broken down (i.e. base components like leather patches).
+- `NO_STERILE` ... This item is not sterile. Only applies to CBMs.
+- `NPC_ACTIVATE` ... NPCs can activate this item as an alternative attack. Currently by throwing it
+ right after activation. Implied by "BOMB".
+- `NPC_ALT_ATTACK` ... Shouldn't be set directly. Implied by "NPC_ACTIVATE" and "NPC_THROWN".
+- `NPC_THROWN` ... NPCs will throw this item (without activating it first) as an alternative attack.
+- `NPC_THROW_NOW` ... NPCs will try to throw this item away, preferably at enemies. Implies
+ "TRADER_AVOID" and "NPC_THROWN".
+- `PSEUDO` ... Used internally to mark items that are referred to in the crafting inventory but are
+ not actually items. They can be used as tools, but not as components. Implies "TRADER_AVOID".
+- `RADIOACTIVE` ... Is radioactive (can be used with LEAK_*).
+- `RAIN_PROTECT` ... Protects from sunlight and from rain, when wielded.
+- `REDUCED_BASHING` ... Gunmod flag; reduces the item's bashing damage by 50%.
+- `REDUCED_WEIGHT` ... Gunmod flag; reduces the item's base weight by 25%.
+- `REQUIRES_TINDER` ... Requires tinder to be present on the tile this item tries to start a fire
+ on.
+- `SLEEP_AID` ... This item helps in sleeping.
+- `SLEEP_IGNORE` ... This item is not shown as before-sleep warning.
+- `SLOW_WIELD` ... Has an additional time penalty upon wielding. For melee weapons and guns this is
+ offset by the relevant skill. Stacks with "NEEDS_UNFOLD".
+- `TACK` ... Item can be used as tack for a mount.
+- `TIE_UP` ... Item can be used to tie up a creature.
+- `TINDER` ... This item can be used as tinder for lighting a fire with a REQUIRES_TINDER flagged
+ firestarter.
+- `TRADER_AVOID` ... NPCs will not start with this item. Use this for active items (e.g. flashlight
+ (on)), dangerous items (e.g. active bomb), fake item or unusual items (e.g. unique quest item).
+- `UNBREAKABLE_MELEE` ... Does never get damaged when used as melee weapon.
+- `UNRECOVERABLE` ... Cannot be recovered from a disassembly.
## Guns
-- ```BACKBLAST``` Causes a small explosion behind the person firing the weapon. Currently not implemented?
-- ```BIPOD``` Handling bonus only applies on MOUNTABLE map/vehicle tiles. Does not include wield time penalty (see SLOW_WIELD).
-- ```CHARGE``` Has to be charged to fire. Higher charges do more damage.
-- ```COLLAPSIBLE_STOCK``` Reduces weapon volume proportional to the base size of the gun (excluding any mods). Does not include wield time penalty (see NEEDS_UNFOLD).
-- ```CONSUMABLE``` Makes a gunpart have a chance to get damaged depending on ammo fired, and definable fields 'consume_chance' and 'consume_divisor'.
-- ```CROSSBOW``` Counts as a crossbow for the purpose of gunmod compatibility. Default behavior is to match the skill used by that weapon.
-- ```DISABLE_SIGHTS``` Prevents use of the base weapon sights
-- ```FIRE_100``` Uses 100 shots per firing.
-- ```FIRE_50``` Uses 50 shots per firing.
-- ```FIRE_TWOHAND``` Gun can only be fired if player has two free hands.
-- ```IRREMOVABLE``` Makes so that the gunmod cannot be removed.
-- ```MECH_BAT``` This is an exotic battery designed to power military mechs.
-- ```MOUNTED_GUN``` Gun can only be used on terrain / furniture with the "MOUNTABLE" flag, if you're a normal human. If you're an oversized mutant (Inconveniently Large, Large, Freakishly Huge, Huge), you can fire it regularly in exchange for dispersion and recoil penalties.
-- ```NEVER_JAMS``` Never malfunctions.
-- ```NO_UNLOAD``` Cannot be unloaded.
-- ```PRIMITIVE_RANGED_WEAPON``` Allows using non-gunsmith tools to repair it (but not reinforce).
-- ```PUMP_ACTION``` Gun has a rails on its pump action, allowing to install only mods with PUMP_RAIL_COMPATIBLE flag on underbarrel slot.
-- ```PUMP_RAIL_COMPATIBLE``` Mod can be installed on underbarrel slot of guns with rails on their pump action.
-- ```RELOAD_AND_SHOOT``` Firing automatically reloads and then shoots.
-- ```RELOAD_EJECT``` Ejects shell from gun on reload instead of when fired.
-- ```RELOAD_ONE``` Only reloads one round at a time.
-- ```STR_DRAW``` If the weapon also has a strength requirement, lacking the requirement will penalize damage, range, and dispersion until you're unable to fire if below 50% the listed strength, instead of being a strict limit like it normally would be.
-- ```STR_RELOAD``` Reload speed is affected by strength.
-- ```UNDERWATER_GUN``` Gun is optimized for usage underwater, does perform badly outside of water.
-- ```WATERPROOF_GUN``` Gun does not rust and can be used underwater.
+- `BACKBLAST` Causes a small explosion behind the person firing the weapon. Currently not
+ implemented?
+- `BIPOD` Handling bonus only applies on MOUNTABLE map/vehicle tiles. Does not include wield time
+ penalty (see SLOW_WIELD).
+- `CHARGE` Has to be charged to fire. Higher charges do more damage.
+- `COLLAPSIBLE_STOCK` Reduces weapon volume proportional to the base size of the gun (excluding any
+ mods). Does not include wield time penalty (see NEEDS_UNFOLD).
+- `CONSUMABLE` Makes a gunpart have a chance to get damaged depending on ammo fired, and definable
+ fields 'consume_chance' and 'consume_divisor'.
+- `CROSSBOW` Counts as a crossbow for the purpose of gunmod compatibility. Default behavior is to
+ match the skill used by that weapon.
+- `DISABLE_SIGHTS` Prevents use of the base weapon sights
+- `FIRE_100` Uses 100 shots per firing.
+- `FIRE_50` Uses 50 shots per firing.
+- `FIRE_TWOHAND` Gun can only be fired if player has two free hands.
+- `IRREMOVABLE` Makes so that the gunmod cannot be removed.
+- `MECH_BAT` This is an exotic battery designed to power military mechs.
+- `MOUNTED_GUN` Gun can only be used on terrain / furniture with the "MOUNTABLE" flag, if you're a
+ normal human. If you're an oversized mutant (Inconveniently Large, Large, Freakishly Huge, Huge),
+ you can fire it regularly in exchange for dispersion and recoil penalties.
+- `NEVER_JAMS` Never malfunctions.
+- `NO_UNLOAD` Cannot be unloaded.
+- `PRIMITIVE_RANGED_WEAPON` Allows using non-gunsmith tools to repair it (but not reinforce).
+- `PUMP_ACTION` Gun has a rails on its pump action, allowing to install only mods with
+ PUMP_RAIL_COMPATIBLE flag on underbarrel slot.
+- `PUMP_RAIL_COMPATIBLE` Mod can be installed on underbarrel slot of guns with rails on their pump
+ action.
+- `RELOAD_AND_SHOOT` Firing automatically reloads and then shoots.
+- `RELOAD_EJECT` Ejects shell from gun on reload instead of when fired.
+- `RELOAD_ONE` Only reloads one round at a time.
+- `STR_DRAW` If the weapon also has a strength requirement, lacking the requirement will penalize
+ damage, range, and dispersion until you're unable to fire if below 50% the listed strength,
+ instead of being a strict limit like it normally would be.
+- `STR_RELOAD` Reload speed is affected by strength.
+- `UNDERWATER_GUN` Gun is optimized for usage underwater, does perform badly outside of water.
+- `WATERPROOF_GUN` Gun does not rust and can be used underwater.
### Firing modes
-- ```MELEE``` Melee attack using properties of the gun or auxiliary gunmod
-- ```NPC_AVOID``` NPC's will not attempt to use this mode
-- ```SIMULTANEOUS``` All rounds fired concurrently (not sequentially) with recoil added only once (at the end)
+- `MELEE` Melee attack using properties of the gun or auxiliary gunmod
+- `NPC_AVOID` NPC's will not attempt to use this mode
+- `SIMULTANEOUS` All rounds fired concurrently (not sequentially) with recoil added only once (at
+ the end)
### Faults
#### Flags
-- ```SILENT``` Makes the "faulty " text NOT appear next to item on general UI. Otherwise the fault works the same.
+- `SILENT` Makes the "faulty " text NOT appear next to item on general UI. Otherwise the fault works
+ the same.
#### Parameters
-- ```turns_into``` Causes this fault to apply to the item just mended.
-- ```also_mends``` Causes this fault to be mended (in addition to fault selected) once that fault is mended.
-
+- `turns_into` Causes this fault to apply to the item just mended.
+- `also_mends` Causes this fault to be mended (in addition to fault selected) once that fault is
+ mended.
## Magazines
-- ```MAG_BULKY``` Can be stashed in an appropriate oversize ammo pouch (intended for bulky or awkwardly shaped magazines)
-- ```MAG_COMPACT``` Can be stashed in an appropriate ammo pouch (intended for compact magazines)
-- ```MAG_DESTROY``` Magazine is destroyed when the last round is consumed (intended for ammo belts). Has precedence over MAG_EJECT.
-- ```MAG_EJECT``` Magazine is ejected from the gun/tool when the last round is consumed
-- ```SPEEDLOADER``` Acts like a magazine, except it transfers rounds to the target gun instead of being inserted into it.
-
+- `MAG_BULKY` Can be stashed in an appropriate oversize ammo pouch (intended for bulky or awkwardly
+ shaped magazines)
+- `MAG_COMPACT` Can be stashed in an appropriate ammo pouch (intended for compact magazines)
+- `MAG_DESTROY` Magazine is destroyed when the last round is consumed (intended for ammo belts). Has
+ precedence over MAG_EJECT.
+- `MAG_EJECT` Magazine is ejected from the gun/tool when the last round is consumed
+- `SPEEDLOADER` Acts like a magazine, except it transfers rounds to the target gun instead of being
+ inserted into it.
## MAP SPECIALS
-- ```mx_bandits_block``` ... Road block made by bandits from tree logs.
-- ```mx_burned_ground``` ... Fire has ravaged this place.
-- ```mx_point_burned_ground``` ... Fire has ravaged this place. (partial application)
-- ```mx_casings``` ... Several types of spent casings (solitary, groups, entire overmap tile)
-- ```mx_clay_deposit``` ... A small surface clay deposit.
-- ```mx_clearcut``` ... All trees become stumps.
-- ```mx_collegekids``` ... Corpses and items.
-- ```mx_corpses``` ... Up to 5 corpses with everyday loot.
-- ```mx_crater``` ... Crater with rubble (and radioactivity).
-- ```mx_drugdeal``` ... Corpses and some drugs.
-- ```mx_dead_vegetation``` ... Kills all plants. (aftermath of acid rain etc.)
-- ```mx_point_dead_vegetation``` ... Kills all plants. (aftermath of acid rain etc.) (partial application)
-- ```mx_grove``` ... All trees and shrubs become a single species of tree.
-- ```mx_grave``` ... A grave in the open field, with a corpse and some everyday loot.
-- ```mx_helicopter``` ... Metal wreckage and some items.
-- ```mx_jabberwock``` ... A *chance* of a jabberwock.
-- ```mx_looters``` ... Up to 5 bandits spawn in the building.
-- ```mx_marloss_pilgrimage``` A sect of people worshiping fungaloids.
-- ```mx_mayhem``` ... Several types of road mayhem (firefights, crashed cars etc).
-- ```mx_military``` ... Corpses and some military items.
-- ```mx_minefield``` ... A military roadblock at the entry of the bridges with landmines scattered in the front of it.
-- ```mx_null``` ... No special at all.
-- ```mx_pond``` ... A small pond.
-- ```mx_portal_in``` ... Another portal to neither space.
-- ```mx_portal``` ... Portal to neither space, with several types of surrounding environment.
-- ```mx_roadblock``` ... Roadblock furniture with turrets and some cars.
-- ```mx_roadworks``` ... Partially closed damaged road with chance of work equipment and utility vehicles.
-- ```mx_science``` ... Corpses and some scientist items.
-- ```mx_shia``` ... A *chance* of Shia, if Crazy Catalcysm is enabled.
-- ```mx_shrubbery``` ... All trees and shrubs become a single species of shrub.
-- ```mx_spider``` ... A big spider web, complete with spiders and eggs.
-- ```mx_supplydrop``` ... Crates with some military items in it.
-
+- `mx_bandits_block` ... Road block made by bandits from tree logs.
+- `mx_burned_ground` ... Fire has ravaged this place.
+- `mx_point_burned_ground` ... Fire has ravaged this place. (partial application)
+- `mx_casings` ... Several types of spent casings (solitary, groups, entire overmap tile)
+- `mx_clay_deposit` ... A small surface clay deposit.
+- `mx_clearcut` ... All trees become stumps.
+- `mx_collegekids` ... Corpses and items.
+- `mx_corpses` ... Up to 5 corpses with everyday loot.
+- `mx_crater` ... Crater with rubble (and radioactivity).
+- `mx_drugdeal` ... Corpses and some drugs.
+- `mx_dead_vegetation` ... Kills all plants. (aftermath of acid rain etc.)
+- `mx_point_dead_vegetation` ... Kills all plants. (aftermath of acid rain etc.) (partial
+ application)
+- `mx_grove` ... All trees and shrubs become a single species of tree.
+- `mx_grave` ... A grave in the open field, with a corpse and some everyday loot.
+- `mx_helicopter` ... Metal wreckage and some items.
+- `mx_jabberwock` ... A _chance_ of a jabberwock.
+- `mx_looters` ... Up to 5 bandits spawn in the building.
+- `mx_marloss_pilgrimage` A sect of people worshiping fungaloids.
+- `mx_mayhem` ... Several types of road mayhem (firefights, crashed cars etc).
+- `mx_military` ... Corpses and some military items.
+- `mx_minefield` ... A military roadblock at the entry of the bridges with landmines scattered in
+ the front of it.
+- `mx_null` ... No special at all.
+- `mx_pond` ... A small pond.
+- `mx_portal_in` ... Another portal to neither space.
+- `mx_portal` ... Portal to neither space, with several types of surrounding environment.
+- `mx_roadblock` ... Roadblock furniture with turrets and some cars.
+- `mx_roadworks` ... Partially closed damaged road with chance of work equipment and utility
+ vehicles.
+- `mx_science` ... Corpses and some scientist items.
+- `mx_shia` ... A _chance_ of Shia, if Crazy Catalcysm is enabled.
+- `mx_shrubbery` ... All trees and shrubs become a single species of shrub.
+- `mx_spider` ... A big spider web, complete with spiders and eggs.
+- `mx_supplydrop` ... Crates with some military items in it.
## Material Phases
-- ```GAS```
-- ```LIQUID```
-- ```NULL```
-- ```PLASMA```
-- ```SOLID```
-
+- `GAS`
+- `LIQUID`
+- `NULL`
+- `PLASMA`
+- `SOLID`
## Melee
### Flags
-- ```ALWAYS_TWOHAND``` Item is always wielded with two hands. Without this, the items volume and weight are used to calculate this.
-- ```DIAMOND``` Diamond coating adds 30% bonus to cutting and piercing damage
-- ```MESSY``` Creates more mess when pulping
-- ```NO_CVD``` Item can never be used with a CVD machine
-- ```NO_RELOAD``` Item can never be reloaded (even if has a valid ammo type).
-- ```NO_UNWIELD``` Cannot unwield this item.
-- ```POLEARM``` Item is clumsy up close and does 70% of normal damage against adjacent targets. Should be paired with REACH_ATTACK. Simple reach piercing weapons like spears should not get this flag.
-- ```REACH_ATTACK``` Allows to perform reach attack.
-- ```SHEATH_KNIFE``` Item can be sheathed in a knife sheath, it applicable to small/medium knives (with volume not bigger than 2)
-- ```SHEATH_SWORD``` Item can be sheathed in a sword scabbard
-- ```SPEAR``` When making reach attacks intervening THIN_OBSTACLE terrain is not an obstacle. Should be paired with REACH_ATTACK.
-- ```UNARMED_WEAPON``` Wielding this item still counts as unarmed combat.
-- ```WHIP``` Has a chance of disarming the opponent.
-
+- `ALWAYS_TWOHAND` Item is always wielded with two hands. Without this, the items volume and weight
+ are used to calculate this.
+- `DIAMOND` Diamond coating adds 30% bonus to cutting and piercing damage
+- `MESSY` Creates more mess when pulping
+- `NO_CVD` Item can never be used with a CVD machine
+- `NO_RELOAD` Item can never be reloaded (even if has a valid ammo type).
+- `NO_UNWIELD` Cannot unwield this item.
+- `POLEARM` Item is clumsy up close and does 70% of normal damage against adjacent targets. Should
+ be paired with REACH_ATTACK. Simple reach piercing weapons like spears should not get this flag.
+- `REACH_ATTACK` Allows to perform reach attack.
+- `SHEATH_KNIFE` Item can be sheathed in a knife sheath, it applicable to small/medium knives (with
+ volume not bigger than 2)
+- `SHEATH_SWORD` Item can be sheathed in a sword scabbard
+- `SPEAR` When making reach attacks intervening THIN_OBSTACLE terrain is not an obstacle. Should be
+ paired with REACH_ATTACK.
+- `UNARMED_WEAPON` Wielding this item still counts as unarmed combat.
+- `WHIP` Has a chance of disarming the opponent.
## Monster Groups
@@ -831,22 +952,23 @@ The condition flags limit when monsters can spawn.
### Seasons
-Multiple season conditions will be combined together so that any of those conditions become valid time of year spawn times.
+Multiple season conditions will be combined together so that any of those conditions become valid
+time of year spawn times.
-- ```AUTUMN```
-- ```SPRING```
-- ```SUMMER```
-- ```WINTER```
+- `AUTUMN`
+- `SPRING`
+- `SUMMER`
+- `WINTER`
### Time of day
-Multiple time of day conditions will be combined together so that any of those conditions become valid time of day spawn times.
-
-- ```DAWN```
-- ```DAY```
-- ```DUSK```
-- ```NIGHT```
+Multiple time of day conditions will be combined together so that any of those conditions become
+valid time of day spawn times.
+- `DAWN`
+- `DAY`
+- `DUSK`
+- `NIGHT`
## Monsters
@@ -854,406 +976,430 @@ Flags used to describe monsters and define their properties and abilities.
### Anger, Fear and Placation Triggers
-- ```FIRE``` There's a fire nearby.
-- ```FRIEND_ATTACKED``` A monster of the same type was attacked.
-- ```FRIEND_DIED``` A monster of the same type died.
-- ```HURT``` The monster is hurt.
-- ```MEAT``` Meat or a corpse is nearby.
-- ```NULL``` Source use only?
-- ```PLAYER_CLOSE``` The player gets within a few tiles distance.
-- ```PLAYER_WEAK``` The player is hurt.
-- ```SOUND``` Heard a sound.
-- ```STALK``` Increases when following the player.
+- `FIRE` There's a fire nearby.
+- `FRIEND_ATTACKED` A monster of the same type was attacked.
+- `FRIEND_DIED` A monster of the same type died.
+- `HURT` The monster is hurt.
+- `MEAT` Meat or a corpse is nearby.
+- `NULL` Source use only?
+- `PLAYER_CLOSE` The player gets within a few tiles distance.
+- `PLAYER_WEAK` The player is hurt.
+- `SOUND` Heard a sound.
+- `STALK` Increases when following the player.
### Categories
-- ```CLASSIC``` Only monsters we expect in a classic zombie movie.
-- ```NULL``` No category.
-- ```WILDLIFE``` Natural animals.
+- `CLASSIC` Only monsters we expect in a classic zombie movie.
+- `NULL` No category.
+- `WILDLIFE` Natural animals.
### Death Functions
Multiple death functions can be used. Not all combinations make sense.
-- ```ACID``` Acid instead of a body. not the same as the ACID_BLOOD flag. In most cases you want both.
-- ```AMIGARA``` Removes hypnosis if the last one.
-- ```BLOBSPLIT``` Creates more blobs.
-- ```BOOMER``` Explodes in vomit.
-- ```BROKEN``` Spawns a broken robot item, its id calculated like this: the prefix "mon_" is removed from the monster id, than the prefix "broken_" is added. Example: mon_eyebot -> broken_eyebot
-- ```DISAPPEAR``` Hallucination disappears.
-- ```DISINTEGRATE``` Falls apart.
-- ```EXPLODE``` Damaging explosion.
-- ```FIREBALL``` 10 percent chance to explode in a fireball.
-- ```FLAME_EXPLOSION``` guaranteed to explode and starts fires.
-- ```FUNGUS``` Explodes in spores.
-- ```GAMEOVER``` Game over man! Game over! Defense mode.
-- ```GUILT``` Moral penalty. There is also a flag with a similar effect.
-- ```KILL_BREATHERS``` All breathers die.
-- ```KILL_VINES``` Kill all nearby vines.
-- ```MELT``` Normal death, but melts.
-- ```NORMAL``` Drop a body, leave gibs.
-- ```RATKING``` Cure verminitis.
-- ```SMOKEBURST``` Explode like a huge smoke bomb.
-- ```THING``` Turn into a full thing.
-- ```TRIFFID_HEART``` Destroys all roots.
-- ```VINE_CUT``` Kill adjacent vine if it's cut.
-- ```WORM``` Spawns 2 half-worms
+- `ACID` Acid instead of a body. not the same as the ACID_BLOOD flag. In most cases you want both.
+- `AMIGARA` Removes hypnosis if the last one.
+- `BLOBSPLIT` Creates more blobs.
+- `BOOMER` Explodes in vomit.
+- `BROKEN` Spawns a broken robot item, its id calculated like this: the prefix "mon_" is removed
+ from the monster id, than the prefix "broken_" is added. Example: mon_eyebot -> broken_eyebot
+- `DISAPPEAR` Hallucination disappears.
+- `DISINTEGRATE` Falls apart.
+- `EXPLODE` Damaging explosion.
+- `FIREBALL` 10 percent chance to explode in a fireball.
+- `FLAME_EXPLOSION` guaranteed to explode and starts fires.
+- `FUNGUS` Explodes in spores.
+- `GAMEOVER` Game over man! Game over! Defense mode.
+- `GUILT` Moral penalty. There is also a flag with a similar effect.
+- `KILL_BREATHERS` All breathers die.
+- `KILL_VINES` Kill all nearby vines.
+- `MELT` Normal death, but melts.
+- `NORMAL` Drop a body, leave gibs.
+- `RATKING` Cure verminitis.
+- `SMOKEBURST` Explode like a huge smoke bomb.
+- `THING` Turn into a full thing.
+- `TRIFFID_HEART` Destroys all roots.
+- `VINE_CUT` Kill adjacent vine if it's cut.
+- `WORM` Spawns 2 half-worms
### Flags
-- ```ABSORBS_SPLITS``` Consumes objects it moves over, and if it absorbs enough it will split into a copy.
-- ```ABSORBS``` Consumes objects it moves over. (Modders use this).
-- ```ACIDPROOF``` Immune to acid.
-- ```ACIDTRAIL``` Leaves a trail of acid.
-- ```ACID_BLOOD``` Makes monster bleed acid. Fun stuff! Does not automatically dissolve in a pool of acid on death.
-- ```ANIMAL``` Is an _animal_ for purposes of the `Animal Empathy` trait.
-- ```AQUATIC``` Confined to water.
-- ```ARTHROPOD_BLOOD``` Forces monster to bleed hemolymph.
-- ```ATTACKMON``` Attacks other monsters.
-- ```BADVENOM``` Attack may **severely** poison the player.
-- ```BASHES``` Bashes down doors.
-- ```BILE_BLOOD``` Makes monster bleed bile.
-- ```BIRDFOOD``` Becomes friendly / tamed with bird food.
-- ```BLEED``` Causes the player to bleed.
-- ```BONES``` May produce bones and sinews when butchered.
-- ```BORES``` Tunnels through just about anything (15x bash multiplier: dark wyrms' bash skill 12->180)
-- ```CAN_DIG``` Can dig _and_ walk.
-- ```CAN_OPEN_DOORS``` Can open doors on its path.
-- ```CANPLAY``` This creature can be played with if it's a pet.
-- ```CATFOOD``` Becomes friendly / tamed with cat food.
-- ```CATTLEFODDER``` Becomes friendly / tamed with cattle fodder.
-- ```CBM_CIV``` May produce a common CBM a power CBM when butchered.
-- ```CBM_OP``` May produce a CBM or two from 'bionics_op' item group when butchered.
-- ```CBM_POWER``` May produce a power CBM when butchered, independent of CBM.
-- ```CBM_SCI``` May produce a CBM from 'bionics_sci' item group when butchered.
-- ```CBM_SUBS``` May produce a CBM or two from bionics_subs and a power CBM when butchered.
-- ```CBM_TECH``` May produce a CBM or two from 'bionics_tech' item group and a power CBM when butchered.
-- ```CHITIN``` May produce chitin when butchered.
-- ```CLIMBS``` Can climb.
-- ```COLDROOF``` Immune to cold damage.
-- ```CURRENT``` this water is flowing.
-- ```DESTROYS``` Bashes down walls and more. (2.5x bash multiplier, where base is the critter's max melee bashing)
-- ```DIGS``` Digs through the ground.
-- ```DOGFOOD``` Becomes friendly / tamed with dog food.
-- ```DRIPS_GASOLINE``` Occasionally drips gasoline on move.
-- ```DRIPS_NAPALM``` Occasionally drips napalm on move.
-- ```ELECTRIC``` Shocks unarmed attackers.
-- ```ELECTRONIC``` e.g. A Robot; affected by emp blasts and other stuff.
-- ```FAT``` May produce fat when butchered.
-- ```FEATHER``` May produce feathers when butchered.
-- ```FILTHY``` Any clothing it drops will be filthy.
-- ```FIREPROOF``` Immune to fire.
-- ```FIREY``` Burns stuff and is immune to fire.
-- ```FISHABLE``` It is fishable.
-- ```FLAMMABLE``` Monster catches fire, burns, and spreads fire to nearby objects.
-- ```FLIES``` Can fly (over water, etc.)
-- ```FUR``` May produce fur when butchered.
-- ```GOODHEARING``` Pursues sounds more than most monsters.
-- ```GRABS``` Its attacks may grab you!
-- ```GROUP_BASH``` Gets help from monsters around it when bashing.
-- ```GROUP_MORALE``` More courageous when near friends.
-- ```GUILT``` You feel guilty for killing it.
-- ```HARDTOSHOOT``` It's one size smaller for ranged attacks, no less then MS_TINY
-- ```HEARS``` It can hear you.
-- ```HIT_AND_RUN``` Flee for several turns after a melee attack.
-- ```HUMAN``` It's a live human, as long as it's alive.
-- ```CONSOLE_DESPAWN``` Despawns when a nearby console is properly hacked.
-- ```IMMOBILE``` Doesn't move (e.g. turrets)
-- ```ID_CARD_DESPAWN``` Despawns when a science ID card is used on a nearby console
-- ```INTERIOR_AMMO``` Monster contains ammo inside itself, no need to load on launch. Prevents ammo from being dropped on disable.
-- ```KEENNOSE``` Keen sense of smell.
-- ```LARVA``` Creature is a larva. Currently used for gib and blood handling.
-- ```LEATHER``` May produce leather when butchered.
-- ```LOUDMOVES``` Mkes move noises as if ~2 sizes louder, even if flying.
-- ```MECH_RECON_VISION``` This mech grants you night-vision and enhanced overmap sight radius when piloted.
-- ```MECH_DEFENSIVE``` This mech can protect you thoroughly when piloted.
-- ```MILITARY_MECH``` Is a military-grade mech.
-- ```MILKABLE``` Produces milk when milked.
-- ```NIGHT_INVISIBILITY``` Monster becomes invisible if it's more than one tile away and the lighting on its tile is LL_LOW or less. Visibility is not affected by night vision.
-- ```NOGIB``` Does not leave gibs / meat chunks when killed with huge damage.
-- ```NOHEAD``` Headshots not allowed!
-- ```NO_BREATHE``` Creature can't drown and is unharmed by gas, smoke or poison.
-- ```NO_BREED``` Creature doesn't reproduce even though it has reproduction data - useful when using copy-from to make child versions of adult creatures
-- ```NO_FUNG_DMG``` Creature is immune to fungal spores and can't be fungalized.
-- ```PAY_BOT``` Creature can be turned into a pet for a limited time in exchange of e-money.
-- ```PET_MOUNTABLE``` Creature can be ridden or attached to an harness.
-- ```PET_HARNESSABLE```Creature can be attached to an harness.
-- ```NULL``` Source use only.
-- ```PACIFIST``` That monster will never do melee attacks.
-- ```PARALYZE``` Attack may paralyze the player with venom.
-- ```PLASTIC``` Absorbs physical damage to a great degree.
-- ```POISON``` Poisonous to eat.
-- ```PUSH_MON``` Can push creatures out of its way.
-- ```QUEEN``` When it dies, local populations start to die off too.
-- ```REVIVES``` Monster corpse will revive after a short period of time.
-- ```RIDEABLE_MECH``` This monster is a mech suit that can be piloted.
-- ```SEES``` It can see you (and will run/follow).
-- ```SHEARABLE``` This monster can be sheared for wool.
-- ```SLUDGEPROOF``` Ignores the effect of sludge trails.
-- ```SLUDGETRAIL``` Causes the monster to leave a sludge trap trail when moving.
-- ```SMELLS``` It can smell you.
-- ```STUMBLES``` Stumbles in its movement.
-- ```SUNDEATH``` Dies in full sunlight.
-- ```SWARMS``` Groups together and form loose packs.
-- ```SWIMS``` Treats water as 50 movement point terrain.
-- ```VENOM``` Attack may poison the player.
-- ```VERMIN``` Obsolete flag for inconsequential monsters, now prevents loading.
-- ```WARM``` Warm blooded.
-- ```WEBWALK``` Doesn't destroy webs.
-- ```WOOL``` May produce wool when butchered.
+- `ABSORBS_SPLITS` Consumes objects it moves over, and if it absorbs enough it will split into a
+ copy.
+- `ABSORBS` Consumes objects it moves over. (Modders use this).
+- `ACIDPROOF` Immune to acid.
+- `ACIDTRAIL` Leaves a trail of acid.
+- `ACID_BLOOD` Makes monster bleed acid. Fun stuff! Does not automatically dissolve in a pool of
+ acid on death.
+- `ANIMAL` Is an _animal_ for purposes of the `Animal Empathy` trait.
+- `AQUATIC` Confined to water.
+- `ARTHROPOD_BLOOD` Forces monster to bleed hemolymph.
+- `ATTACKMON` Attacks other monsters.
+- `BADVENOM` Attack may **severely** poison the player.
+- `BASHES` Bashes down doors.
+- `BILE_BLOOD` Makes monster bleed bile.
+- `BIRDFOOD` Becomes friendly / tamed with bird food.
+- `BLEED` Causes the player to bleed.
+- `BONES` May produce bones and sinews when butchered.
+- `BORES` Tunnels through just about anything (15x bash multiplier: dark wyrms' bash skill 12->180)
+- `CAN_DIG` Can dig _and_ walk.
+- `CAN_OPEN_DOORS` Can open doors on its path.
+- `CANPLAY` This creature can be played with if it's a pet.
+- `CATFOOD` Becomes friendly / tamed with cat food.
+- `CATTLEFODDER` Becomes friendly / tamed with cattle fodder.
+- `CBM_CIV` May produce a common CBM a power CBM when butchered.
+- `CBM_OP` May produce a CBM or two from 'bionics_op' item group when butchered.
+- `CBM_POWER` May produce a power CBM when butchered, independent of CBM.
+- `CBM_SCI` May produce a CBM from 'bionics_sci' item group when butchered.
+- `CBM_SUBS` May produce a CBM or two from bionics_subs and a power CBM when butchered.
+- `CBM_TECH` May produce a CBM or two from 'bionics_tech' item group and a power CBM when butchered.
+- `CHITIN` May produce chitin when butchered.
+- `CLIMBS` Can climb.
+- `COLDROOF` Immune to cold damage.
+- `CURRENT` this water is flowing.
+- `DESTROYS` Bashes down walls and more. (2.5x bash multiplier, where base is the critter's max
+ melee bashing)
+- `DIGS` Digs through the ground.
+- `DOGFOOD` Becomes friendly / tamed with dog food.
+- `DRIPS_GASOLINE` Occasionally drips gasoline on move.
+- `DRIPS_NAPALM` Occasionally drips napalm on move.
+- `ELECTRIC` Shocks unarmed attackers.
+- `ELECTRONIC` e.g. A Robot; affected by emp blasts and other stuff.
+- `FAT` May produce fat when butchered.
+- `FEATHER` May produce feathers when butchered.
+- `FILTHY` Any clothing it drops will be filthy.
+- `FIREPROOF` Immune to fire.
+- `FIREY` Burns stuff and is immune to fire.
+- `FISHABLE` It is fishable.
+- `FLAMMABLE` Monster catches fire, burns, and spreads fire to nearby objects.
+- `FLIES` Can fly (over water, etc.)
+- `FUR` May produce fur when butchered.
+- `GOODHEARING` Pursues sounds more than most monsters.
+- `GRABS` Its attacks may grab you!
+- `GROUP_BASH` Gets help from monsters around it when bashing.
+- `GROUP_MORALE` More courageous when near friends.
+- `GUILT` You feel guilty for killing it.
+- `HARDTOSHOOT` It's one size smaller for ranged attacks, no less then MS_TINY
+- `HEARS` It can hear you.
+- `HIT_AND_RUN` Flee for several turns after a melee attack.
+- `HUMAN` It's a live human, as long as it's alive.
+- `CONSOLE_DESPAWN` Despawns when a nearby console is properly hacked.
+- `IMMOBILE` Doesn't move (e.g. turrets)
+- `ID_CARD_DESPAWN` Despawns when a science ID card is used on a nearby console
+- `INTERIOR_AMMO` Monster contains ammo inside itself, no need to load on launch. Prevents ammo from
+ being dropped on disable.
+- `KEENNOSE` Keen sense of smell.
+- `LARVA` Creature is a larva. Currently used for gib and blood handling.
+- `LEATHER` May produce leather when butchered.
+- `LOUDMOVES` Mkes move noises as if ~2 sizes louder, even if flying.
+- `MECH_RECON_VISION` This mech grants you night-vision and enhanced overmap sight radius when
+ piloted.
+- `MECH_DEFENSIVE` This mech can protect you thoroughly when piloted.
+- `MILITARY_MECH` Is a military-grade mech.
+- `MILKABLE` Produces milk when milked.
+- `NIGHT_INVISIBILITY` Monster becomes invisible if it's more than one tile away and the lighting on
+ its tile is LL_LOW or less. Visibility is not affected by night vision.
+- `NOGIB` Does not leave gibs / meat chunks when killed with huge damage.
+- `NOHEAD` Headshots not allowed!
+- `NO_BREATHE` Creature can't drown and is unharmed by gas, smoke or poison.
+- `NO_BREED` Creature doesn't reproduce even though it has reproduction data - useful when using
+ copy-from to make child versions of adult creatures
+- `NO_FUNG_DMG` Creature is immune to fungal spores and can't be fungalized.
+- `PAY_BOT` Creature can be turned into a pet for a limited time in exchange of e-money.
+- `PET_MOUNTABLE` Creature can be ridden or attached to an harness.
+- `PET_HARNESSABLE`Creature can be attached to an harness.
+- `NULL` Source use only.
+- `PACIFIST` That monster will never do melee attacks.
+- `PARALYZE` Attack may paralyze the player with venom.
+- `PLASTIC` Absorbs physical damage to a great degree.
+- `POISON` Poisonous to eat.
+- `PUSH_MON` Can push creatures out of its way.
+- `QUEEN` When it dies, local populations start to die off too.
+- `REVIVES` Monster corpse will revive after a short period of time.
+- `RIDEABLE_MECH` This monster is a mech suit that can be piloted.
+- `SEES` It can see you (and will run/follow).
+- `SHEARABLE` This monster can be sheared for wool.
+- `SLUDGEPROOF` Ignores the effect of sludge trails.
+- `SLUDGETRAIL` Causes the monster to leave a sludge trap trail when moving.
+- `SMELLS` It can smell you.
+- `STUMBLES` Stumbles in its movement.
+- `SUNDEATH` Dies in full sunlight.
+- `SWARMS` Groups together and form loose packs.
+- `SWIMS` Treats water as 50 movement point terrain.
+- `VENOM` Attack may poison the player.
+- `VERMIN` Obsolete flag for inconsequential monsters, now prevents loading.
+- `WARM` Warm blooded.
+- `WEBWALK` Doesn't destroy webs.
+- `WOOL` May produce wool when butchered.
### Monster Defense and Attacks
-- ```ACIDSPLASH``` Splash acid on the attacker
-- ```NONE``` No special attack-back
-- ```ZAPBACK``` Shock attacker on hit
+- `ACIDSPLASH` Splash acid on the attacker
+- `NONE` No special attack-back
+- `ZAPBACK` Shock attacker on hit
### Sizes
-- ```HUGE``` Tank
-- ```LARGE``` Cow
-- ```MEDIUM``` Human
-- ```SMALL``` Dog
-- ```TINY``` Squirrel
+- `HUGE` Tank
+- `LARGE` Cow
+- `MEDIUM` Human
+- `SMALL` Dog
+- `TINY` Squirrel
### Special attacks
-Some special attacks are also valid use actions for tools and weapons.
-See `monsters.json` for examples on how to use these attacks.
-Also see `monster_attacks.json` for more special attacks, for example, impale and scratch.
-
-- ```ACID_ACCURATE``` Shoots acid that is accurate at long ranges, but less so up close.
-- ```ACID_BARF``` Barfs corroding, blinding acid.
-- ```ACID``` Spit acid.
-- ```ANTQUEEN``` Hatches/grows: `egg > ant > soldier`.
-- ```BIO_OP_BIOJUTSU``` Attack with a random special martial art maneuver.
-- ```BIO_OP_TAKEDOWN``` Attack with special martial art takedown maneuver.
-- ```BIO_OP_DISARM``` Attack with a special martial art disarm maneuver.
-- ```BIO_OP_IMPALE``` Attack with a strong martial art maneuver.
-- ```BITE``` Bite attack that can cause deep infected wounds.
-- ```BMG_TUR``` Barrett .50BMG rifle fires.
-- ```BOOMER_GLOW``` Spit glowing bile.
-- ```BOOMER``` Spit bile.
-- ```BRANDISH``` Brandish a knife at the player.
-- ```BREATHE``` Spawns a `breather`
-- ```CALLBLOBS``` Calls 2/3 of nearby blobs to defend this monster, and sends 1/3 of nearby blobs after the player.
-- ```CHICKENBOT``` LEGACY - Robot can attack with tazer, M4, or MGL depending on distance.
-- ```COPBOT``` Cop-bot alerts and then tazes the player.
-- ```DANCE``` Monster dances.
-- ```DARKMAN``` Can cause darkness and wraiths to spawn.
-- ```DERMATIK_GROWTH``` Dermatik larva grows into an adult.
-- ```DERMATIK``` Attempts to lay dermatik eggs in the player.
-- ```DISAPPEAR``` Hallucination disappears.
-- ```DOGTHING``` The dog _thing_ spawns into a tentacle dog.
-- ```FEAR_PARALYZE``` Paralyze the player with fear.
-- ```FLAMETHROWER``` Shoots a stream of fire.
-- ```FLESH_GOLEM``` Attack the player with claw, and inflict disease `downed` if the attack connects.
-- ```FORMBLOB``` Spawns blobs?
-- ```FRAG_TUR``` MGL fires frag rounds.
-- ```FUNGUS_BIG_BLOSSOM``` Spreads fire suppressing fungal haze.
-- ```FUNGUS_BRISTLE``` Perform barbed tendril attack that can cause fungal infections.
-- ```FUNGUS_CORPORATE``` Used solely by Crazy Cataclysm. This will cause runtime errors if used without out, and spawns SpOreos on top of the creature.
-- ```FUNGUS_FORTIFY``` Grows Fungal hedgerows, and advances player on the mycus threshold path.
-- ```FUNGUS_GROWTH``` Grows a young fungaloid into an adult.
-- ```FUNGUS_HAZE``` Spawns fungal fields.
-- ```FUNGUS_INJECT``` Perform needle attack that can cause fungal infections.
-- ```FUNGUS_SPROUT``` Grows a fungal wall.
-- ```FUNGUS``` Releases fungal spores and attempts to infect the player.
-- ```GENERATOR``` Regenerates health.
-- ```GENE_STING``` Shoot a dart at the player that causes a mutation if it connects.
-- ```GRAB``` GRAB the target, and drag it around.
-- ```GRAB``` Grabs the player, slowing on hit, making movement and dodging impossible and blocking harder.
-- ```GROWPLANTS``` Spawns underbrush, or promotes it to `> young tree > tree`.
-- ```GROW_VINE``` Grows creeper vines.
-- ```HOWL``` "an ear-piercing howl!"
-- ```JACKSON``` Converts zombies into zombie dancers.
-- ```LASER``` Laser turret fires.
-- ```LEAP``` leap away to an unobstructed tile.
-- ```LONGSWIPE``` Does high damage claw attack, which can even hit some distance away.
-- ```LUNGE``` Perform a jumping attack from some distance away, which can down the target.
-- ```MULTI_ROBOT``` Robot can attack with tazer, flamethrower, M4, MGL, or 120mm cannon depending on distance.
-- ```NONE``` No special attack.
-- ```PARA_STING``` Shoot a paralyzing dart at the player.
-- ```PARROT``` Parrots the speech defined in `speech.json`, picks one of the lines randomly. "speaker" points to a monster id.
-- ```PARROT_AT_DANGER``` Performs the same function as PARROT, but only if the creature sees an angry monster from a hostile faction.
-- ```PAID_BOT``` For creature with PAY_BOT flag, removes the ally status when the pet effect runs out.
-- ```PHOTOGRAPH``` Photograph the player. Causes a robot attack?
-- ```PLANT``` Fungal spores take seed and grow into a fungaloid.
-- ```PULL_METAL_WEAPON``` Pull weapon that's made of iron or steel from the player's hand.
-- ```RANGED_PULL``` Pull targets towards attacker.
-- ```RATKING``` Inflicts disease `rat`
-- ```RATTLE``` "a sibilant rattling sound!"
-- ```RESURRECT``` Revives the dead--again.
-- ```RIFLE_TUR``` Rifle turret fires.
-- ```RIOTBOT``` Sprays teargas or relaxation gas, can handcuff players, and can use a blinding flash.
-- ```SCIENCE``` Various science/technology related attacks (e.g. manhacks, radioactive beams, etc.)
-- ```SEARCHLIGHT``` Tracks targets with a searchlight.
-- ```SHOCKING_REVEAL``` Shoots bolts of lightning, and reveals a SHOCKING FACT! Very fourth-wall breaking. Used solely by Crazy Cataclysm.
-- ```SHOCKSTORM``` Shoots bolts of lightning.
-- ```SHRIEK_ALERT``` "a very terrible shriek!"
-- ```SHRIEK_STUN``` "a stunning shriek!", causes a small bash, can cause a stun.
-- ```SHRIEK``` "a terrible shriek!"
-- ```SLIMESPRING``` Can provide a morale boost to the player, and cure bite and bleed effects.
-- ```SMASH``` Smashes the target for massive damage, sending it flying for a number of tiles equal to `("melee_dice" * "melee_dice_sides" * 3) / 10`.
-- ```SMG``` SMG turret fires.
-- ```SPIT_SAP``` Spit sap.
-- ```STARE``` Stare at the player and inflict teleglow.
-- ```STRETCH_ATTACK``` Long ranged piercing attack.
-- ```STRETCH_BITE``` Long ranged bite attack.
-- ```SUICIDE``` Dies after attacking.
-- ```TAZER``` Shock the player.
-- ```TENTACLE``` Lashes a tentacle at the player.
-- ```TRIFFID_GROWTH``` Young triffid grows into an adult.
-- ```TRIFFID_HEARTBEAT``` Grows and crumbles root walls around the player, and spawns more monsters.
-- ```UPGRADE``` Upgrades a regular zombie into a special zombie.
-- ```VINE``` Attacks with vine.
-- ```VORTEX``` Forms a vortex/tornado that causes damage and throws creatures around.
-
+Some special attacks are also valid use actions for tools and weapons. See `monsters.json` for
+examples on how to use these attacks. Also see `monster_attacks.json` for more special attacks, for
+example, impale and scratch.
+
+- `ACID_ACCURATE` Shoots acid that is accurate at long ranges, but less so up close.
+- `ACID_BARF` Barfs corroding, blinding acid.
+- `ACID` Spit acid.
+- `ANTQUEEN` Hatches/grows: `egg > ant > soldier`.
+- `BIO_OP_BIOJUTSU` Attack with a random special martial art maneuver.
+- `BIO_OP_TAKEDOWN` Attack with special martial art takedown maneuver.
+- `BIO_OP_DISARM` Attack with a special martial art disarm maneuver.
+- `BIO_OP_IMPALE` Attack with a strong martial art maneuver.
+- `BITE` Bite attack that can cause deep infected wounds.
+- `BMG_TUR` Barrett .50BMG rifle fires.
+- `BOOMER_GLOW` Spit glowing bile.
+- `BOOMER` Spit bile.
+- `BRANDISH` Brandish a knife at the player.
+- `BREATHE` Spawns a `breather`
+- `CALLBLOBS` Calls 2/3 of nearby blobs to defend this monster, and sends 1/3 of nearby blobs after
+ the player.
+- `CHICKENBOT` LEGACY - Robot can attack with tazer, M4, or MGL depending on distance.
+- `COPBOT` Cop-bot alerts and then tazes the player.
+- `DANCE` Monster dances.
+- `DARKMAN` Can cause darkness and wraiths to spawn.
+- `DERMATIK_GROWTH` Dermatik larva grows into an adult.
+- `DERMATIK` Attempts to lay dermatik eggs in the player.
+- `DISAPPEAR` Hallucination disappears.
+- `DOGTHING` The dog _thing_ spawns into a tentacle dog.
+- `FEAR_PARALYZE` Paralyze the player with fear.
+- `FLAMETHROWER` Shoots a stream of fire.
+- `FLESH_GOLEM` Attack the player with claw, and inflict disease `downed` if the attack connects.
+- `FORMBLOB` Spawns blobs?
+- `FRAG_TUR` MGL fires frag rounds.
+- `FUNGUS_BIG_BLOSSOM` Spreads fire suppressing fungal haze.
+- `FUNGUS_BRISTLE` Perform barbed tendril attack that can cause fungal infections.
+- `FUNGUS_CORPORATE` Used solely by Crazy Cataclysm. This will cause runtime errors if used without
+ out, and spawns SpOreos on top of the creature.
+- `FUNGUS_FORTIFY` Grows Fungal hedgerows, and advances player on the mycus threshold path.
+- `FUNGUS_GROWTH` Grows a young fungaloid into an adult.
+- `FUNGUS_HAZE` Spawns fungal fields.
+- `FUNGUS_INJECT` Perform needle attack that can cause fungal infections.
+- `FUNGUS_SPROUT` Grows a fungal wall.
+- `FUNGUS` Releases fungal spores and attempts to infect the player.
+- `GENERATOR` Regenerates health.
+- `GENE_STING` Shoot a dart at the player that causes a mutation if it connects.
+- `GRAB` GRAB the target, and drag it around.
+- `GRAB` Grabs the player, slowing on hit, making movement and dodging impossible and blocking
+ harder.
+- `GROWPLANTS` Spawns underbrush, or promotes it to `> young tree > tree`.
+- `GROW_VINE` Grows creeper vines.
+- `HOWL` "an ear-piercing howl!"
+- `JACKSON` Converts zombies into zombie dancers.
+- `LASER` Laser turret fires.
+- `LEAP` leap away to an unobstructed tile.
+- `LONGSWIPE` Does high damage claw attack, which can even hit some distance away.
+- `LUNGE` Perform a jumping attack from some distance away, which can down the target.
+- `MULTI_ROBOT` Robot can attack with tazer, flamethrower, M4, MGL, or 120mm cannon depending on
+ distance.
+- `NONE` No special attack.
+- `PARA_STING` Shoot a paralyzing dart at the player.
+- `PARROT` Parrots the speech defined in `speech.json`, picks one of the lines randomly. "speaker"
+ points to a monster id.
+- `PARROT_AT_DANGER` Performs the same function as PARROT, but only if the creature sees an angry
+ monster from a hostile faction.
+- `PAID_BOT` For creature with PAY_BOT flag, removes the ally status when the pet effect runs out.
+- `PHOTOGRAPH` Photograph the player. Causes a robot attack?
+- `PLANT` Fungal spores take seed and grow into a fungaloid.
+- `PULL_METAL_WEAPON` Pull weapon that's made of iron or steel from the player's hand.
+- `RANGED_PULL` Pull targets towards attacker.
+- `RATKING` Inflicts disease `rat`
+- `RATTLE` "a sibilant rattling sound!"
+- `RESURRECT` Revives the dead--again.
+- `RIFLE_TUR` Rifle turret fires.
+- `RIOTBOT` Sprays teargas or relaxation gas, can handcuff players, and can use a blinding flash.
+- `SCIENCE` Various science/technology related attacks (e.g. manhacks, radioactive beams, etc.)
+- `SEARCHLIGHT` Tracks targets with a searchlight.
+- `SHOCKING_REVEAL` Shoots bolts of lightning, and reveals a SHOCKING FACT! Very fourth-wall
+ breaking. Used solely by Crazy Cataclysm.
+- `SHOCKSTORM` Shoots bolts of lightning.
+- `SHRIEK_ALERT` "a very terrible shriek!"
+- `SHRIEK_STUN` "a stunning shriek!", causes a small bash, can cause a stun.
+- `SHRIEK` "a terrible shriek!"
+- `SLIMESPRING` Can provide a morale boost to the player, and cure bite and bleed effects.
+- `SMASH` Smashes the target for massive damage, sending it flying for a number of tiles equal to
+ `("melee_dice" * "melee_dice_sides" * 3) / 10`.
+- `SMG` SMG turret fires.
+- `SPIT_SAP` Spit sap.
+- `STARE` Stare at the player and inflict teleglow.
+- `STRETCH_ATTACK` Long ranged piercing attack.
+- `STRETCH_BITE` Long ranged bite attack.
+- `SUICIDE` Dies after attacking.
+- `TAZER` Shock the player.
+- `TENTACLE` Lashes a tentacle at the player.
+- `TRIFFID_GROWTH` Young triffid grows into an adult.
+- `TRIFFID_HEARTBEAT` Grows and crumbles root walls around the player, and spawns more monsters.
+- `UPGRADE` Upgrades a regular zombie into a special zombie.
+- `VINE` Attacks with vine.
+- `VORTEX` Forms a vortex/tornado that causes damage and throws creatures around.
## Mutations
#### Flags
-- ```NO_RADIATION``` This mutation grants immunity to radiations.
-- ```NO_THIRST``` Your thirst is not modified by food or drinks.
-- ```UNARMED_BONUS``` You get a bonus to unarmed bash and cut damage equal to unarmed_skill/2 up to 4.
-
+- `NO_RADIATION` This mutation grants immunity to radiations.
+- `NO_THIRST` Your thirst is not modified by food or drinks.
+- `UNARMED_BONUS` You get a bonus to unarmed bash and cut damage equal to unarmed_skill/2 up to 4.
### Categories
These branches are also the valid entries for the categories of `dreams` in `dreams.json`
-- ```MUTCAT_ALPHA``` "You feel...better. Somehow."
-- ```MUTCAT_BEAST``` "Your heart races and you see blood for a moment."
-- ```MUTCAT_BIRD``` "Your body lightens and you long for the sky."
-- ```MUTCAT_CATTLE``` "Your mind and body slow down. You feel peaceful."
-- ```MUTCAT_CEPHALOPOD``` "Your mind is overcome by images of eldritch horrors...and then they pass."
-- ```MUTCAT_CHIMERA``` "You need to roar, bask, bite, and flap. NOW."
-- ```MUTCAT_ELFA``` "Nature is becoming one with you..."
-- ```MUTCAT_FISH``` "You are overcome by an overwhelming longing for the ocean."
-- ```MUTCAT_INSECT``` "You hear buzzing, and feel your body harden."
-- ```MUTCAT_LIZARD``` "For a heartbeat, your body cools down."
-- ```MUTCAT_MEDICAL``` "Your can feel the blood rushing through your veins and a strange, medicated feeling washes over your senses."
-- ```MUTCAT_PLANT``` "You feel much closer to nature."
-- ```MUTCAT_RAPTOR``` "Mmm...sweet bloody flavor...tastes like victory."
-- ```MUTCAT_RAT``` "You feel a momentary nausea."
-- ```MUTCAT_SLIME``` "Your body loses all rigidity for a moment."
-- ```MUTCAT_SPIDER``` "You feel insidious."
-- ```MUTCAT_TROGLOBITE``` "You yearn for a cool, dark place to hide."
-
+- `MUTCAT_ALPHA` "You feel...better. Somehow."
+- `MUTCAT_BEAST` "Your heart races and you see blood for a moment."
+- `MUTCAT_BIRD` "Your body lightens and you long for the sky."
+- `MUTCAT_CATTLE` "Your mind and body slow down. You feel peaceful."
+- `MUTCAT_CEPHALOPOD` "Your mind is overcome by images of eldritch horrors...and then they pass."
+- `MUTCAT_CHIMERA` "You need to roar, bask, bite, and flap. NOW."
+- `MUTCAT_ELFA` "Nature is becoming one with you..."
+- `MUTCAT_FISH` "You are overcome by an overwhelming longing for the ocean."
+- `MUTCAT_INSECT` "You hear buzzing, and feel your body harden."
+- `MUTCAT_LIZARD` "For a heartbeat, your body cools down."
+- `MUTCAT_MEDICAL` "Your can feel the blood rushing through your veins and a strange, medicated
+ feeling washes over your senses."
+- `MUTCAT_PLANT` "You feel much closer to nature."
+- `MUTCAT_RAPTOR` "Mmm...sweet bloody flavor...tastes like victory."
+- `MUTCAT_RAT` "You feel a momentary nausea."
+- `MUTCAT_SLIME` "Your body loses all rigidity for a moment."
+- `MUTCAT_SPIDER` "You feel insidious."
+- `MUTCAT_TROGLOBITE` "You yearn for a cool, dark place to hide."
## Overmap
### Overmap connections
-- ```ORTHOGONAL``` The connection generally prefers straight lines, avoids turning wherever possible.
+- `ORTHOGONAL` The connection generally prefers straight lines, avoids turning wherever possible.
### Overmap specials
#### Flags
-- ```BEE``` Location is related to bees. Used to classify location.
-- ```BLOB``` Location should "blob" outward from the defined location with a chance to be placed in adjacent locations.
-- ```CLASSIC``` Location is allowed when classic zombies are enabled.
-- ```FUNGAL``` Location is related to fungi. Used to classify location.
-- ```TRIFFID``` Location is related to triffids. Used to classify location.
-- ```LAKE``` Location is is placed on a lake and will be ignored for placement if the overmap doesn't contain any lake terrain.
-- ```UNIQUE``` Location is unique and will only occur once per overmap. `occurrences` is overridden to define a percent chance (e.g. `"occurrences" : [75, 100]` is 75%)
+- `BEE` Location is related to bees. Used to classify location.
+- `BLOB` Location should "blob" outward from the defined location with a chance to be placed in
+ adjacent locations.
+- `CLASSIC` Location is allowed when classic zombies are enabled.
+- `FUNGAL` Location is related to fungi. Used to classify location.
+- `TRIFFID` Location is related to triffids. Used to classify location.
+- `LAKE` Location is is placed on a lake and will be ignored for placement if the overmap doesn't
+ contain any lake terrain.
+- `UNIQUE` Location is unique and will only occur once per overmap. `occurrences` is overridden to
+ define a percent chance (e.g. `"occurrences" : [75, 100]` is 75%)
### Overmap terrains
#### Flags
-- ```KNOWN_DOWN``` There's a known way down.
-- ```KNOWN_UP``` There's a known way up.
-- ```LINEAR``` For roads etc, which use ID_straight, ID_curved, ID_tee, ID_four_way.
-- ```NO_ROTATE``` The terrain can't be rotated (ID_north, ID_east, ID_south, and ID_west instances will NOT be generated, just ID).
-- ```RIVER``` It's a river tile.
-- ```SIDEWALK``` Has sidewalks on the sides adjacent to roads.
-- ```LAKE``` Consider this location to be a valid lake terrain for mapgen purposes.
-- ```LAKE_SHORE``` Consider this location to be a valid lake shore terrain for mapgen purposes.
-- ```SOURCE_FUEL``` For NPC AI, this location may contain fuel for looting.
-- ```SOURCE_FOOD``` For NPC AI, this location may contain food for looting.
-- ```SOURCE_FARMING``` For NPC AI, this location may contain useful farming supplies for looting.
-- ```SOURCE_FABRICATION``` For NPC AI, this location may contain fabrication tools and components for looting.
-- ```SOURCE_GUN``` For NPC AI, this location may contain guns for looting.
-- ```SOURCE_AMMO``` For NPC AI, this location may contain ammo for looting.
-- ```SOURCE_BOOKS``` For NPC AI, this location may contain books for looting.
-- ```SOURCE_WEAPON``` For NPC AI, this location may contain weapons for looting.
-- ```SOURCE_FORAGE``` For NPC AI, this location may contain plants to forage.
-- ```SOURCE_COOKING``` For NPC AI, this location may contain useful tools and ingredients to aid in cooking.
-- ```SOURCE_TAILORING``` For NPC AI, this location may contain useful tools for tailoring.
-- ```SOURCE_DRINK``` For NPC AI, this location may contain drink for looting.
-- ```SOURCE_VEHICLES``` For NPC AI, this location may contain vehicles/parts/vehicle tools, to loot.
-- ```SOURCE_ELECTRONICS``` For NPC AI, this location may contain useful electronics to loot.
-- ```SOURCE_CONSTRUCTION``` For NPC AI, this location may contain useful tools/components for construction.
-- ```SOURCE_CHEMISTRY``` For NPC AI, this location may contain useful chemistry tools/components.
-- ```SOURCE_CLOTHING``` For NPC AI, this location may contain useful clothing to loot.
-- ```SOURCE_SAFETY``` For NPC AI, this location may be safe/sheltered and a good place for a base.
-- ```SOURCE_ANIMALS``` For NPC AI, this location may contain useful animals for farming/riding.
-- ```SOURCE_MEDICINE``` For NPC AI, this location may contain useful medicines for looting.
-- ```SOURCE_LUXURY``` For NPC AI, this location may contain valuable/feel-good items to sell/keep.
-- ```SOURCE_PEOPLE``` For NPC AI, this location may have other survivors.
-- ```RISK_HIGH``` For NPC AI, this location has a high risk associated with it - labs/superstores etc.
-- ```RISK_LOW``` For NPC AI, this location is secluded and remote, and appears to be safe.
-- ```GENERIC_LOOT``` This is a place that may contain any of the above, but at a lower frequency - usually a house.
+- `KNOWN_DOWN` There's a known way down.
+- `KNOWN_UP` There's a known way up.
+- `LINEAR` For roads etc, which use ID_straight, ID_curved, ID_tee, ID_four_way.
+- `NO_ROTATE` The terrain can't be rotated (ID_north, ID_east, ID_south, and ID_west instances will
+ NOT be generated, just ID).
+- `RIVER` It's a river tile.
+- `SIDEWALK` Has sidewalks on the sides adjacent to roads.
+- `LAKE` Consider this location to be a valid lake terrain for mapgen purposes.
+- `LAKE_SHORE` Consider this location to be a valid lake shore terrain for mapgen purposes.
+- `SOURCE_FUEL` For NPC AI, this location may contain fuel for looting.
+- `SOURCE_FOOD` For NPC AI, this location may contain food for looting.
+- `SOURCE_FARMING` For NPC AI, this location may contain useful farming supplies for looting.
+- `SOURCE_FABRICATION` For NPC AI, this location may contain fabrication tools and components for
+ looting.
+- `SOURCE_GUN` For NPC AI, this location may contain guns for looting.
+- `SOURCE_AMMO` For NPC AI, this location may contain ammo for looting.
+- `SOURCE_BOOKS` For NPC AI, this location may contain books for looting.
+- `SOURCE_WEAPON` For NPC AI, this location may contain weapons for looting.
+- `SOURCE_FORAGE` For NPC AI, this location may contain plants to forage.
+- `SOURCE_COOKING` For NPC AI, this location may contain useful tools and ingredients to aid in
+ cooking.
+- `SOURCE_TAILORING` For NPC AI, this location may contain useful tools for tailoring.
+- `SOURCE_DRINK` For NPC AI, this location may contain drink for looting.
+- `SOURCE_VEHICLES` For NPC AI, this location may contain vehicles/parts/vehicle tools, to loot.
+- `SOURCE_ELECTRONICS` For NPC AI, this location may contain useful electronics to loot.
+- `SOURCE_CONSTRUCTION` For NPC AI, this location may contain useful tools/components for
+ construction.
+- `SOURCE_CHEMISTRY` For NPC AI, this location may contain useful chemistry tools/components.
+- `SOURCE_CLOTHING` For NPC AI, this location may contain useful clothing to loot.
+- `SOURCE_SAFETY` For NPC AI, this location may be safe/sheltered and a good place for a base.
+- `SOURCE_ANIMALS` For NPC AI, this location may contain useful animals for farming/riding.
+- `SOURCE_MEDICINE` For NPC AI, this location may contain useful medicines for looting.
+- `SOURCE_LUXURY` For NPC AI, this location may contain valuable/feel-good items to sell/keep.
+- `SOURCE_PEOPLE` For NPC AI, this location may have other survivors.
+- `RISK_HIGH` For NPC AI, this location has a high risk associated with it - labs/superstores etc.
+- `RISK_LOW` For NPC AI, this location is secluded and remote, and appears to be safe.
+- `GENERIC_LOOT` This is a place that may contain any of the above, but at a lower frequency -
+ usually a house.
## Recipes
### Categories
-- ```CC_AMMO```
-- ```CC_ARMOR```
-- ```CC_CHEM```
-- ```CC_DRINK```
-- ```CC_ELECTRONIC```
-- ```CC_FOOD```
-- ```CC_MISC```
-- ```CC_WEAPON```
+- `CC_AMMO`
+- `CC_ARMOR`
+- `CC_CHEM`
+- `CC_DRINK`
+- `CC_ELECTRONIC`
+- `CC_FOOD`
+- `CC_MISC`
+- `CC_WEAPON`
### Flags
-- ```ALLOW_ROTTEN``` Explicitly allow rotten components when crafting non-perishables.
-- ```BLIND_EASY``` Easy to craft with little to no light.
-- ```BLIND_HARD``` Possible to craft with little to no light, but difficult.
-- ```SECRET``` Not automatically learned at character creation time based on high skill levels.
-- ```UNCRAFT_LIQUIDS_CONTAINED``` Spawn liquid items in its default container.
-- ```FULL_MAGAZINE``` If this recipe requires magazines, it needs one that is full. For deconstruction recipes, it will spawn a full magazine when deconstructed.
-
+- `ALLOW_ROTTEN` Explicitly allow rotten components when crafting non-perishables.
+- `BLIND_EASY` Easy to craft with little to no light.
+- `BLIND_HARD` Possible to craft with little to no light, but difficult.
+- `SECRET` Not automatically learned at character creation time based on high skill levels.
+- `UNCRAFT_LIQUIDS_CONTAINED` Spawn liquid items in its default container.
+- `FULL_MAGAZINE` If this recipe requires magazines, it needs one that is full. For deconstruction
+ recipes, it will spawn a full magazine when deconstructed.
## Scenarios
### Flags
-- ```ALLOW_OUTSIDE``` Allows placing player outside of building, useful for outdoor start.
-- ```BAD_DAY``` Player starts the game drunk, depressed and sick with the flu.
-- ```BOARDED``` Start in boarded building (windows and doors are boarded, movable furniture is moved to windows and doors).
-- ```BORDERED``` Initial start location is bordered by an enormous wall of solid rock.
-- ```CHALLENGE``` Game won't choose this scenario in random game types.
-- ```CITY_START``` Scenario is available only when city size value in world options is more than 0.
-- ```FIRE_START``` Player starts the game with fire nearby.
-- ```HELI_CRASH``` Player starts the game with various limbs wounds.
-- ```INFECTED``` Player starts the game infected.
-- ```LONE_START``` If starting NPC spawn option is switched to "Scenario-based", this scenario won't spawn a fellow NPC on game start.
-- ```SCEN_ONLY``` Profession can be chosen only as part of the appropriate scenario.
-- ```SUR_START``` Surrounded start, zombies outside the starting location.
+- `ALLOW_OUTSIDE` Allows placing player outside of building, useful for outdoor start.
+- `BAD_DAY` Player starts the game drunk, depressed and sick with the flu.
+- `BOARDED` Start in boarded building (windows and doors are boarded, movable furniture is moved to
+ windows and doors).
+- `BORDERED` Initial start location is bordered by an enormous wall of solid rock.
+- `CHALLENGE` Game won't choose this scenario in random game types.
+- `CITY_START` Scenario is available only when city size value in world options is more than 0.
+- `FIRE_START` Player starts the game with fire nearby.
+- `HELI_CRASH` Player starts the game with various limbs wounds.
+- `INFECTED` Player starts the game infected.
+- `LONE_START` If starting NPC spawn option is switched to "Scenario-based", this scenario won't
+ spawn a fellow NPC on game start.
+- `SCEN_ONLY` Profession can be chosen only as part of the appropriate scenario.
+- `SUR_START` Surrounded start, zombies outside the starting location.
#### Season Flags
-- ```AUT_START``` ... start in autumn.
-- ```SPR_START``` ... start in spring.
-- ```SUM_ADV_START``` ... start second summer after Cataclysm.
-- ```SUM_START``` ... start in summer.
-- ```WIN_START``` ... start in winter.
-
+- `AUT_START` ... start in autumn.
+- `SPR_START` ... start in spring.
+- `SUM_ADV_START` ... start second summer after Cataclysm.
+- `SUM_START` ... start in summer.
+- `WIN_START` ... start in winter.
## Skills
### Tags
-- ```combat_skill``` The skill is considered a combat skill. It's affected by "PACIFIST", "PRED1", "PRED2", "PRED3", and "PRED4" traits.
-- ```contextual_skill``` The skill is abstract, it depends on context (an indirect item to which it's applied). Neither player nor NPCs can possess it.
-
+- `combat_skill` The skill is considered a combat skill. It's affected by "PACIFIST", "PRED1",
+ "PRED2", "PRED3", and "PRED4" traits.
+- `contextual_skill` The skill is abstract, it depends on context (an indirect item to which it's
+ applied). Neither player nor NPCs can possess it.
## Techniques
@@ -1264,19 +1410,30 @@ Techniques may be used by tools, armors, weapons and anything else that can be w
### WBLOCK_X
-The following weapon techniques have some additional usage. These are defensive techniques that allow the item to assist in blocking attacks in melee, with some additional special uses.
-
-- ```WBLOCK_1``` "Medium blocking ability"
-- ```WBLOCK_2``` "High blocking ability"
-- ```WBLOCK_3``` "Very high blocking ability"
-
-An item with one of these techniques can be wielded to provide a bonus to damage reduced by blocking compared, or armor with the `BLOCK_WHILE_WORN` flag can also provide the use of this bonus while wearing the item, serving as a shield. Additionally, wielding or wearing an item with a combination of one of these techniques plus said flag will allow the item to block projectiles aimed at body parts the item otherwise does not cover (or is not covering, in the case of wielded items that meet those prerequisites). The chance that this will happen is based on the `coverage` percentage of the item used for its normal armor value, reduced by a penalty that depends on which blocking technique it possesses. The chance of it intercepting shots that strike the legs (again, unless the armor was set to cover the legs by default already, in which case it uses `coverage` as normal) is furtther penalized. The feet will always be vulnerable unless (for whatever reason a JSON author may devise, forcefield items for example) an item happens to be a shield that already covers the feet as armor.
-
- Technique | Chance to intercept (head, torso, opposing arm, etc) | Chance to intercept (legs)
------------|------------------------------------------------------|-------------------------------
- WBLOCK_1 | 90% of default coverage value | 75% of default coverage value
- WBLOCK_2 | 90% of default coverage value | 75% of default coverage value
- WBLOCK_3 | 90% of default coverage value | 75% of default coverage value
+The following weapon techniques have some additional usage. These are defensive techniques that
+allow the item to assist in blocking attacks in melee, with some additional special uses.
+
+- `WBLOCK_1` "Medium blocking ability"
+- `WBLOCK_2` "High blocking ability"
+- `WBLOCK_3` "Very high blocking ability"
+
+An item with one of these techniques can be wielded to provide a bonus to damage reduced by blocking
+compared, or armor with the `BLOCK_WHILE_WORN` flag can also provide the use of this bonus while
+wearing the item, serving as a shield. Additionally, wielding or wearing an item with a combination
+of one of these techniques plus said flag will allow the item to block projectiles aimed at body
+parts the item otherwise does not cover (or is not covering, in the case of wielded items that meet
+those prerequisites). The chance that this will happen is based on the `coverage` percentage of the
+item used for its normal armor value, reduced by a penalty that depends on which blocking technique
+it possesses. The chance of it intercepting shots that strike the legs (again, unless the armor was
+set to cover the legs by default already, in which case it uses `coverage` as normal) is furtther
+penalized. The feet will always be vulnerable unless (for whatever reason a JSON author may devise,
+forcefield items for example) an item happens to be a shield that already covers the feet as armor.
+
+| Technique | Chance to intercept (head, torso, opposing arm, etc) | Chance to intercept (legs) |
+| --------- | ---------------------------------------------------- | ----------------------------- |
+| WBLOCK_1 | 90% of default coverage value | 75% of default coverage value |
+| WBLOCK_2 | 90% of default coverage value | 75% of default coverage value |
+| WBLOCK_3 | 90% of default coverage value | 75% of default coverage value |
## Tools
@@ -1284,218 +1441,261 @@ An item with one of these techniques can be wielded to provide a bonus to damage
Melee flags are fully compatible with tool flags, and vice versa.
-- ```ACT_ON_RANGED_HIT``` The item should activate when thrown or fired, then immediately get processed if it spawns on the ground.
-- ```ALLOWS_REMOTE_USE``` This item can be activated or reloaded from adjacent tile without picking it up.
-- ```BELT_CLIP``` The item can be clipped or hooked on to a belt loop of the appropriate size (belt loops are limited by their max_volume and max_weight properties)
-- ```BOMB``` It can be a remote controlled bomb.
-- ```CABLE_SPOOL``` This item is a cable spool and must be processed as such. It has an internal "state" variable which may be in the states "attach_first" or "pay_out_cable" -- in the latter case, set its charges to `max_charges - dist(here, point(vars["source_x"], vars["source_y"]))`. If this results in 0 or a negative number, set its state back to "attach_first".
-- ```CANNIBALISM``` The item is a food that contains human flesh, and applies all applicable effects when consumed.
-- ```CHARGEDIM``` If illuminated, light intensity fades with charge, starting at 20% charge left.
-- ```DIG_TOOL``` If wielded, digs thorough terrain like rock and walls, as player walks into them. If item also has ```POWERED``` flag, then it digs faster, but uses up the item's ammo as if activating it.
-- ```FIRESTARTER``` Item will start fire with some difficulty.
-- ```FIRE``` Item will start a fire immediately.
-- ```FISH_GOOD``` When used for fishing, it's a good tool (requires that the matching use_action has been set).
-- ```FISH_POOR``` When used for fishing, it's a poor tool (requires that the matching use_action has been set).
-- ```HAS_RECIPE``` Used by the E-Ink tablet to indicates it's currently showing a recipe.
-- ```IS_UPS``` Item is Unified Power Supply. Used in active item processing
-- ```LIGHT_[X]``` Illuminates the area with light intensity `[X]` where `[X]` is an intensity value. (e.x. `LIGHT_4` or `LIGHT_100`). Note: this flags sets `itype::light_emission` field and then is removed (can't be found using `has_flag`);
-- ```MC_MOBILE```, ```MC_RANDOM_STUFF```, ```MC_SCIENCE_STUFF```, ```MC_USED```, ```MC_HAS_DATA``` Memory card related flags, see `iuse.cpp`
-- ```NO_DROP``` Item should never exist on map tile as a discrete item (must be contained by another item)
-- ```NO_UNLOAD``` Cannot be unloaded.
-- ```POWERED``` If turned ON, item uses its own source of power, instead of relying on power of the user
-- ```RADIOCARITEM``` Item can be put into a remote controlled car.
-- ```RADIOSIGNAL_1``` Activated per radios signal 1 (Red).
-- ```RADIOSIGNAL_2``` Activated per radios signal 2 (Blue).
-- ```RADIOSIGNAL_3``` Activated per radios signal 3 (Green).
-- ```RADIO_ACTIVATION``` Item can be activated by a remote control (also requires RADIOSIGNAL_*).
-- ```RADIO_INVOKE_PROC``` After being activated via radio signal the item will have its charges removed. Can be used for bypassing bomb countdown.
-- ```RADIO_CONTROLLABLE``` It can be moved around via a remote control.
-- ```RADIO_MODABLE``` Indicates the item can be made into a radio-activated item.
-- ```RADIO_MOD``` The item has been made into a radio-activated item.
-- ```RECHARGE``` Gain charges when placed in a cargo area with a recharge station.
-- ```SAFECRACK``` This item can be used to unlock safes.
-- ```USES_BIONIC_POWER``` The item has no charges of its own, and runs off of the player's bionic power.
-- ```USE_UPS``` Item is charges from an UPS / it uses the charges of an UPS instead of its own.
-- ```NAT_UPS``` Silences the (UPS) suffix from USE_UPS.
-- ```WATER_EXTINGUISH``` Is extinguishable in water or under precipitation. Converts items (requires "reverts_to" or use_action "transform" to be set).
-- ```WET``` Item is wet and will slowly dry off (e.g. towel).
-- ```WIND_EXTINGUISH``` This item will be extinguished by the wind.
-- ```WRITE_MESSAGE``` This item could be used to write messages on signs.
+- `ACT_ON_RANGED_HIT` The item should activate when thrown or fired, then immediately get processed
+ if it spawns on the ground.
+- `ALLOWS_REMOTE_USE` This item can be activated or reloaded from adjacent tile without picking it
+ up.
+- `BELT_CLIP` The item can be clipped or hooked on to a belt loop of the appropriate size (belt
+ loops are limited by their max_volume and max_weight properties)
+- `BOMB` It can be a remote controlled bomb.
+- `CABLE_SPOOL` This item is a cable spool and must be processed as such. It has an internal "state"
+ variable which may be in the states "attach_first" or "pay_out_cable" -- in the latter case, set
+ its charges to `max_charges - dist(here, point(vars["source_x"], vars["source_y"]))`. If this
+ results in 0 or a negative number, set its state back to "attach_first".
+- `CANNIBALISM` The item is a food that contains human flesh, and applies all applicable effects
+ when consumed.
+- `CHARGEDIM` If illuminated, light intensity fades with charge, starting at 20% charge left.
+- `DIG_TOOL` If wielded, digs thorough terrain like rock and walls, as player walks into them. If
+ item also has `POWERED` flag, then it digs faster, but uses up the item's ammo as if activating
+ it.
+- `FIRESTARTER` Item will start fire with some difficulty.
+- `FIRE` Item will start a fire immediately.
+- `FISH_GOOD` When used for fishing, it's a good tool (requires that the matching use_action has
+ been set).
+- `FISH_POOR` When used for fishing, it's a poor tool (requires that the matching use_action has
+ been set).
+- `HAS_RECIPE` Used by the E-Ink tablet to indicates it's currently showing a recipe.
+- `IS_UPS` Item is Unified Power Supply. Used in active item processing
+- `LIGHT_[X]` Illuminates the area with light intensity `[X]` where `[X]` is an intensity value.
+ (e.x. `LIGHT_4` or `LIGHT_100`). Note: this flags sets `itype::light_emission` field and then is
+ removed (can't be found using `has_flag`);
+- `MC_MOBILE`, `MC_RANDOM_STUFF`, `MC_SCIENCE_STUFF`, `MC_USED`, `MC_HAS_DATA` Memory card related
+ flags, see `iuse.cpp`
+- `NO_DROP` Item should never exist on map tile as a discrete item (must be contained by another
+ item)
+- `NO_UNLOAD` Cannot be unloaded.
+- `POWERED` If turned ON, item uses its own source of power, instead of relying on power of the user
+- `RADIOCARITEM` Item can be put into a remote controlled car.
+- `RADIOSIGNAL_1` Activated per radios signal 1 (Red).
+- `RADIOSIGNAL_2` Activated per radios signal 2 (Blue).
+- `RADIOSIGNAL_3` Activated per radios signal 3 (Green).
+- `RADIO_ACTIVATION` Item can be activated by a remote control (also requires RADIOSIGNAL_*).
+- `RADIO_INVOKE_PROC` After being activated via radio signal the item will have its charges removed.
+ Can be used for bypassing bomb countdown.
+- `RADIO_CONTROLLABLE` It can be moved around via a remote control.
+- `RADIO_MODABLE` Indicates the item can be made into a radio-activated item.
+- `RADIO_MOD` The item has been made into a radio-activated item.
+- `RECHARGE` Gain charges when placed in a cargo area with a recharge station.
+- `SAFECRACK` This item can be used to unlock safes.
+- `USES_BIONIC_POWER` The item has no charges of its own, and runs off of the player's bionic power.
+- `USE_UPS` Item is charges from an UPS / it uses the charges of an UPS instead of its own.
+- `NAT_UPS` Silences the (UPS) suffix from USE_UPS.
+- `WATER_EXTINGUISH` Is extinguishable in water or under precipitation. Converts items (requires
+ "reverts_to" or use_action "transform" to be set).
+- `WET` Item is wet and will slowly dry off (e.g. towel).
+- `WIND_EXTINGUISH` This item will be extinguished by the wind.
+- `WRITE_MESSAGE` This item could be used to write messages on signs.
### Flags that apply to items
These flags **do not apply to item types**.
-Those flags are added by the game code to specific items (that specific welder, not *all* welders).
-
-- ```COLD``` Item is cold (see EATEN_COLD).
-- ```DIRTY``` Item (liquid) was dropped on the ground and is now irreparably dirty.
-- ```FIELD_DRESS_FAILED``` Corpse was damaged by unskillful field dressing. Affects butcher results.
-- ```FIELD_DRESS``` Corpse was field dressed. Affects butcher results.
-- ```FIT``` Reduces encumbrance by one.
-- ```FROZEN``` Item is frozen solid (used by freezer).
-- ```HIDDEN_ITEM``` This item cannot be seen in AIM.
-- ```HOT``` Item is hot (see EATEN_HOT).
-- ```LITCIG``` Marks a lit smoking item (cigarette, joint etc.).
-- ```MUSHY``` FREEZERBURN item was frozen and is now mushy and tasteless and will go bad after freezing again.
-- ```NO_PARASITES``` Invalidates parasites count set in food->type->comestible->parasites
-- ```QUARTERED``` Corpse was quartered into parts. Affects butcher results, weight, volume.
-- ```REVIVE_SPECIAL``` ... Corpses revives when the player is nearby.
-- ```USE_UPS``` The tool has the UPS mod and is charged from an UPS.
-- ```WARM``` A hidden flag used to track an item's journey to/from hot, buffers between HOT and cold.
-- ```WET``` Item is wet and will slowly dry off (e.g. towel).
-
+Those flags are added by the game code to specific items (that specific welder, not _all_ welders).
+
+- `COLD` Item is cold (see EATEN_COLD).
+- `DIRTY` Item (liquid) was dropped on the ground and is now irreparably dirty.
+- `FIELD_DRESS_FAILED` Corpse was damaged by unskillful field dressing. Affects butcher results.
+- `FIELD_DRESS` Corpse was field dressed. Affects butcher results.
+- `FIT` Reduces encumbrance by one.
+- `FROZEN` Item is frozen solid (used by freezer).
+- `HIDDEN_ITEM` This item cannot be seen in AIM.
+- `HOT` Item is hot (see EATEN_HOT).
+- `LITCIG` Marks a lit smoking item (cigarette, joint etc.).
+- `MUSHY` FREEZERBURN item was frozen and is now mushy and tasteless and will go bad after freezing
+ again.
+- `NO_PARASITES` Invalidates parasites count set in food->type->comestible->parasites
+- `QUARTERED` Corpse was quartered into parts. Affects butcher results, weight, volume.
+- `REVIVE_SPECIAL` ... Corpses revives when the player is nearby.
+- `USE_UPS` The tool has the UPS mod and is charged from an UPS.
+- `WARM` A hidden flag used to track an item's journey to/from hot, buffers between HOT and cold.
+- `WET` Item is wet and will slowly dry off (e.g. towel).
## Vehicle Parts
### Flags
-- ```ADVANCED_PLANTER``` This planter doesn't spill seeds and avoids damaging itself on non-diggable surfaces.
-- ```AISLE_LIGHT```
-- ```AISLE``` Player can move over this part with less speed penalty than normal.
-- ```ALTERNATOR``` Recharges batteries installed on the vehicle. Can only be installed on a part with ```E_ALTERNATOR``` flag.
-- ```ANCHOR_POINT``` Allows secure seatbelt attachment.
-- ```ANIMAL_CTRL``` Can harness an animal, need HARNESS_bodytype flag to specify bodytype of animal.
-- ```ARMOR``` Protects the other vehicle parts it's installed over during collisions.
-- ```ATOMIC_LIGHT```
-- ```AUTOCLAVE``` Acts as an autoclave.
-- ```AUTOPILOT``` This part will enable a vehicle to have a simple autopilot.
-- ```BATTERY_MOUNT```
-- ```BED``` A bed where the player can sleep.
-- ```BEEPER``` Generates noise when the vehicle moves backward.
-- ```BELTABLE``` Seatbelt can be attached to this part.
-- ```BIKE_RACK_VEH``` Can be used to merge an adjacent single tile wide vehicle, or split a single tile wide vehicle off into its own vehicle.
-- ```BOARDABLE``` The player can safely move over or stand on this part while the vehicle is moving.
-- ```CAMERA_CONTROL```
-- ```CAMERA```
-- ```CAPTURE_MOSNTER_VEH``` Can be used to capture monsters when mounted on a vehicle.
-- ```CARGO_LOCKING``` This cargo area is inaccessible to NPCs. Can only be installed on a part with ```LOCKABLE_CARGO``` flag.
-- ```CARGO``` Cargo holding area.
-- ```CHEMLAB``` Acts as a chemistry set for crafting.
-- ```CHIMES``` Generates continuous noise when used.
-- ```CIRCLE_LIGHT``` Projects a circular radius of light when turned on.
-- ```CONE_LIGHT``` Projects a cone of light when turned on.
-- ```CONTROL_ANIMAL``` These controls can only be used to control a vehicle pulled by an animal (such as reins etc).
-- ```CONTROLS``` Can be used to control the vehicle.
-- ```COOLER``` There is separate command to toggle this part.
-- ```COVERED``` Prevents items in cargo parts from emitting any light.
-- ```CRAFTRIG``` Acts as a dehydrator, vacuum sealer and reloading press for crafting purposes. Potentially to include additional tools in the future.
-- ```CTRL_ELECTRONIC``` Controls electrical and electronic systems of the vehicle.
-- ```CURTAIN``` Can be installed over a part flagged with ```WINDOW```, and functions the same as blinds found on windows in buildings.
-- ```DIFFICULTY_REMOVE```
-- ```DISHWASHER``` Can be used to wash filthy non-soft items en masse.
-- ```DOME_LIGHT```
-- ```DOOR_MOTOR``` Can only be installed on a part with ```OPENABLE``` flag.
-- ```E_ALTERNATOR``` Is an engine that can power an alternator.
-- ```E_COLD_START``` Is an engine that starts much slower in cold weather.
-- ```E_COMBUSTION``` Is an engine that burns its fuel and can backfire or explode when damaged.
-- ```E_HEATER``` Is an engine and has a heater to warm internal vehicle items when on.
-- ```E_HIGHER_SKILL``` Is an engine that is more difficult to install as more engines are installed.
-- ```E_NO_POWER_DECAY``` Engines with this flag do not affect total vehicle power suffering diminishing returns.
-- ```E_STARTS_INSTANTLY``` Is an engine that starts instantly, like food pedals.
-- ```EMITTER``` Emits while enabled (emissions are defined by ```emissions``` entry).
-- ```ENABLED_DRAINS_EPOWER``` Produces `epower` watts while enabled (use negative numbers to drain power). This is independent from reactor power production.
-- ```ENGINE``` Is an engine and contributes towards vehicle mechanical power.
-- ```EVENTURN``` Only on during even turns.
-- ```EXTENDS_VISION``` Extends player vision (cameras, mirrors, etc.)
-- ```EXTRA_DRAG``` tells the vehicle that the part exerts engine power reduction.
-- ```FAUCET```
-- ```FLAT_SURF``` Part with a flat hard surface (e.g. table).
-- ```FLOATS``` Provide buoyancy to boats
-- ```FLUIDTANK``` Is a fluid tank.
-- ```FOLDABLE```
-- ```FORGE``` Acts as a forge for crafting.
-- ```FREEZER``` Can freeze items in below zero degrees Celsius temperature.
-- ```FRIDGE``` Can refrigerate items.
-- ```FUNNEL```
-- ```HALF_CIRCLE_LIGHT``` Projects a directed half-circular radius of light when turned on.
-- ```HARNESS_bodytype``` Replace bodytype with `any` to accept any type, or with the targeted type.
-- ```HORN``` Generates noise when used.
-- ```INITIAL_PART``` When starting a new vehicle via the construction menu, this vehicle part will be the initial part of the vehicle (if the used item matches the item required for this part). The items of parts with this flag are automatically added as component to the vehicle start construction.
-- ```INTERNAL``` Can only be installed on a part with ```CARGO``` flag.
-- ```KITCHEN``` Acts as a kitchen unit and heat source for crafting.
-- ```LIGHT```
-- ```LOCKABLE_CARGO``` Cargo containers that are able to have a lock installed.
-- ```MOUNTABLE``` Player can fire mounted weapons from here.
-- ```MUFFLER``` Muffles the noise a vehicle makes while running.
-- ```MULTISQUARE``` Causes this part and any adjacent parts with the same ID to act as a singular part.
-- ```MUSCLE_ARMS``` Power of the engine with such flag depends on player's strength (it's less effective than ```MUSCLE_LEGS```).
-- ```MUSCLE_LEGS``` Power of the engine with such flag depends on player's strength.
-- ```NAILABLE``` Attached with nails
-- ```NEEDS_BATTERY_MOUNT```
-- ```NEEDS_WHEEL_MOUNT_HEAVY``` Can only be installed on a part with ```WHEEL_MOUNT_HEAVY``` flag.
-- ```NEEDS_WHEEL_MOUNT_LIGHT``` Can only be installed on a part with ```WHEEL_MOUNT_LIGHT``` flag.
-- ```NEEDS_WHEEL_MOUNT_MEDIUM``` Can only be installed on a part with ```WHEEL_MOUNT_MEDIUM``` flag.
-- ```NEEDS_WINDOW``` Can only be installed on a part with ```WINDOW``` flag.
-- ```NO_JACK```
-- ```NOINSTALL``` Cannot be installed.
-- ```OBSTACLE``` Cannot walk through part, unless the part is also ```OPENABLE```.
-- ```ODDTURN``` Only on during odd turns.
-- ```ON_CONTROLS``` Can only be installed on a part with ```CONTROLS``` flag.
-- ```ON_ROOF``` - Parts with this flag could only be installed on a roof (parts with ```ROOF``` flag).
-- ```OPAQUE``` Cannot be seen through.
-- ```OPENABLE``` Can be opened or closed.
-- ```OPENCLOSE_INSIDE``` Can be opened or closed, but only from inside the vehicle.
-- ```OVER``` Can be mounted over other parts.
-- ```PERPETUAL``` If paired with REACTOR, part produces electrical power without consuming fuel.
-- ```PLANTER``` Plants seeds into tilled dirt, spilling them when the terrain underneath is unsuitable. It is damaged by running it over non-```DIGGABLE``` surfaces.
-- ```PLOW``` Tills the soil underneath the part while active. Takes damage from unsuitable terrain at a level proportional to the speed of the vehicle.
-- ```POWER_TRANSFER``` Transmits power to and from an attached thingy (probably a vehicle).
-- ```PROTRUSION``` Part sticks out so no other parts can be installed over it.
-- ```RAIL``` This wheel allows vehicle to move on rails.
-- ```REACTOR``` When enabled, part consumes fuel to generate epower.
-- ```REAPER``` Cuts down mature crops, depositing them on the square.
-- ```RECHARGE``` Recharge items with the same flag. ( Currently only the rechargeable battery mod. )
-- ```REMOTE_CONTROLS```
-- ```REVERSIBLE``` Removal has identical requirements to installation but is twice as quick
-- ```ROOF``` Covers a section of the vehicle. Areas of the vehicle that have a roof and roofs on surrounding sections, are considered inside. Otherwise they're outside.
-- ```ROTOR``` Allows vehicle to generate lift. Actual lift depends on engine power sum of all rotor's diameters.
-- ```SCOOP``` Pulls items from underneath the vehicle to the cargo space of the part. Also mops up liquids.
-- ```SEAT``` A seat where the player can sit or sleep.
-- ```SEATBELT``` Helps prevent the player from being ejected from the vehicle during an accident. Can only be installed on a part with ```BELTABLE``` flag.
-- ```SECURITY```
-- ```SHARP``` Striking a monster with this part does cutting damage instead of bashing damage, and prevents stunning the monster.
-- ```SOLAR_PANEL``` Recharges vehicle batteries when exposed to sunlight. Has a 1 in 4 chance of being broken on car generation.
-- ```SPACE_HEATER``` There is separate command to toggle this part.
-- ```STABLE``` Similar to `WHEEL`, but if the vehicle is only a 1x1 section, this single wheel counts as enough wheels.
-- ```STEERABLE``` This wheel is steerable.
-- ```STEREO```
-- ```TOOL_NONE``` Can be removed/installed without any tools
-- ```TOOL_SCREWDRIVER``` Attached with screws, can be removed/installed with a screwdriver
-- ```TOOL_WRENCH``` Attached with bolts, can be removed/installed with a wrench
-- ```TOWEL``` Can be used to dry yourself up.
-- ```TRACK``` Allows the vehicle installed on, to be marked and tracked on map.
-- ```TRACKED``` Contributes to steering effectiveness but doesn't count as a steering axle for install difficulty and still contributes to drag for the center of steering calculation.
-- ```TRANSFORM_TERRAIN``` Transform terrain (using rules defined in ```transform_terrain```).
-- ```TURRET_CONTROLS``` If part with this flag is installed over the turret, it allows to set said turret's targeting mode to full auto. Can only be installed on a part with ```TURRET``` flag.
-- ```TURRET_MOUNT``` Parts with this flag are suitable for installing turrets.
-- ```TURRET``` Is a weapon turret. Can only be installed on a part with ```TURRET_MOUNT``` flag.
-- ```UNMOUNT_ON_DAMAGE``` Part breaks off the vehicle when destroyed by damage. Item is new and typically undamaged.
-- ```UNMOUNT_ON_MOVE``` Dismount this part when the vehicle moves. Doesn't drop the part, unless you give it special handling.
-- ```VARIABLE_SIZE``` Has 'bigness' for power, wheel radius, etc.
-- ```VISION```
-- ```WASHING_MACHINE``` Can be used to wash filthy clothes en masse.
-- ```WATER_WHEEL``` Recharges vehicle batteries when in flowing water.
-- ```WELDRIG``` Acts as a welder for crafting.
-- ```WHEEL``` Counts as a wheel in wheel calculations.
-- ```WIDE_CONE_LIGHT``` Projects a wide cone of light when turned on.
-- ```WIND_POWERED``` This engine is powered by wind ( sails etc ).
-- ```WIND_TURBINE``` Recharges vehicle batteries when exposed to wind.
-- ```WINDOW``` Can see through this part and can install curtains over it.
-- ```WORKBENCH``` Can craft at this part, must be paired with a workbench json entry.
+- `ADVANCED_PLANTER` This planter doesn't spill seeds and avoids damaging itself on non-diggable
+ surfaces.
+- `AISLE_LIGHT`
+- `AISLE` Player can move over this part with less speed penalty than normal.
+- `ALTERNATOR` Recharges batteries installed on the vehicle. Can only be installed on a part with
+ `E_ALTERNATOR` flag.
+- `ANCHOR_POINT` Allows secure seatbelt attachment.
+- `ANIMAL_CTRL` Can harness an animal, need HARNESS_bodytype flag to specify bodytype of animal.
+- `ARMOR` Protects the other vehicle parts it's installed over during collisions.
+- `ATOMIC_LIGHT`
+- `AUTOCLAVE` Acts as an autoclave.
+- `AUTOPILOT` This part will enable a vehicle to have a simple autopilot.
+- `BATTERY_MOUNT`
+- `BED` A bed where the player can sleep.
+- `BEEPER` Generates noise when the vehicle moves backward.
+- `BELTABLE` Seatbelt can be attached to this part.
+- `BIKE_RACK_VEH` Can be used to merge an adjacent single tile wide vehicle, or split a single tile
+ wide vehicle off into its own vehicle.
+- `BOARDABLE` The player can safely move over or stand on this part while the vehicle is moving.
+- `CAMERA_CONTROL`
+- `CAMERA`
+- `CAPTURE_MOSNTER_VEH` Can be used to capture monsters when mounted on a vehicle.
+- `CARGO_LOCKING` This cargo area is inaccessible to NPCs. Can only be installed on a part with
+ `LOCKABLE_CARGO` flag.
+- `CARGO` Cargo holding area.
+- `CHEMLAB` Acts as a chemistry set for crafting.
+- `CHIMES` Generates continuous noise when used.
+- `CIRCLE_LIGHT` Projects a circular radius of light when turned on.
+- `CONE_LIGHT` Projects a cone of light when turned on.
+- `CONTROL_ANIMAL` These controls can only be used to control a vehicle pulled by an animal (such as
+ reins etc).
+- `CONTROLS` Can be used to control the vehicle.
+- `COOLER` There is separate command to toggle this part.
+- `COVERED` Prevents items in cargo parts from emitting any light.
+- `CRAFTRIG` Acts as a dehydrator, vacuum sealer and reloading press for crafting purposes.
+ Potentially to include additional tools in the future.
+- `CTRL_ELECTRONIC` Controls electrical and electronic systems of the vehicle.
+- `CURTAIN` Can be installed over a part flagged with `WINDOW`, and functions the same as blinds
+ found on windows in buildings.
+- `DIFFICULTY_REMOVE`
+- `DISHWASHER` Can be used to wash filthy non-soft items en masse.
+- `DOME_LIGHT`
+- `DOOR_MOTOR` Can only be installed on a part with `OPENABLE` flag.
+- `E_ALTERNATOR` Is an engine that can power an alternator.
+- `E_COLD_START` Is an engine that starts much slower in cold weather.
+- `E_COMBUSTION` Is an engine that burns its fuel and can backfire or explode when damaged.
+- `E_HEATER` Is an engine and has a heater to warm internal vehicle items when on.
+- `E_HIGHER_SKILL` Is an engine that is more difficult to install as more engines are installed.
+- `E_NO_POWER_DECAY` Engines with this flag do not affect total vehicle power suffering diminishing
+ returns.
+- `E_STARTS_INSTANTLY` Is an engine that starts instantly, like food pedals.
+- `EMITTER` Emits while enabled (emissions are defined by `emissions` entry).
+- `ENABLED_DRAINS_EPOWER` Produces `epower` watts while enabled (use negative numbers to drain
+ power). This is independent from reactor power production.
+- `ENGINE` Is an engine and contributes towards vehicle mechanical power.
+- `EVENTURN` Only on during even turns.
+- `EXTENDS_VISION` Extends player vision (cameras, mirrors, etc.)
+- `EXTRA_DRAG` tells the vehicle that the part exerts engine power reduction.
+- `FAUCET`
+- `FLAT_SURF` Part with a flat hard surface (e.g. table).
+- `FLOATS` Provide buoyancy to boats
+- `FLUIDTANK` Is a fluid tank.
+- `FOLDABLE`
+- `FORGE` Acts as a forge for crafting.
+- `FREEZER` Can freeze items in below zero degrees Celsius temperature.
+- `FRIDGE` Can refrigerate items.
+- `FUNNEL`
+- `HALF_CIRCLE_LIGHT` Projects a directed half-circular radius of light when turned on.
+- `HARNESS_bodytype` Replace bodytype with `any` to accept any type, or with the targeted type.
+- `HORN` Generates noise when used.
+- `INITIAL_PART` When starting a new vehicle via the construction menu, this vehicle part will be
+ the initial part of the vehicle (if the used item matches the item required for this part). The
+ items of parts with this flag are automatically added as component to the vehicle start
+ construction.
+- `INTERNAL` Can only be installed on a part with `CARGO` flag.
+- `KITCHEN` Acts as a kitchen unit and heat source for crafting.
+- `LIGHT`
+- `LOCKABLE_CARGO` Cargo containers that are able to have a lock installed.
+- `MOUNTABLE` Player can fire mounted weapons from here.
+- `MUFFLER` Muffles the noise a vehicle makes while running.
+- `MULTISQUARE` Causes this part and any adjacent parts with the same ID to act as a singular part.
+- `MUSCLE_ARMS` Power of the engine with such flag depends on player's strength (it's less effective
+ than `MUSCLE_LEGS`).
+- `MUSCLE_LEGS` Power of the engine with such flag depends on player's strength.
+- `NAILABLE` Attached with nails
+- `NEEDS_BATTERY_MOUNT`
+- `NEEDS_WHEEL_MOUNT_HEAVY` Can only be installed on a part with `WHEEL_MOUNT_HEAVY` flag.
+- `NEEDS_WHEEL_MOUNT_LIGHT` Can only be installed on a part with `WHEEL_MOUNT_LIGHT` flag.
+- `NEEDS_WHEEL_MOUNT_MEDIUM` Can only be installed on a part with `WHEEL_MOUNT_MEDIUM` flag.
+- `NEEDS_WINDOW` Can only be installed on a part with `WINDOW` flag.
+- `NO_JACK`
+- `NOINSTALL` Cannot be installed.
+- `OBSTACLE` Cannot walk through part, unless the part is also `OPENABLE`.
+- `ODDTURN` Only on during odd turns.
+- `ON_CONTROLS` Can only be installed on a part with `CONTROLS` flag.
+- `ON_ROOF` - Parts with this flag could only be installed on a roof (parts with `ROOF` flag).
+- `OPAQUE` Cannot be seen through.
+- `OPENABLE` Can be opened or closed.
+- `OPENCLOSE_INSIDE` Can be opened or closed, but only from inside the vehicle.
+- `OVER` Can be mounted over other parts.
+- `PERPETUAL` If paired with REACTOR, part produces electrical power without consuming fuel.
+- `PLANTER` Plants seeds into tilled dirt, spilling them when the terrain underneath is unsuitable.
+ It is damaged by running it over non-`DIGGABLE` surfaces.
+- `PLOW` Tills the soil underneath the part while active. Takes damage from unsuitable terrain at a
+ level proportional to the speed of the vehicle.
+- `POWER_TRANSFER` Transmits power to and from an attached thingy (probably a vehicle).
+- `PROTRUSION` Part sticks out so no other parts can be installed over it.
+- `RAIL` This wheel allows vehicle to move on rails.
+- `REACTOR` When enabled, part consumes fuel to generate epower.
+- `REAPER` Cuts down mature crops, depositing them on the square.
+- `RECHARGE` Recharge items with the same flag. ( Currently only the rechargeable battery mod. )
+- `REMOTE_CONTROLS`
+- `REVERSIBLE` Removal has identical requirements to installation but is twice as quick
+- `ROOF` Covers a section of the vehicle. Areas of the vehicle that have a roof and roofs on
+ surrounding sections, are considered inside. Otherwise they're outside.
+- `ROTOR` Allows vehicle to generate lift. Actual lift depends on engine power sum of all rotor's
+ diameters.
+- `SCOOP` Pulls items from underneath the vehicle to the cargo space of the part. Also mops up
+ liquids.
+- `SEAT` A seat where the player can sit or sleep.
+- `SEATBELT` Helps prevent the player from being ejected from the vehicle during an accident. Can
+ only be installed on a part with `BELTABLE` flag.
+- `SECURITY`
+- `SHARP` Striking a monster with this part does cutting damage instead of bashing damage, and
+ prevents stunning the monster.
+- `SOLAR_PANEL` Recharges vehicle batteries when exposed to sunlight. Has a 1 in 4 chance of being
+ broken on car generation.
+- `SPACE_HEATER` There is separate command to toggle this part.
+- `STABLE` Similar to `WHEEL`, but if the vehicle is only a 1x1 section, this single wheel counts as
+ enough wheels.
+- `STEERABLE` This wheel is steerable.
+- `STEREO`
+- `TOOL_NONE` Can be removed/installed without any tools
+- `TOOL_SCREWDRIVER` Attached with screws, can be removed/installed with a screwdriver
+- `TOOL_WRENCH` Attached with bolts, can be removed/installed with a wrench
+- `TOWEL` Can be used to dry yourself up.
+- `TRACK` Allows the vehicle installed on, to be marked and tracked on map.
+- `TRACKED` Contributes to steering effectiveness but doesn't count as a steering axle for install
+ difficulty and still contributes to drag for the center of steering calculation.
+- `TRANSFORM_TERRAIN` Transform terrain (using rules defined in `transform_terrain`).
+- `TURRET_CONTROLS` If part with this flag is installed over the turret, it allows to set said
+ turret's targeting mode to full auto. Can only be installed on a part with `TURRET` flag.
+- `TURRET_MOUNT` Parts with this flag are suitable for installing turrets.
+- `TURRET` Is a weapon turret. Can only be installed on a part with `TURRET_MOUNT` flag.
+- `UNMOUNT_ON_DAMAGE` Part breaks off the vehicle when destroyed by damage. Item is new and
+ typically undamaged.
+- `UNMOUNT_ON_MOVE` Dismount this part when the vehicle moves. Doesn't drop the part, unless you
+ give it special handling.
+- `VARIABLE_SIZE` Has 'bigness' for power, wheel radius, etc.
+- `VISION`
+- `WASHING_MACHINE` Can be used to wash filthy clothes en masse.
+- `WATER_WHEEL` Recharges vehicle batteries when in flowing water.
+- `WELDRIG` Acts as a welder for crafting.
+- `WHEEL` Counts as a wheel in wheel calculations.
+- `WIDE_CONE_LIGHT` Projects a wide cone of light when turned on.
+- `WIND_POWERED` This engine is powered by wind ( sails etc ).
+- `WIND_TURBINE` Recharges vehicle batteries when exposed to wind.
+- `WINDOW` Can see through this part and can install curtains over it.
+- `WORKBENCH` Can craft at this part, must be paired with a workbench json entry.
### Vehicle parts requiring other vehicle parts
-The requirement for other vehicle parts is defined for a json flag by setting ```requires_flag``` for the flag. ```requires_flag``` is the other flag that a part with this flag requires.
-
+The requirement for other vehicle parts is defined for a json flag by setting `requires_flag` for
+the flag. `requires_flag` is the other flag that a part with this flag requires.
### Fuel types
-- ```NULL``` None
-- ```battery``` Electrifying.
-- ```diesel``` Refined dino.
-- ```gasoline``` Refined dino.
-- ```plasma``` Superheated.
-- ```plutonium``` 1.21 Gigawatts!
-- ```water``` Clean.
-- ```wind``` Wind powered.
+- `NULL` None
+- `battery` Electrifying.
+- `diesel` Refined dino.
+- `gasoline` Refined dino.
+- `plasma` Superheated.
+- `plutonium` 1.21 Gigawatts!
+- `water` Clean.
+- `wind` Wind powered.
diff --git a/doc/JSON_INFO.md b/doc/JSON_INFO.md
index e0cf198790be..548f4a27f94c 100644
--- a/doc/JSON_INFO.md
+++ b/doc/JSON_INFO.md
@@ -5,33 +5,33 @@ Use the `Home` key to return to the top.
- [Introduction](#introduction)
- [Navigating the JSON](#navigating-the-json)
- [File descriptions](#file-descriptions)
- * [`data/json/`](#datajson)
- * [`data/json/items/`](#datajsonitems)
- + [`data/json/items/comestibles`](#datajsonitemscomestibles)
- * [`data/json/requirements/`](#datajsonrequirements)
- * [`data/json/vehicles/`](#datajsonvehicles)
+ - [`data/json/`](#datajson)
+ - [`data/json/items/`](#datajsonitems)
+ - [`data/json/items/comestibles`](#datajsonitemscomestibles)
+ - [`data/json/requirements/`](#datajsonrequirements)
+ - [`data/json/vehicles/`](#datajsonvehicles)
- [Generic properties and formatting](#generic-properties-and-formatting)
- * [Generic properties](#generic-properties)
- * [Formatting](#formatting)
- + [Time duration](#time-duration)
- + [Other formatting](#other-formatting)
+ - [Generic properties](#generic-properties)
+ - [Formatting](#formatting)
+ - [Time duration](#time-duration)
+ - [Other formatting](#other-formatting)
- [Description and content of each JSON file](#description-and-content-of-each-json-file)
- * [`data/json/` JSONs](#datajson-jsons)
- + [Ascii_arts](#ascii_arts)
- + [Body_parts](#body_parts)
- + [Bionics](#bionics)
- + [Dreams](#dreams)
- + [Disease](#disease_type)
- + [Item Groups](#item-groups)
- + [Item Category](#item-category)
- + [Materials](#materials)
- + [Monster Groups](#monster-groups)
+ - [`data/json/` JSONs](#datajson-jsons)
+ - [Ascii_arts](#ascii_arts)
+ - [Body_parts](#body_parts)
+ - [Bionics](#bionics)
+ - [Dreams](#dreams)
+ - [Disease](#disease_type)
+ - [Item Groups](#item-groups)
+ - [Item Category](#item-category)
+ - [Materials](#materials)
+ - [Monster Groups](#monster-groups)
- [Group definition](#group-definition)
- [Monster definition](#monster-definition)
- + [Monster Factions](#monster-factions)
- + [Monsters](#monsters)
- + [Names](#names)
- + [Profession item substitution](#profession-item-substitution)
+ - [Monster Factions](#monster-factions)
+ - [Monsters](#monsters)
+ - [Names](#names)
+ - [Profession item substitution](#profession-item-substitution)
- [`description`](#-description-)
- [`name`](#-name-)
- [`points`](#-points-)
@@ -42,223 +42,236 @@ Use the `Home` key to return to the top.
- [`flags`](#-flags-)
- [`cbms`](#-cbms-)
- [`traits`](#-traits-)
- + [Recipes](#recipes)
- + [Constructions](#constructions)
- + [Construction groups](#construction-groups)
- + [Construction sequences](#construction-sequences)
- + [Scent Types](#scent_types)
- + [Scores and Achievements](#scores-and-achievements)
+ - [Recipes](#recipes)
+ - [Constructions](#constructions)
+ - [Construction groups](#construction-groups)
+ - [Construction sequences](#construction-sequences)
+ - [Scent Types](#scent_types)
+ - [Scores and Achievements](#scores-and-achievements)
- [`event_transformation`](#event_transformation)
- [`event_statistic`](#event_statistic)
- [`score`](#score)
- [`achievement`](#achievement)
- + [Skills](#skills)
- + [Traits/Mutations](#traits-mutations)
- + [Vehicle Groups](#vehicle-groups)
- + [Vehicle Parts](#vehicle-parts)
- + [Part Resistance](#part-resistance)
- + [Vehicle Placement](#vehicle-placement)
- + [Vehicle Spawn](#vehicle-spawn)
- + [Vehicles](#vehicles)
+ - [Skills](#skills)
+ - [Traits/Mutations](#traits-mutations)
+ - [Vehicle Groups](#vehicle-groups)
+ - [Vehicle Parts](#vehicle-parts)
+ - [Part Resistance](#part-resistance)
+ - [Vehicle Placement](#vehicle-placement)
+ - [Vehicle Spawn](#vehicle-spawn)
+ - [Vehicles](#vehicles)
- [`data/json/items/` JSONs](#datajsonitems-jsons)
- + [Generic Items](#generic-items)
- + [Ammo](#ammo)
- + [Magazine](#magazine)
- + [Armor](#armor)
- + [Pet Armor](#pet-armor)
- + [Books](#books)
- - [Color Key](#color-key)
- + [Comestibles](#comestibles)
- + [Containers](#containers)
- + [Melee](#melee)
- - [`Melee Weapon_category`](#melee-weapon_category)
- + [Gun](#gun)
- - [`Ranged Weapon_category`](#ranged-weapon_category)
- + [Gunmod](#gunmod)
- + [Batteries](#batteries)
- + [Tools](#tools)
- + [Seed Data](#seed-data)
- + [Brewing Data](#brewing-data)
- + [Relic Data](#relic-data)
- + [Artifact Data](#artifact-data)
- - [`Effects_carried`](#effects-carried)
- - [`effects_worn`](#effects-worn)
- - [`effects_wielded`](#effects-wielded)
- - [`effects_activated`](#effects-activated)
- + [Software Data](#software-data)
- + [Fuel data](#fuel-data)
- + [Use Actions](#use-actions)
+ - [Generic Items](#generic-items)
+ - [Ammo](#ammo)
+ - [Magazine](#magazine)
+ - [Armor](#armor)
+ - [Pet Armor](#pet-armor)
+ - [Books](#books)
+ - [Color Key](#color-key)
+ - [Comestibles](#comestibles)
+ - [Containers](#containers)
+ - [Melee](#melee)
+ - [`Melee Weapon_category`](#melee-weapon_category)
+ - [Gun](#gun)
+ - [`Ranged Weapon_category`](#ranged-weapon_category)
+ - [Gunmod](#gunmod)
+ - [Batteries](#batteries)
+ - [Tools](#tools)
+ - [Seed Data](#seed-data)
+ - [Brewing Data](#brewing-data)
+ - [Relic Data](#relic-data)
+ - [Artifact Data](#artifact-data)
+ - [`Effects_carried`](#effects-carried)
+ - [`effects_worn`](#effects-worn)
+ - [`effects_wielded`](#effects-wielded)
+ - [`effects_activated`](#effects-activated)
+ - [Software Data](#software-data)
+ - [Fuel data](#fuel-data)
+ - [Use Actions](#use-actions)
- [`json/` JSONs](#json-jsons)
- + [Harvest](#harvest)
- - [`id`](#-id-)
- - [`type`](#-type-)
- - [`message`](#-message-)
- - [`entries`](#-entries-)
- + [Furniture](#furniture)
- - [`type`](#-type--1)
- - [`move_cost_mod`](#-move-cost-mod-)
- - [`light_emitted`](#-light-emitted-)
- - [`required_str`](#-required-str-)
- - [`crafting_pseudo_item`](#-crafting-pseudo-item-)
- - [`workbench`](#-workbench-)
- - [`plant_data`](#-plant-data-)
- + [Terrain](#terrain)
- - [`type`](#-type--2)
- - [`move_cost`](#-move-cost-)
- - [`light_emitted`](#-light-emitted--1)
- - [`trap`](#-trap-)
- - [`harvestable`](#-harvestable-)
- - [`transforms_into`](#-transforms-into-)
- - [`harvest_season`](#-harvest-season-)
- - [`roof`](#-roof-)
- + [Common To Furniture And Terrain](#common-to-furniture-and-terrain)
- - [`id`](#-id--1)
- - [`name`](#-name--1)
- - [`flags`](#-flags--1)
- - [`connects_to`](#-connects-to-)
- - [`symbol`](#-symbol-)
- - [`looks_like`](#-looks-like-)
- - [`color` or `bgcolor`](#-color--or--bgcolor-)
- - [`max_volume`](#-max-volume-)
- - [`examine_action`](#-examine-action-)
- - [`close" And "open`](#-close--and--open-)
- - [`bash`](#-bash-)
- - [`deconstruct`](#-deconstruct-)
- - [`pry`](#-pry-)
- - [`map_bash_info`](#-map-bash-info-)
- - [`str_min`, `str_max`, `str_min_blocked`, `str_max_blocked`, `str_min_supported`, `str_max_supported`](#-str-min----str-max----str-min-blocked----str-max-blocked----str-min-supported----str-max-supported-)
- - [`sound`, `sound_fail`, `sound_vol`, `sound_fail_vol`](#-sound----sound-fail----sound-vol----sound-fail-vol-)
- - [`furn_set`, `ter_set`](#-furn-set----ter-set-)
- - [`explosive`](#-explosive-)
- - [`destroy_only`](#-destroy-only-)
- - [`bash_below`](#-bash-below-)
- - [`tent_centers`, `collapse_radius`](#-tent-centers----collapse-radius-)
- - [`items`](#-items--1)
- - [`map_deconstruct_info`](#-map-deconstruct-info-)
- - [`furn_set`, `ter_set`](#-furn-set----ter-set--1)
- - [`items`](#-items-2)
- - [`prying_result`](#-prying-result-)
- - [`new_ter_type`, `new_furn_type`](#-new-furn-type----new-ter-type-)
- - [`success_message`, `fail_message`, `break_message`](#-success-message----fail-message----break-message-)
- - [`pry_quality`, `pry_bonus_mult`, `difficulty`](#-pry-quality----pry-bonus-mult----difficulty-)
- - [`noise`, `break_noise`, `sound`, `break_sound`](#-noise----break-noise----sound----break-sound-)
- - [`breakable`, `break_ter_type`, `break_furn_type`](#-breakable----break-ter-type----break-furn-type-)
- - [`break_items`](#-break-items-)
- + [`plant_data`](#plant_data-1)
- - [`transform`](#-transform-)
- - [`base`](#-base-)
- - [`growth_multiplier`](#-growth-multiplier-)
- - [`harvest_multiplier`](#-harvest-multiplier-)
- + [clothing_mod](#clothing_mod)
+ - [Harvest](#harvest)
+ - [`id`](#-id-)
+ - [`type`](#-type-)
+ - [`message`](#-message-)
+ - [`entries`](#-entries-)
+ - [Furniture](#furniture)
+ - [`type`](#-type--1)
+ - [`move_cost_mod`](#-move-cost-mod-)
+ - [`light_emitted`](#-light-emitted-)
+ - [`required_str`](#-required-str-)
+ - [`crafting_pseudo_item`](#-crafting-pseudo-item-)
+ - [`workbench`](#-workbench-)
+ - [`plant_data`](#-plant-data-)
+ - [Terrain](#terrain)
+ - [`type`](#-type--2)
+ - [`move_cost`](#-move-cost-)
+ - [`light_emitted`](#-light-emitted--1)
+ - [`trap`](#-trap-)
+ - [`harvestable`](#-harvestable-)
+ - [`transforms_into`](#-transforms-into-)
+ - [`harvest_season`](#-harvest-season-)
+ - [`roof`](#-roof-)
+ - [Common To Furniture And Terrain](#common-to-furniture-and-terrain)
+ - [`id`](#-id--1)
+ - [`name`](#-name--1)
+ - [`flags`](#-flags--1)
+ - [`connects_to`](#-connects-to-)
+ - [`symbol`](#-symbol-)
+ - [`looks_like`](#-looks-like-)
+ - [`color` or `bgcolor`](#-color--or--bgcolor-)
+ - [`max_volume`](#-max-volume-)
+ - [`examine_action`](#-examine-action-)
+ - [`close" And "open`](#-close--and--open-)
+ - [`bash`](#-bash-)
+ - [`deconstruct`](#-deconstruct-)
+ - [`pry`](#-pry-)
+ - [`map_bash_info`](#-map-bash-info-)
+ - [`str_min`, `str_max`, `str_min_blocked`, `str_max_blocked`, `str_min_supported`, `str_max_supported`](#-str-min----str-max----str-min-blocked----str-max-blocked----str-min-supported----str-max-supported-)
+ - [`sound`, `sound_fail`, `sound_vol`, `sound_fail_vol`](#-sound----sound-fail----sound-vol----sound-fail-vol-)
+ - [`furn_set`, `ter_set`](#-furn-set----ter-set-)
+ - [`explosive`](#-explosive-)
+ - [`destroy_only`](#-destroy-only-)
+ - [`bash_below`](#-bash-below-)
+ - [`tent_centers`, `collapse_radius`](#-tent-centers----collapse-radius-)
+ - [`items`](#-items--1)
+ - [`map_deconstruct_info`](#-map-deconstruct-info-)
+ - [`furn_set`, `ter_set`](#-furn-set----ter-set--1)
+ - [`items`](#-items-2)
+ - [`prying_result`](#-prying-result-)
+ - [`new_ter_type`, `new_furn_type`](#-new-furn-type----new-ter-type-)
+ - [`success_message`, `fail_message`, `break_message`](#-success-message----fail-message----break-message-)
+ - [`pry_quality`, `pry_bonus_mult`, `difficulty`](#-pry-quality----pry-bonus-mult----difficulty-)
+ - [`noise`, `break_noise`, `sound`, `break_sound`](#-noise----break-noise----sound----break-sound-)
+ - [`breakable`, `break_ter_type`, `break_furn_type`](#-breakable----break-ter-type----break-furn-type-)
+ - [`break_items`](#-break-items-)
+ - [`plant_data`](#plant_data-1)
+ - [`transform`](#-transform-)
+ - [`base`](#-base-)
+ - [`growth_multiplier`](#-growth-multiplier-)
+ - [`harvest_multiplier`](#-harvest-multiplier-)
+ - [clothing_mod](#clothing_mod)
- [Scenarios](#scenarios)
- * [`description`](#-description--1)
- * [`name`](#-name--2)
- * [`points`](#-points--1)
- * [`items`](#-items--3)
- * [`flags`](#-flags--2)
- * [`cbms`](#-cbms--1)
- * [`traits", "forced_traits", "forbidden_traits`](#-traits----forced-traits----forbidden-traits-)
- * [`allowed_locs`](#-allowed-locs-)
- * [`start_name`](#-start-name-)
- * [`professions`](#-professions-)
- * [`map_special`](#-map-special-)
- * [`missions`](#-missions-)
+ - [`description`](#-description--1)
+ - [`name`](#-name--2)
+ - [`points`](#-points--1)
+ - [`items`](#-items--3)
+ - [`flags`](#-flags--2)
+ - [`cbms`](#-cbms--1)
+ - [`traits", "forced_traits", "forbidden_traits`](#-traits----forced-traits----forbidden-traits-)
+ - [`allowed_locs`](#-allowed-locs-)
+ - [`start_name`](#-start-name-)
+ - [`professions`](#-professions-)
+ - [`map_special`](#-map-special-)
+ - [`missions`](#-missions-)
- [Starting locations](#starting-locations)
- * [`name`](#-name--3)
- * [`target`](#-target-)
- * [`flags`](#-flags--3)
- + [`tile_config`](#-tile-config-)
+ - [`name`](#-name--3)
+ - [`target`](#-target-)
+ - [`flags`](#-flags--3)
+ - [`tile_config`](#-tile-config-)
- [Mutation overlay ordering](#mutation-overlay-ordering)
- * [`id`](#-id--2)
- * [`order`](#-order-)
+ - [`id`](#-id--2)
+ - [`order`](#-order-)
- [MOD_INFO](#mod_info)
- [MOD tileset](#mod-tileset)
- * [`compatibility`](#-compatibility-)
- * [`tiles-new`](#-tiles-new-)
+ - [`compatibility`](#-compatibility-)
+ - [`tiles-new`](#-tiles-new-)
- [Field types](#-field-types-)
# Introduction
-This document describes the contents of the json files used in Cataclysm: Dark days ahead. You are probably reading this if you want to add or change content of Catacysm: Dark days ahead and need to learn more about what to find where and what each file and property does.
+
+This document describes the contents of the json files used in Cataclysm: Dark days ahead. You are
+probably reading this if you want to add or change content of Catacysm: Dark days ahead and need to
+learn more about what to find where and what each file and property does.
# Navigating the JSON
-A lot of the JSON involves cross-references to other JSON entities. To make it easier to navigate, we provide a script `tools/json_tools/cddatags.py` that can build a `tags` file for you.
-To run the script you'll need Python 3. On Windows you'll probably need to install that, and associate `.py` files with Python. Then open a command prompt, navigate to your CDDA folder, and run `tools\json_tools\cddatags.py`.
+A lot of the JSON involves cross-references to other JSON entities. To make it easier to navigate,
+we provide a script `tools/json_tools/cddatags.py` that can build a `tags` file for you.
+
+To run the script you'll need Python 3. On Windows you'll probably need to install that, and
+associate `.py` files with Python. Then open a command prompt, navigate to your CDDA folder, and run
+`tools\json_tools\cddatags.py`.
-To use this feature your editor will need [ctags support](http://ctags.sourceforge.net/). When that's working you should be able to easily jump to the definition of any entity. For example, by positioning your cursor over an id and hitting the appropriate key combination.
+To use this feature your editor will need [ctags support](http://ctags.sourceforge.net/). When
+that's working you should be able to easily jump to the definition of any entity. For example, by
+positioning your cursor over an id and hitting the appropriate key combination.
-* In Vim, this feature exists by default, and you can jump to a definition using [`^]`](http://vimdoc.sourceforge.net/htmldoc/tagsrch.html#tagsrch.txt).
-* In Notepad++ go to "Plugins" -> "Plugins Admin" and enable the "TagLEET" plugin. Then select any id and press Alt+Space to open the references window.
+- In Vim, this feature exists by default, and you can jump to a definition using
+ [`^]`](http://vimdoc.sourceforge.net/htmldoc/tagsrch.html#tagsrch.txt).
+- In Notepad++ go to "Plugins" -> "Plugins Admin" and enable the "TagLEET" plugin. Then select any
+ id and press Alt+Space to open the references window.
# File descriptions
-Here's a quick summary of what each of the JSON files contain, broken down by folder. This list is not comprehensive, but covers the broad strokes.
+
+Here's a quick summary of what each of the JSON files contain, broken down by folder. This list is
+not comprehensive, but covers the broad strokes.
## `data/json/`
-| Filename | Description
-|--- |---
-| achievements.json | achievements
-| anatomy.json | a listing of player body parts - do not edit
-| ascii_arts.json | ascii arts for item descriptions
-| bionics.json | bionics, does NOT include bionic effects
-| body_parts.json | an expansion of anatomy.json - do not edit
-| clothing_mods.json | definition of clothing mods
-| construction.json | definition of construction menu tasks
-| default_blacklist.json | a standard blacklist of joke monsters
-| doll_speech.json | talk doll speech messages
-| dreams.json | dream text and linked mutation categories
-| disease.json | disease definitions
-| effects.json | common effects and their effects
-| emit.json | smoke and gas emissions
-| flags.json | common flags and their descriptions
-| furniture.json | furniture, and features treated like furniture
-| game_balance.json | various options to tweak game balance
-| gates.json | gate terrain definitions
-| harvest.json | item drops for butchering corpses
-| health_msgs.json | messages displayed when the player wakes
-| item_actions.json | descriptions of standard item actions
-| item_category.json | item categories and their default sort
-| item_groups.json | item spawn groups
-| lab_notes.json | lab computer messages
-| martialarts.json | martial arts styles and buffs
-| materials.json | material types
-| monster_attacks.json | monster attacks
-| monster_drops.json | monster item drops on death
-| monster_factions.json | monster factions
-| monstergroups.json | monster spawn groups
-| monstergroups_egg.json | monster spawn groups from eggs
-| monsters.json | monster descriptions, mostly zombies
-| morale_types.json | morale modifier messages
-| mutation_category.json | messages for mutation categories
-| mutation_ordering.json | draw order for mutation and CBM overlays in tiles mode
-| mutations.json | traits/mutations
-| names.json | names used for NPC/player name generation
-| overmap_connections.json | connections for roads and tunnels in the overmap
-| overmap_terrain.json | overmap terrain
-| player_activities.json | player activities
-| professions.json | profession definitions
-| recipes.json | crafting/disassembly recipes
-| regional_map_settings.json | settings for the entire map generation
-| road_vehicles.json | vehicle spawn information for roads
-| rotatable_symbols.json | rotatable symbols - do not edit
-| scent_types.json | type of scent available
-| scores.json | scores
-| skills.json | skill descriptions and ID's
-| snippets.json | flier/poster descriptions
-| species.json | monster species
-| speech.json | monster vocalizations
-| statistics.json | statistics and transformations used to define scores and achievements
-| start_locations.json | starting locations for scenarios
-| techniques.json | generic for items and martial arts
-| terrain.json | terrain types and definitions
-| test_regions.json | test regions
-| tips.json | tips of the day
-| tool_qualities.json | standard tool qualities and their actions
-| traps.json | standard traps
-| tutorial.json | messages for the tutorial (that is out of date)
-| vehicle_groups.json | vehicle spawn groups
-| vehicle_parts.json | vehicle parts, does NOT affect flag effects
-| vitamin.json | vitamins and their deficiencies
+| Filename | Description |
+| -------------------------- | --------------------------------------------------------------------- |
+| achievements.json | achievements |
+| anatomy.json | a listing of player body parts - do not edit |
+| ascii_arts.json | ascii arts for item descriptions |
+| bionics.json | bionics, does NOT include bionic effects |
+| body_parts.json | an expansion of anatomy.json - do not edit |
+| clothing_mods.json | definition of clothing mods |
+| construction.json | definition of construction menu tasks |
+| default_blacklist.json | a standard blacklist of joke monsters |
+| doll_speech.json | talk doll speech messages |
+| dreams.json | dream text and linked mutation categories |
+| disease.json | disease definitions |
+| effects.json | common effects and their effects |
+| emit.json | smoke and gas emissions |
+| flags.json | common flags and their descriptions |
+| furniture.json | furniture, and features treated like furniture |
+| game_balance.json | various options to tweak game balance |
+| gates.json | gate terrain definitions |
+| harvest.json | item drops for butchering corpses |
+| health_msgs.json | messages displayed when the player wakes |
+| item_actions.json | descriptions of standard item actions |
+| item_category.json | item categories and their default sort |
+| item_groups.json | item spawn groups |
+| lab_notes.json | lab computer messages |
+| martialarts.json | martial arts styles and buffs |
+| materials.json | material types |
+| monster_attacks.json | monster attacks |
+| monster_drops.json | monster item drops on death |
+| monster_factions.json | monster factions |
+| monstergroups.json | monster spawn groups |
+| monstergroups_egg.json | monster spawn groups from eggs |
+| monsters.json | monster descriptions, mostly zombies |
+| morale_types.json | morale modifier messages |
+| mutation_category.json | messages for mutation categories |
+| mutation_ordering.json | draw order for mutation and CBM overlays in tiles mode |
+| mutations.json | traits/mutations |
+| names.json | names used for NPC/player name generation |
+| overmap_connections.json | connections for roads and tunnels in the overmap |
+| overmap_terrain.json | overmap terrain |
+| player_activities.json | player activities |
+| professions.json | profession definitions |
+| recipes.json | crafting/disassembly recipes |
+| regional_map_settings.json | settings for the entire map generation |
+| road_vehicles.json | vehicle spawn information for roads |
+| rotatable_symbols.json | rotatable symbols - do not edit |
+| scent_types.json | type of scent available |
+| scores.json | scores |
+| skills.json | skill descriptions and ID's |
+| snippets.json | flier/poster descriptions |
+| species.json | monster species |
+| speech.json | monster vocalizations |
+| statistics.json | statistics and transformations used to define scores and achievements |
+| start_locations.json | starting locations for scenarios |
+| techniques.json | generic for items and martial arts |
+| terrain.json | terrain types and definitions |
+| test_regions.json | test regions |
+| tips.json | tips of the day |
+| tool_qualities.json | standard tool qualities and their actions |
+| traps.json | standard traps |
+| tutorial.json | messages for the tutorial (that is out of date) |
+| vehicle_groups.json | vehicle spawn groups |
+| vehicle_parts.json | vehicle parts, does NOT affect flag effects |
+| vitamin.json | vitamins and their deficiencies |
selected subfolders
@@ -266,33 +279,33 @@ selected subfolders
See below for specifics on the various items
-| Filename | Description
-|--- |---
-| ammo.json | common base components like batteries and marbles
-| ammo_types.json | standard ammo types by gun
-| archery.json | bows and arrows
-| armor.json | armor and clothing
-| bionics.json | Compact Bionic Modules (CBMs)
-| biosignatures.json | animal waste
-| books.json | books
-| chemicals_and_resources.json | chemical precursors
-| comestibles.json | food/drinks
-| containers.json | containers
-| crossbows.json | crossbows and bolts
-| fake.json | fake items for bionics or mutations
-| fuel.json | liquid fuels
-| grenades.json | grenades and throwable explosives
-| handloaded_bullets.json | random ammo
-| melee.json | anything that doesn't go in the other item jsons, melee weapons
-| migration.json | conversions of non-existent items from save games to current items
-| newspaper.json | flyers, newspapers, and survivor notes. snippets.json for messages
-| obsolete.json | items being removed from the game
-| ranged.json | guns
-| software.json | software for SD-cards and USB sticks
-| tool_armor.json | clothes and armor that can be (a)ctivated
-| toolmod.json | modifications of tools
-| tools.json | tools and items that can be (a)ctivated
-| vehicle_parts.json | components of vehicles when they aren't on the vehicle
+| Filename | Description |
+| ---------------------------- | ------------------------------------------------------------------ |
+| ammo.json | common base components like batteries and marbles |
+| ammo_types.json | standard ammo types by gun |
+| archery.json | bows and arrows |
+| armor.json | armor and clothing |
+| bionics.json | Compact Bionic Modules (CBMs) |
+| biosignatures.json | animal waste |
+| books.json | books |
+| chemicals_and_resources.json | chemical precursors |
+| comestibles.json | food/drinks |
+| containers.json | containers |
+| crossbows.json | crossbows and bolts |
+| fake.json | fake items for bionics or mutations |
+| fuel.json | liquid fuels |
+| grenades.json | grenades and throwable explosives |
+| handloaded_bullets.json | random ammo |
+| melee.json | anything that doesn't go in the other item jsons, melee weapons |
+| migration.json | conversions of non-existent items from save games to current items |
+| newspaper.json | flyers, newspapers, and survivor notes. snippets.json for messages |
+| obsolete.json | items being removed from the game |
+| ranged.json | guns |
+| software.json | software for SD-cards and USB sticks |
+| tool_armor.json | clothes and armor that can be (a)ctivated |
+| toolmod.json | modifications of tools |
+| tools.json | tools and items that can be (a)ctivated |
+| vehicle_parts.json | components of vehicles when they aren't on the vehicle |
### `data/json/items/comestibles`
@@ -300,67 +313,70 @@ See below for specifics on the various items
Standard components and tools for crafting
-| Filename | Description
-|--- |---
-| ammo.json | ammo components
-| cooking_components.json | common ingredient sets
-| cooking_requirements.json | cooking tools and heat sources
-| materials.json | thread, fabric, and other basic materials
-| toolsets.json | sets of tools commonly used together
-| uncraft.json | common results of taking stuff apart
-| vehicle.json | tools to work on vehicles
-
+| Filename | Description |
+| ------------------------- | ----------------------------------------- |
+| ammo.json | ammo components |
+| cooking_components.json | common ingredient sets |
+| cooking_requirements.json | cooking tools and heat sources |
+| materials.json | thread, fabric, and other basic materials |
+| toolsets.json | sets of tools commonly used together |
+| uncraft.json | common results of taking stuff apart |
+| vehicle.json | tools to work on vehicles |
## `data/json/vehicles/`
Groups of vehicle definitions with self-explanatory names of files:
-| Filename
-|---
-| bikes.json
-| boats.json
-| cars.json
-| carts.json
-| custom_vehicles.json
-| emergency.json
-| farm.json
-| helicopters.json
-| military.json
-| trains.json
-| trucks.json
-| utility.json
-| vans_busses.json
-| vehicles.json
+| Filename |
+| -------------------- |
+| bikes.json |
+| boats.json |
+| cars.json |
+| carts.json |
+| custom_vehicles.json |
+| emergency.json |
+| farm.json |
+| helicopters.json |
+| military.json |
+| trains.json |
+| trucks.json |
+| utility.json |
+| vans_busses.json |
+| vehicles.json |
# Generic properties and formatting
+
This section describes properties and formatting applied to all of the JSON files.
## Generic properties
-A few properties are applicable to most if not all json files and do not need to be described for each json file. These properties are:
-
-| Identifier | Description
-|--- |---
-| type | The type of object this json entry is describing. Setting this entry to 'armor' for example means the game will expect properties specific to armor in that entry. Also ties in with 'copy-from' (see below), if you want to inherit properties of another object, it must be of the same tipe.
-| [copy-from](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | The identifier of the item you wish to copy properties from. This allows you to make an exact copy of an item __of the same type__ and only provide entries that should change from the item you copied from.
-| [extends](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | Modders can add an "extends" field to their definition to append entries to a list instead of overriding the entire list.
-| [delete](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | Modders can also add a "delete" field that removes elements from lists instead of overriding the entire list.
-| [abstract](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | Creates an abstract item (an item that does not end up in the game and solely exists in the json to be copied-from. Use this _instead of_ 'id'.
+A few properties are applicable to most if not all json files and do not need to be described for
+each json file. These properties are:
+| Identifier | Description |
+| --------------------------------------------------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| type | The type of object this json entry is describing. Setting this entry to 'armor' for example means the game will expect properties specific to armor in that entry. Also ties in with 'copy-from' (see below), if you want to inherit properties of another object, it must be of the same tipe. |
+| [copy-from](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | The identifier of the item you wish to copy properties from. This allows you to make an exact copy of an item **of the same type** and only provide entries that should change from the item you copied from. |
+| [extends](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | Modders can add an "extends" field to their definition to append entries to a list instead of overriding the entire list. |
+| [delete](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | Modders can also add a "delete" field that removes elements from lists instead of overriding the entire list. |
+| [abstract](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/doc/JSON_INHERITANCE.md) | Creates an abstract item (an item that does not end up in the game and solely exists in the json to be copied-from. Use this _instead of_ 'id'. |
## Formatting
+
When editing JSON files make sure you apply the correct formatting as shown below.
### Time duration
-A string containing one or more pairs of number and time duration unit. Number and unit, as well as each pair, can be separated by an arbitrary amount of spaces.
-Available units:
+A string containing one or more pairs of number and time duration unit. Number and unit, as well as
+each pair, can be separated by an arbitrary amount of spaces. Available units:
+
- "hours", "hour", "h" - one hour
- "days", "day", "d" - one day
- "minutes", "minute", "m" - one minute
- "turns", "turn", "t" - one turn,
Examples:
+
- " +1 day -23 hours 50m " `(1*24*60 - 23*60 + 50 == 110 minutes)`
- "1 turn 1 minutes 9 turns" (1 minute and 10 seconds because 1 turn is 1 second)
@@ -370,7 +386,10 @@ Examples:
"//" : "comment", // Preferred method of leaving comments inside json files.
```
-Some json strings are extracted for translation, for example item names, descriptions, etc. The exact extraction is handled in `lang/extract_json_strings.py`. Apart from the obvious way of writing a string without translation context, the string can also have an optional translation context (and sometimes a plural form), by writing it like:
+Some json strings are extracted for translation, for example item names, descriptions, etc. The
+exact extraction is handled in `lang/extract_json_strings.py`. Apart from the obvious way of writing
+a string without translation context, the string can also have an optional translation context (and
+sometimes a plural form), by writing it like:
```JSON
"name": { "ctxt": "foo", "str": "bar", "str_pl": "baz" }
@@ -382,8 +401,8 @@ or, if the plural form is the same as the singular form:
"name": { "ctxt": "foo", "str_sp": "foo" }
```
-You can also add comments for translators by adding a "//~" entry like below. The
-order of the entries does not matter.
+You can also add comments for translators by adding a "//~" entry like below. The order of the
+entries does not matter.
```JSON
"name": {
@@ -392,134 +411,139 @@ order of the entries does not matter.
}
```
-Currently, only some JSON values support this syntax (see [here](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/TRANSLATING.md#translation) for a list of supported values and more detailed explanation).
+Currently, only some JSON values support this syntax (see
+[here](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/TRANSLATING.md#translation) for
+a list of supported values and more detailed explanation).
# Description and content of each JSON file
-This section describes each json file and their contents. Each json has their own unique properties that are not shared with other Json files (for example 'chapters' property used in books does not apply to armor). This will make sure properties are only described and used within the context of the appropriate JSON file.
+This section describes each json file and their contents. Each json has their own unique properties
+that are not shared with other Json files (for example 'chapters' property used in books does not
+apply to armor). This will make sure properties are only described and used within the context of
+the appropriate JSON file.
## `data/json/` JSONs
### Ascii_arts
-| Identifier | Description
-|--- |---
-| id | Unique ID. Must be one continuous word, use underscores if necessary.
-| picture | Array of string, each entry is a line of an ascii picture and must be at most 42 columns long.
+| Identifier | Description |
+| ---------- | ---------------------------------------------------------------------------------------------- |
+| id | Unique ID. Must be one continuous word, use underscores if necessary. |
+| picture | Array of string, each entry is a line of an ascii picture and must be at most 42 columns long. |
```C++
- {
- "type": "ascii_art",
- "id": "cashcard",
- "picture": [
- "",
- "",
- "",
- " ╔═══════════════════╗",
- " ║ ║",
- " ║ ╔═ ╔═╔═╗╔═║ ║ ║",
- " ║ ║═ ┼ ║ ║═║╚╗║═║ ║",
- " ║ ╚═ ╚═║ ║═╝║ ║ ║",
- " ║ ║",
- " ║ RIVTECH TRUST ║",
- " ║ ║",
- " ║ ║",
- " ║ 555 993 55221 066 ║",
- " ╚═══════════════════╝"
- ]
- }
+{
+ "type": "ascii_art",
+ "id": "cashcard",
+ "picture": [
+ "",
+ "",
+ "",
+ " ╔═══════════════════╗",
+ " ║ ║",
+ " ║ ╔═ ╔═╔═╗╔═║ ║ ║",
+ " ║ ║═ ┼ ║ ║═║╚╗║═║ ║",
+ " ║ ╚═ ╚═║ ║═╝║ ║ ║",
+ " ║ ║",
+ " ║ RIVTECH TRUST ║",
+ " ║ ║",
+ " ║ ║",
+ " ║ 555 993 55221 066 ║",
+ " ╚═══════════════════╝"
+ ]
+}
```
### Body_parts
-| Identifier | Description
-|--- |---
-| id | (_mandatory_) Unique ID. Must be one continuous word, use underscores if necessary.
-| name | (_mandatory_) In-game name displayed.
-| accusative | (_mandatory_) Accusative form for this bodypart.
-| heading | (_mandatory_) How it's displayed in headings.
-| heading_multiple | (_optional_) Plural form of heading. (default: heading value)
-| hp_bar_ui_text | (_optional_) How it's displayed next to the hp bar in the panel. (default: empty string)
-| encumbrance_text | (_optional_) Description of effect when encumbered. (default: empty string)
-| main_part | (_optional_) What is the main part this one is attached to. (default: self)
-| base_hp | (_optional_) The amount of hp this part has before any modification. (default: `60`)
-| opposite_part | (_optional_) What is the opposite part ot this one in case of a pair. (default: self)
-| essential | (_optional_) Whether the character dies if this part drops to `0` HP.
-| hit_size | (_optional_) Float. Size of the body part when doing an unweighted selection. (default: `0.`)
-| hit_size_relative | (_optional_) Float. Hit sizes for attackers who are smaller, equal in size, and bigger. (default: `[ 0, 0, 0 ]`
-| hit_difficulty | (_optional_) Float. How hard is it to hit a given body part, assuming "owner" is hit. Higher number means good hits will veer towards this part, lower means this part is unlikely to be hit by inaccurate attacks. Formula is `chance *= pow(hit_roll, hit_difficulty)` (default: `0`)
-| side | (_optional_) Which side this body part is on. Default both.
-| stylish_bonus | (_optional_) Mood bonus associated with wearing fancy clothing on this part. (default: `0`)
-| hot_morale_mod | (_optional_) Mood effect of being too hot on this part. (default: `0`)
-| cold_morale_mod | (_optional_) Mood effect of being too cold on this part. (default: `0`)
-| squeamish_penalty | (_optional_) Mood effect of wearing filthy clothing on this part. (default: `0`)
-| bionic_slots | (_optional_) How many bionic slots does this part have.
+| Identifier | Description |
+| ----------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| id | (_mandatory_) Unique ID. Must be one continuous word, use underscores if necessary. |
+| name | (_mandatory_) In-game name displayed. |
+| accusative | (_mandatory_) Accusative form for this bodypart. |
+| heading | (_mandatory_) How it's displayed in headings. |
+| heading_multiple | (_optional_) Plural form of heading. (default: heading value) |
+| hp_bar_ui_text | (_optional_) How it's displayed next to the hp bar in the panel. (default: empty string) |
+| encumbrance_text | (_optional_) Description of effect when encumbered. (default: empty string) |
+| main_part | (_optional_) What is the main part this one is attached to. (default: self) |
+| base_hp | (_optional_) The amount of hp this part has before any modification. (default: `60`) |
+| opposite_part | (_optional_) What is the opposite part ot this one in case of a pair. (default: self) |
+| essential | (_optional_) Whether the character dies if this part drops to `0` HP. |
+| hit_size | (_optional_) Float. Size of the body part when doing an unweighted selection. (default: `0.`) |
+| hit_size_relative | (_optional_) Float. Hit sizes for attackers who are smaller, equal in size, and bigger. (default: `[ 0, 0, 0 ]` |
+| hit_difficulty | (_optional_) Float. How hard is it to hit a given body part, assuming "owner" is hit. Higher number means good hits will veer towards this part, lower means this part is unlikely to be hit by inaccurate attacks. Formula is `chance *= pow(hit_roll, hit_difficulty)` (default: `0`) |
+| side | (_optional_) Which side this body part is on. Default both. |
+| stylish_bonus | (_optional_) Mood bonus associated with wearing fancy clothing on this part. (default: `0`) |
+| hot_morale_mod | (_optional_) Mood effect of being too hot on this part. (default: `0`) |
+| cold_morale_mod | (_optional_) Mood effect of being too cold on this part. (default: `0`) |
+| squeamish_penalty | (_optional_) Mood effect of wearing filthy clothing on this part. (default: `0`) |
+| bionic_slots | (_optional_) How many bionic slots does this part have. |
```C++
- {
- "id": "torso",
- "type": "body_part",
- "name": "torso",
- "accusative": { "ctxt": "bodypart_accusative", "str": "torso" },
- "heading": "Torso",
- "heading_multiple": "Torso",
- "hp_bar_ui_text": "TORSO",
- "encumbrance_text": "Dodging and melee is hampered.",
- "main_part": "torso",
- "opposite_part": "torso",
- "hit_size": 45,
- "hit_size_relative": [ 20, 33.33, 36.57 ],
- "hit_difficulty": 1,
- "side": "both",
- "legacy_id": "TORSO",
- "stylish_bonus": 6,
- "hot_morale_mod": 2,
- "cold_morale_mod": 2,
- "squeamish_penalty": 6,
- "base_hp": 60,
- "bionic_slots": 80
- }
+{
+ "id": "torso",
+ "type": "body_part",
+ "name": "torso",
+ "accusative": { "ctxt": "bodypart_accusative", "str": "torso" },
+ "heading": "Torso",
+ "heading_multiple": "Torso",
+ "hp_bar_ui_text": "TORSO",
+ "encumbrance_text": "Dodging and melee is hampered.",
+ "main_part": "torso",
+ "opposite_part": "torso",
+ "hit_size": 45,
+ "hit_size_relative": [ 20, 33.33, 36.57 ],
+ "hit_difficulty": 1,
+ "side": "both",
+ "legacy_id": "TORSO",
+ "stylish_bonus": 6,
+ "hot_morale_mod": 2,
+ "cold_morale_mod": 2,
+ "squeamish_penalty": 6,
+ "base_hp": 60,
+ "bionic_slots": 80
+}
```
### Bionics
-| Identifier | Description
-|--- |---
-| id | (_mandatory_) Unique ID. Must be one continuous word, use underscores if necessary.
-| name | (_mandatory_) In-game name displayed.
-| description | (_mandatory_) In-game description.
-| flags | (_optional_) A list of flags. See JSON_FLAGS.md for supported values.
-| act_cost | (_optional_) How many kJ it costs to activate the bionic. Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`)
-| deact_cost | (_optional_) How many kJ it costs to deactivate the bionic. Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`)
-| react_cost | (_optional_) How many kJ it costs over time to keep this bionic active, does nothing without a non-zero "time". Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`)
-| time | (_optional_) How long, when activated, between drawing cost. If 0, it draws power once. (default: `0`)
-| upgraded_bionic | (_optional_) Bionic that can be upgraded by installing this one.
-| available_upgrades | (_optional_) Upgrades available for this bionic, i.e. the list of bionics having this one referenced by `upgraded_bionic`.
-| encumbrance | (_optional_) A list of body parts and how much this bionic encumber them.
-| weight_capacity_bonus | (_optional_) Bonus to weight carrying capacity in grams, can be negative. Strings can be used - "5000 g" or "5 kg" (default: `0`)
-| weight_capacity_modifier | (_optional_) Factor modifying base weight carrying capacity. (default: `1`)
-| canceled_mutations | (_optional_) A list of mutations/traits that are removed when this bionic is installed (e.g. because it replaces the fault biological part).
-| included_bionics | (_optional_) Additional bionics that are installed automatically when this bionic is installed. This can be used to install several bionics from one CBM item, which is useful as each of those can be activated independently.
-| included | (_optional_) Whether this bionic is included with another. If true this bionic does not require a CBM item to be defined. (default: `false`)
-| env_protec | (_optional_) How much environmental protection does this bionic provide on the specified body parts.
-| bash_protec | (_optional_) How much bash protection does this bionic provide on the specified body parts.
-| cut_protec | (_optional_) How much cut protection does this bionic provide on the specified body parts.
-| bullet_protect | (_optional_) How much bullet protect does this bionic provide on the specified body parts.
-| occupied_bodyparts | (_optional_) A list of body parts occupied by this bionic, and the number of bionic slots it take on those parts.
-| capacity | (_optional_) Amount of power storage added by this bionic. Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`)
-| fuel_options | (_optional_) A list of fuel that this bionic can use to produce bionic power.
-| is_remote_fueled | (_optional_) If true this bionic allows you to plug your power banks to an external power source (solar backpack, UPS, vehicle etc) via a cable. (default: `false`)
-| fuel_capacity | (_optional_) Volume of fuel this bionic can store.
-| fuel_efficiency | (_optional_) Fraction of fuel energy converted into power. (default: `0`)
-| fuel_multiplier | (_optional_) Multiplies the amount of fuel when loading into the bionic (default: `1`)
-| passive_fuel_efficiency | (_optional_) Fraction of fuel energy passively converted into power. Useful for CBM using PERPETUAL fuel like `muscle`, `wind` or `sun_light`. (default: `0`)
-| exothermic_power_gen | (_optional_) If true this bionic emits heat when producing power. (default: `false`)
-| coverage_power_gen_penalty | (_optional_) Fraction of coverage diminishing fuel_efficiency. Float between 0.0 and 1.0. (default: `nullopt`)
-| power_gen_emission | (_optional_) `emit_id` of the field emitted by this bionic when it produces energy. Emit_ids are defined in `emit.json`.
-| stat_bonus | (_optional_) List of passive stat bonus. Stat are designated as follow: "DEX", "INT", "STR", "PER".
-| enchantments | (_optional_) List of enchantments applied by this CBM (see MAGIC.md for instructions on enchantment. NB: enchantments are not necessarily magic.)
-| learned_spells | (_optional_) List of spells (with levels) you gain when installing this CBM, and lose when you uninstall this CBM. Spell classes are automatically gained.
-| fake_item | (_optional_) ID of fake item used by this bionic. Mandatory for gun and weapon bionics.
+| Identifier | Description |
+| -------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| id | (_mandatory_) Unique ID. Must be one continuous word, use underscores if necessary. |
+| name | (_mandatory_) In-game name displayed. |
+| description | (_mandatory_) In-game description. |
+| flags | (_optional_) A list of flags. See JSON_FLAGS.md for supported values. |
+| act_cost | (_optional_) How many kJ it costs to activate the bionic. Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`) |
+| deact_cost | (_optional_) How many kJ it costs to deactivate the bionic. Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`) |
+| react_cost | (_optional_) How many kJ it costs over time to keep this bionic active, does nothing without a non-zero "time". Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`) |
+| time | (_optional_) How long, when activated, between drawing cost. If 0, it draws power once. (default: `0`) |
+| upgraded_bionic | (_optional_) Bionic that can be upgraded by installing this one. |
+| available_upgrades | (_optional_) Upgrades available for this bionic, i.e. the list of bionics having this one referenced by `upgraded_bionic`. |
+| encumbrance | (_optional_) A list of body parts and how much this bionic encumber them. |
+| weight_capacity_bonus | (_optional_) Bonus to weight carrying capacity in grams, can be negative. Strings can be used - "5000 g" or "5 kg" (default: `0`) |
+| weight_capacity_modifier | (_optional_) Factor modifying base weight carrying capacity. (default: `1`) |
+| canceled_mutations | (_optional_) A list of mutations/traits that are removed when this bionic is installed (e.g. because it replaces the fault biological part). |
+| included_bionics | (_optional_) Additional bionics that are installed automatically when this bionic is installed. This can be used to install several bionics from one CBM item, which is useful as each of those can be activated independently. |
+| included | (_optional_) Whether this bionic is included with another. If true this bionic does not require a CBM item to be defined. (default: `false`) |
+| env_protec | (_optional_) How much environmental protection does this bionic provide on the specified body parts. |
+| bash_protec | (_optional_) How much bash protection does this bionic provide on the specified body parts. |
+| cut_protec | (_optional_) How much cut protection does this bionic provide on the specified body parts. |
+| bullet_protect | (_optional_) How much bullet protect does this bionic provide on the specified body parts. |
+| occupied_bodyparts | (_optional_) A list of body parts occupied by this bionic, and the number of bionic slots it take on those parts. |
+| capacity | (_optional_) Amount of power storage added by this bionic. Strings can be used "1 kJ"/"1000 J"/"1000000 mJ" (default: `0`) |
+| fuel_options | (_optional_) A list of fuel that this bionic can use to produce bionic power. |
+| is_remote_fueled | (_optional_) If true this bionic allows you to plug your power banks to an external power source (solar backpack, UPS, vehicle etc) via a cable. (default: `false`) |
+| fuel_capacity | (_optional_) Volume of fuel this bionic can store. |
+| fuel_efficiency | (_optional_) Fraction of fuel energy converted into power. (default: `0`) |
+| fuel_multiplier | (_optional_) Multiplies the amount of fuel when loading into the bionic (default: `1`) |
+| passive_fuel_efficiency | (_optional_) Fraction of fuel energy passively converted into power. Useful for CBM using PERPETUAL fuel like `muscle`, `wind` or `sun_light`. (default: `0`) |
+| exothermic_power_gen | (_optional_) If true this bionic emits heat when producing power. (default: `false`) |
+| coverage_power_gen_penalty | (_optional_) Fraction of coverage diminishing fuel_efficiency. Float between 0.0 and 1.0. (default: `nullopt`) |
+| power_gen_emission | (_optional_) `emit_id` of the field emitted by this bionic when it produces energy. Emit_ids are defined in `emit.json`. |
+| stat_bonus | (_optional_) List of passive stat bonus. Stat are designated as follow: "DEX", "INT", "STR", "PER". |
+| enchantments | (_optional_) List of enchantments applied by this CBM (see MAGIC.md for instructions on enchantment. NB: enchantments are not necessarily magic.) |
+| learned_spells | (_optional_) List of spells (with levels) you gain when installing this CBM, and lose when you uninstall this CBM. Spell classes are automatically gained. |
+| fake_item | (_optional_) ID of fake item used by this bionic. Mandatory for gun and weapon bionics. |
```C++
{
@@ -552,16 +576,17 @@ This section describes each json file and their contents. Each json has their ow
}
```
-Bionics effects are defined in the code and new effects cannot be created through JSON alone.
-When adding a new bionic, if it's not included with another one, you must also add the corresponding CBM item in `data/json/items/bionics.json`. Even for a faulty bionic.
+Bionics effects are defined in the code and new effects cannot be created through JSON alone. When
+adding a new bionic, if it's not included with another one, you must also add the corresponding CBM
+item in `data/json/items/bionics.json`. Even for a faulty bionic.
### Dreams
-| Identifier | Description
-|--- |---
-| messages | List of potential dreams.
-| category | Mutation category needed to dream.
-| strength | Mutation category strength required (1 = 20-34, 2 = 35-49, 3 = 50+).
+| Identifier | Description |
+| ---------- | -------------------------------------------------------------------- |
+| messages | List of potential dreams. |
+| category | Mutation category needed to dream. |
+| strength | Mutation category strength required (1 = 20-34, 2 = 35-49, 3 = 50+). |
```C++
{
@@ -576,30 +601,29 @@ When adding a new bionic, if it's not included with another one, you must also a
### Disease
-| Identifier | Description
-|--- |---
-| id | Unique ID. Must be one continuous word, use underscores if necessary.
-| min_duration | The minimum duration the disease can last. Uses strings "x m", "x s","x d".
-| max_duration | The maximum duration the disease can last.
-| min_intensity | The minimum intensity of the effect applied by the disease
-| max_intensity | The maximum intensity of the effect.
-| health_threshold | The amount of health above which one is immune to the disease. Must be between -200 and 200. (optional )
-| symptoms | The effect applied by the disease.
-| affected_bodyparts | The list of bodyparts on which the effect is applied. (optional, default to num_bp)
-
+| Identifier | Description |
+| ------------------ | -------------------------------------------------------------------------------------------------------- |
+| id | Unique ID. Must be one continuous word, use underscores if necessary. |
+| min_duration | The minimum duration the disease can last. Uses strings "x m", "x s","x d". |
+| max_duration | The maximum duration the disease can last. |
+| min_intensity | The minimum intensity of the effect applied by the disease |
+| max_intensity | The maximum intensity of the effect. |
+| health_threshold | The amount of health above which one is immune to the disease. Must be between -200 and 200. (optional ) |
+| symptoms | The effect applied by the disease. |
+| affected_bodyparts | The list of bodyparts on which the effect is applied. (optional, default to num_bp) |
```json
- {
- "type": "disease_type",
- "id": "bad_food",
- "min_duration": "6 m",
- "max_duration": "1 h",
- "min_intensity": 1,
- "max_intensity": 1,
- "affected_bodyparts": [ "TORSO" ],
- "health_threshold": 100,
- "symptoms": "foodpoison"
- }
+{
+ "type": "disease_type",
+ "id": "bad_food",
+ "min_duration": "6 m",
+ "max_duration": "1 h",
+ "min_intensity": 1,
+ "max_intensity": 1,
+ "affected_bodyparts": ["TORSO"],
+ "health_threshold": 100,
+ "symptoms": "foodpoison"
+}
```
### Item Groups
@@ -607,11 +631,11 @@ When adding a new bionic, if it's not included with another one, you must also a
Item groups have been expanded, look at [the detailed docs](ITEM_SPAWN.md) to their new description.
The syntax listed here is still valid.
-| Identifier | Description
-|--- |---
-| id | Unique ID. Must be one continuous word, use underscores if necessary
-| items | List of potential item ID's. Chance of an item spawning is x/T, where X is the value linked to the specific item and T is the total of all item values in a group.
-| groups | ??
+| Identifier | Description |
+| ---------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
+| id | Unique ID. Must be one continuous word, use underscores if necessary |
+| items | List of potential item ID's. Chance of an item spawning is x/T, where X is the value linked to the specific item and T is the total of all item values in a group. |
+| groups | ?? |
```C++
{
@@ -632,13 +656,13 @@ The syntax listed here is still valid.
When you sort your inventory by category, these are the categories that are displayed.
-| Identifier | Description
-|--- |---
-| id | Unique ID. Must be one continuous word, use underscores if necessary
-| name | The name of the category. This is what shows up in-game when you open the inventory.
-| zone | The corresponding loot_zone (see loot_zones.json)
-| sort_rank | Used to sort categories when displaying. Lower values are shown first
-| priority_zones | When set, items in this category will be sorted to the priority zone if the conditions are met. If the user does not have the priority zone in the zone manager, the items get sorted into zone set in the 'zone' property. It is a list of objects. Each object has 3 properties: ID: The id of a LOOT_ZONE (see LOOT_ZONES.json), filthy: boolean. setting this means filthy items of this category will be sorted to the priority zone, flags: array of flags
+| Identifier | Description |
+| -------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| id | Unique ID. Must be one continuous word, use underscores if necessary |
+| name | The name of the category. This is what shows up in-game when you open the inventory. |
+| zone | The corresponding loot_zone (see loot_zones.json) |
+| sort_rank | Used to sort categories when displaying. Lower values are shown first |
+| priority_zones | When set, items in this category will be sorted to the priority zone if the conditions are met. If the user does not have the priority zone in the zone manager, the items get sorted into zone set in the 'zone' property. It is a list of objects. Each object has 3 properties: ID: The id of a LOOT_ZONE (see LOOT_ZONES.json), filthy: boolean. setting this means filthy items of this category will be sorted to the priority zone, flags: array of flags |
```C++
{
@@ -652,35 +676,36 @@ When you sort your inventory by category, these are the categories that are disp
### Materials
-| Identifier | Description
-|--- |---
-| `id` | Unique ID. Lowercase snake_case. Must be one continuous word, use underscores if necessary.
-| `name` | In-game name displayed.
-| `bash_resist` | How well a material resists bashing damage.
-| `cut_resist` | How well a material resists cutting damage.
-| `bullet_resist` | How well a material resists bullet damage.
-| `acid_resist` | Ability of a material to resist acid.
-| `elec_resist` | Ability of a material to resist electricity.
-| `fire_resist` | Ability of a material to resist fire.
-| `chip_resist` | Returns resistance to being damaged by attacks against the item itself.
-| `bash_dmg_verb` | Verb used when material takes bashing damage.
-| `cut_dmg_verb` | Verb used when material takes cutting damage.
-| `dmg_adj` | Description added to damaged item in ascending severity.
-| `dmg_adj` | Adjectives used to describe damage states of a material.
-| `density` | Density of a material.
-| `vitamins` | Vitamins in a material. Usually overridden by item specific values.
-| `wind_resist` | Percentage 0-100. How effective this material is at stopping wind from getting through. Higher values are better. If none of the materials an item is made of specify a value, a default of 99 is assumed.
-| `warmth_when_wet` | Percentage of warmth retained when fully drenched. Default is 0.2.
-| `specific_heat_liquid` | Specific heat of a material when not frozen (J/(g K)). Default 4.186.
-| `specific_heat_solid` | Specific heat of a material when frozen (J/(g K)). Default 2.108.
-| `latent_heat` | Latent heat of fusion for a material (J/g). Default 334.
-| `freeze_point` | Freezing point of this material (F). Default 32 F ( 0 C ).
-| `edible` | Optional boolean. Default is false.
-| `rotting` | Optional boolean. Default is false.
-| `soft` | Optional boolean. Default is false.
-| `reinforces` | Optional boolean. Default is false.
-
-There are six -resist parameters: acid, bash, chip, cut, elec, and fire. These are integer values; the default is 0 and they can be negative to take more damage.
+| Identifier | Description |
+| ---------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `id` | Unique ID. Lowercase snake_case. Must be one continuous word, use underscores if necessary. |
+| `name` | In-game name displayed. |
+| `bash_resist` | How well a material resists bashing damage. |
+| `cut_resist` | How well a material resists cutting damage. |
+| `bullet_resist` | How well a material resists bullet damage. |
+| `acid_resist` | Ability of a material to resist acid. |
+| `elec_resist` | Ability of a material to resist electricity. |
+| `fire_resist` | Ability of a material to resist fire. |
+| `chip_resist` | Returns resistance to being damaged by attacks against the item itself. |
+| `bash_dmg_verb` | Verb used when material takes bashing damage. |
+| `cut_dmg_verb` | Verb used when material takes cutting damage. |
+| `dmg_adj` | Description added to damaged item in ascending severity. |
+| `dmg_adj` | Adjectives used to describe damage states of a material. |
+| `density` | Density of a material. |
+| `vitamins` | Vitamins in a material. Usually overridden by item specific values. |
+| `wind_resist` | Percentage 0-100. How effective this material is at stopping wind from getting through. Higher values are better. If none of the materials an item is made of specify a value, a default of 99 is assumed. |
+| `warmth_when_wet` | Percentage of warmth retained when fully drenched. Default is 0.2. |
+| `specific_heat_liquid` | Specific heat of a material when not frozen (J/(g K)). Default 4.186. |
+| `specific_heat_solid` | Specific heat of a material when frozen (J/(g K)). Default 2.108. |
+| `latent_heat` | Latent heat of fusion for a material (J/g). Default 334. |
+| `freeze_point` | Freezing point of this material (F). Default 32 F ( 0 C ). |
+| `edible` | Optional boolean. Default is false. |
+| `rotting` | Optional boolean. Default is false. |
+| `soft` | Optional boolean. Default is false. |
+| `reinforces` | Optional boolean. Default is false. |
+
+There are six -resist parameters: acid, bash, chip, cut, elec, and fire. These are integer values;
+the default is 0 and they can be negative to take more damage.
```C++
{
@@ -716,25 +741,25 @@ There are six -resist parameters: acid, bash, chip, cut, elec, and fire. These a
#### Group definition
-| Identifier | Description
-|--- |---
-| `name` | Unique ID. Must be one continuous word, use underscores if necessary.
-| `default` | Default monster, automatically fills in any remaining spawn chances.
-| `monsters` | To choose a monster for spawning, the game creates 1000 entries and picks one. Each monster will have a number of entries equal to it's "freq" and the default monster will fill in the remaining. See the table below for how to build the single monster definitions.
-| `is_safe` | (bool) Check to not trigger safe-mode warning.
-| `is_animal` | (bool) Check if that group has only normal animals.
+| Identifier | Description |
+| ----------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `name` | Unique ID. Must be one continuous word, use underscores if necessary. |
+| `default` | Default monster, automatically fills in any remaining spawn chances. |
+| `monsters` | To choose a monster for spawning, the game creates 1000 entries and picks one. Each monster will have a number of entries equal to it's "freq" and the default monster will fill in the remaining. See the table below for how to build the single monster definitions. |
+| `is_safe` | (bool) Check to not trigger safe-mode warning. |
+| `is_animal` | (bool) Check if that group has only normal animals. |
#### Monster definition
-| Identifier | Description
-|--- |---
-| `monster` | The monster's unique ID, eg. `"mon_zombie"`.
-| `freq` | Chance of occurrence, x/1000.
-| `cost_multiplier` | How many monsters each monster in this definition should count as, if spawning a limited number of monsters.
-| `pack_size` | (_optional_) The minimum and maximum number of monsters in this group that should spawn together. (default: `[1,1]`)
-| `conditions` | Conditions limit when monsters spawn. Valid options: `SUMMER`, `WINTER`, `AUTUMN`, `SPRING`, `DAY`, `NIGHT`, `DUSK`, `DAWN`. Multiple Time-of-day conditions (`DAY`, `NIGHT`, `DUSK`, `DAWN`) will be combined together so that any of those conditions makes the spawn valid. Multiple Season conditions (`SUMMER`, `WINTER`, `AUTUMN`, `SPRING`) will be combined together so that any of those conditions makes the spawn valid.
-| `starts` | (_optional_) This entry becomes active after this time. (Measured in hours)
-| `ends` | (_optional_) This entry becomes inactive after this time. (Measured in hours)
+| Identifier | Description |
+| ----------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `monster` | The monster's unique ID, eg. `"mon_zombie"`. |
+| `freq` | Chance of occurrence, x/1000. |
+| `cost_multiplier` | How many monsters each monster in this definition should count as, if spawning a limited number of monsters. |
+| `pack_size` | (_optional_) The minimum and maximum number of monsters in this group that should spawn together. (default: `[1,1]`) |
+| `conditions` | Conditions limit when monsters spawn. Valid options: `SUMMER`, `WINTER`, `AUTUMN`, `SPRING`, `DAY`, `NIGHT`, `DUSK`, `DAWN`. Multiple Time-of-day conditions (`DAY`, `NIGHT`, `DUSK`, `DAWN`) will be combined together so that any of those conditions makes the spawn valid. Multiple Season conditions (`SUMMER`, `WINTER`, `AUTUMN`, `SPRING`) will be combined together so that any of those conditions makes the spawn valid. |
+| `starts` | (_optional_) This entry becomes active after this time. (Measured in hours) |
+| `ends` | (_optional_) This entry becomes inactive after this time. (Measured in hours) |
```C++
{
@@ -751,14 +776,14 @@ There are six -resist parameters: acid, bash, chip, cut, elec, and fire. These a
### Monster Factions
-| Identifier | Description
-|--- |---
-| `name` | Unique ID. Must be one continuous word, use underscores when necessary.
-| `base_faction` | Optional base faction. Relations to other factions are inherited from it and relations of other factions to this one check this.
-| `by_mood` | Be hostile towards this faction when angry, neutral otherwise. Default attitude to all other factions.
-| `neutral` | Always be neutral towards this faction.
-| `friendly` | Always be friendly towards this faction. By default a faction is friendly towards itself.
-| `hate` | Always be hostile towards this faction. Will change target to monsters of this faction if available.
+| Identifier | Description |
+| -------------- | -------------------------------------------------------------------------------------------------------------------------------- |
+| `name` | Unique ID. Must be one continuous word, use underscores when necessary. |
+| `base_faction` | Optional base faction. Relations to other factions are inherited from it and relations of other factions to this one check this. |
+| `by_mood` | Be hostile towards this faction when angry, neutral otherwise. Default attitude to all other factions. |
+| `neutral` | Always be neutral towards this faction. |
+| `friendly` | Always be friendly towards this faction. By default a faction is friendly towards itself. |
+| `hate` | Always be hostile towards this faction. Will change target to monsters of this faction if available. |
```C++
{
@@ -784,9 +809,12 @@ See MONSTERS.md
### Profession item substitution
-Defines item replacements that are applied to the starting items based upon the starting traits. This allows for example to replace wool items with non-wool items when the characters starts with the wool allergy trait.
+Defines item replacements that are applied to the starting items based upon the starting traits.
+This allows for example to replace wool items with non-wool items when the characters starts with
+the wool allergy trait.
-If the JSON objects contains a "item" member, it defines a replacement for the given item, like this:
+If the JSON objects contains a "item" member, it defines a replacement for the given item, like
+this:
```C++
{
@@ -798,12 +826,16 @@ If the JSON objects contains a "item" member, it defines a replacement for the g
]
}
```
+
This defines each item of type "sunglasses" shall be replaced with:
+
- an item "fitover_sunglasses" if the character has the "HYPEROPIC" trait,
- two items "fitover_sunglasses" if the character has the "MYOPIC" trait.
-If the JSON objects contains a "trait" member, it defines a replacement for multiple items that applies when the character has the given trait:
-```C++
+If the JSON objects contains a "trait" member, it defines a replacement for multiple items that
+applies when the character has the given trait:
+
+````C++
{
"type": "profession_item_substitutions",
"trait": "WOOLALLERGY",
@@ -827,21 +859,25 @@ Professions are specified as JSON object with "type" member set to "profession":
"id": "hunter",
...
}
-```
+````
The id member should be the unique id of the profession.
The following properties (mandatory, except if noted otherwise) are supported:
#### `description`
+
(string)
The in-game description.
#### `name`
+
(string or object with members "male" and "female")
-The in-game name, either one gender-neutral string, or an object with gender specific names. Example:
+The in-game name, either one gender-neutral string, or an object with gender specific names.
+Example:
+
```C++
"name": {
"male": "Groom",
@@ -850,25 +886,31 @@ The in-game name, either one gender-neutral string, or an object with gender spe
```
#### `points`
+
(integer)
Point cost of profession. Positive values cost points and negative values grant points.
#### `addictions`
+
(optional, array of addictions)
List of starting addictions. Each entry in the list should be an object with the following members:
+
- "type": the string id of the addiction (see JSON_FLAGS.md),
- "intensity": intensity (integer) of the addiction.
Example:
+
```C++
"addictions": [
{ "type": "nicotine", "intensity": 10 }
]
```
-Mods can modify this list (requires `"edit-mode": "modify"`, see example) via "add:addictions" and "remove:addictions", removing requires only the addiction type. Example:
+Mods can modify this list (requires `"edit-mode": "modify"`, see example) via "add:addictions" and
+"remove:addictions", removing requires only the addiction type. Example:
+
```C++
{
"type": "profession",
@@ -888,17 +930,22 @@ Mods can modify this list (requires `"edit-mode": "modify"`, see example) via "a
(optional, array of skill levels)
List of starting skills. Each entry in the list should be an object with the following members:
+
- "name": the string id of the skill (see skills.json),
-- "level": level (integer) of the skill. This is added to the skill level that can be chosen in the character creation.
+- "level": level (integer) of the skill. This is added to the skill level that can be chosen in the
+ character creation.
Example:
+
```C++
"skills": [
{ "name": "archery", "level": 2 }
]
```
-Mods can modify this list (requires `"edit-mode": "modify"`, see example) via "add:skills" and "remove:skills", removing requires only the skill id. Example:
+Mods can modify this list (requires `"edit-mode": "modify"`, see example) via "add:skills" and
+"remove:skills", removing requires only the skill id. Example:
+
```C++
{
"type": "profession",
@@ -911,16 +958,19 @@ Mods can modify this list (requires `"edit-mode": "modify"`, see example) via "a
{ "name": "computer", "level": 2 }
]
}
-
```
#### `items`
(optional, object with optional members "both", "male" and "female")
-Items the player starts with when selecting this profession. One can specify different items based on the gender of the character. Each lists of items should be an array of items ids, or pairs of item ids and snippet ids. Item ids may appear multiple times, in which case the item is created multiple times. The syntax for each of the three lists is identical.
+Items the player starts with when selecting this profession. One can specify different items based
+on the gender of the character. Each lists of items should be an array of items ids, or pairs of
+item ids and snippet ids. Item ids may appear multiple times, in which case the item is created
+multiple times. The syntax for each of the three lists is identical.
Example:
+
```C++
"items": {
"both": [
@@ -939,9 +989,13 @@ Example:
}
```
-This gives the player pants, two rocks, a t-shirt with the snippet id "allyourbase" (giving it a special description), socks and (depending on the gender) briefs or panties.
+This gives the player pants, two rocks, a t-shirt with the snippet id "allyourbase" (giving it a
+special description), socks and (depending on the gender) briefs or panties.
-Mods can modify the lists of existing professions. This requires the "edit-mode" member with value "modify" (see example). Adding items to the lists can be done with via "add:both" / "add:male" / "add:female". It allows the same content (it allows adding items with snippet ids). Removing items is done via "remove:both" / "remove:male" / "remove:female", which may only contain items ids.
+Mods can modify the lists of existing professions. This requires the "edit-mode" member with value
+"modify" (see example). Adding items to the lists can be done with via "add:both" / "add:male" /
+"add:female". It allows the same content (it allows adding items with snippet ids). Removing items
+is done via "remove:both" / "remove:male" / "remove:female", which may only contain items ids.
Example for mods:
@@ -963,22 +1017,22 @@ Example for mods:
}
```
-This mod removes one of the rocks (the other rock is still created), the t-shirt, adds a 2x4 item and gives female characters a t-shirt with the special snippet id.
+This mod removes one of the rocks (the other rock is still created), the t-shirt, adds a 2x4 item
+and gives female characters a t-shirt with the special snippet id.
#### `pets`
(optional, array of string mtype_ids )
-A list of strings, each is the same as a monster id
-player will start with these as tamed pets.
+A list of strings, each is the same as a monster id player will start with these as tamed pets.
#### `vehicle`
(optional, string vproto_id )
-A string, which is the same as a vehicle ( vproto_id )
-player will start with this as a nearby vehicle.
-( it will find the nearest road and place it there, then mark it as "remembered" on the overmap )
+A string, which is the same as a vehicle ( vproto_id ) player will start with this as a nearby
+vehicle. ( it will find the nearest road and place it there, then mark it as "remembered" on the
+overmap )
#### `flags`
@@ -1080,12 +1134,10 @@ Recipes represent both craft and uncraft (disassembly) recipes.
#### Overlapping recipe component requirements
-If recipes have requirements which overlap, this makes it more
-difficult for the game to calculate whether it is possible to craft a recipe at
-all.
+If recipes have requirements which overlap, this makes it more difficult for the game to calculate
+whether it is possible to craft a recipe at all.
-For example, the survivor telescope recipe has the following requirements
-(amongst others):
+For example, the survivor telescope recipe has the following requirements (amongst others):
```
1 high-quality lens
@@ -1095,24 +1147,22 @@ AND
These overlap because both list the high-quality lens.
-A small amount of overlap (such as the above) can be handled, but if you have
-too many component lists which overlap in too many ways, then you may see an
-error during recipe finalization that your recipe is too complex. In this
-case, the game may not be able to corectly predict whether it can be crafted.
+A small amount of overlap (such as the above) can be handled, but if you have too many component
+lists which overlap in too many ways, then you may see an error during recipe finalization that your
+recipe is too complex. In this case, the game may not be able to corectly predict whether it can be
+crafted.
-To work around this issue, if you do not wish to simplify the recipe
-requirements, then you can split your recipe into multiple steps. For
-example, if we wanted to simplify the above survivor telescope recipe we could
-introduce an intermediate item "survivor eyepiece", which requires one of
-either lens, and then the telescope would require a high-quality lens and an
-eyepiece. Overall, the requirements are the same, but neither recipe has any
-overlap.
+To work around this issue, if you do not wish to simplify the recipe requirements, then you can
+split your recipe into multiple steps. For example, if we wanted to simplify the above survivor
+telescope recipe we could introduce an intermediate item "survivor eyepiece", which requires one of
+either lens, and then the telescope would require a high-quality lens and an eyepiece. Overall, the
+requirements are the same, but neither recipe has any overlap.
-For more details, see [this pull
-request](https://github.com/CleverRaven/Cataclysm-DDA/pull/36657) and the
-[related issue](https://github.com/CleverRaven/Cataclysm-DDA/issues/32311).
+For more details, see [this pull request](https://github.com/CleverRaven/Cataclysm-DDA/pull/36657)
+and the [related issue](https://github.com/CleverRaven/Cataclysm-DDA/issues/32311).
### Constructions
+
```C++
"id": "constr_pit_spiked", // Identifier of the construction
"group": "spike_pit", // Construction group, used to provide description and group related constructions in UI (e.g. different stages of some construction).
@@ -1134,24 +1184,24 @@ request](https://github.com/CleverRaven/Cataclysm-DDA/pull/36657) and the
"vehicle_start": false, // (Optional, default false) Whether it will create a vehicle (for hardcode purposes)
"on_display": false, // (Optional, default true) Whether it will show in player's UI
"dark_craftable": true, // (Optional, default false) Whether it can be constructed in dark
-
```
### Construction groups
+
```C++
"id": "build_wooden_door", // Group identifier
"name": "Build Wooden Door", // Description string displayed in the construction menu
```
### Construction sequences
-Constuction sequences are needed for automatic calculation of basecamp blueprint requirements.
-Each construction sequence describes steps needed to construct specified terrain
-or furniture on an empty dirt tile.
-At any moment, only 1 construction sequence may be defined for any terrain or furniture.
+
+Constuction sequences are needed for automatic calculation of basecamp blueprint requirements. Each
+construction sequence describes steps needed to construct specified terrain or furniture on an empty
+dirt tile. At any moment, only 1 construction sequence may be defined for any terrain or furniture.
For convenience, each non-blacklisted construction recipe that starts from an empty dirt tile
-automatically generates a one-element-long construction sequence, but only as long as
-there are no other such construction recipes that would produce the same sequence
-and there isn't an explicitly defined sequence with same results.
+automatically generates a one-element-long construction sequence, but only as long as there are no
+other such construction recipes that would produce the same sequence and there isn't an explicitly
+defined sequence with same results.
```C++
"id": "f_workbench", // Sequence identifier
@@ -1167,30 +1217,29 @@ and there isn't an explicitly defined sequence with same results.
### Scent_types
-| Identifier | Description
-|--- |---
-| id | Unique ID. Must be one continuous word, use underscores if necessary.
-| receptive_species | Species able to track this scent. Must use valid ids defined in `species.json`
+| Identifier | Description |
+| ----------------- | ------------------------------------------------------------------------------ |
+| id | Unique ID. Must be one continuous word, use underscores if necessary. |
+| receptive_species | Species able to track this scent. Must use valid ids defined in `species.json` |
```json
- {
- "type": "scent_type",
- "id": "sc_flower",
- "receptive_species": [ "MAMMAL", "INSECT", "MOLLUSK", "BIRD" ]
- }
+{
+ "type": "scent_type",
+ "id": "sc_flower",
+ "receptive_species": ["MAMMAL", "INSECT", "MOLLUSK", "BIRD"]
+}
```
### Scores and Achievements
-Scores are defined in two or three steps based on *events*. To see what events
-exist and what data they contain, read [`event.h`](../src/event.h).
+Scores are defined in two or three steps based on _events_. To see what events exist and what data
+they contain, read [`event.h`](../src/event.h).
-Each event contains a certain set of fields. Each field has a string key and a
-`cata_variant` value. The fields should provide all the relevant information
-about the event.
+Each event contains a certain set of fields. Each field has a string key and a `cata_variant` value.
+The fields should provide all the relevant information about the event.
-For example, consider the `gains_skill_level` event. You can see this
-specification for it in `event.h`:
+For example, consider the `gains_skill_level` event. You can see this specification for it in
+`event.h`:
```C++
template<>
@@ -1205,39 +1254,37 @@ struct event_spec {
```
From this, you can see that this event type has three fields:
-* `character`, with the id of the character gaining the level.
-* `skill`, with the id of the skill gained.
-* `new_level`, with the integer level newly acquired in that skill.
-
-Events are generated by the game when in-game circumstances dictate. These
-events can be transformed and summarized in various ways. There are three
-concepts involved: event streams, event statistics, and scores.
-
-* Each `event_type` defined by the game generates an event stream.
-* Further event streams can be defined in json by applying an
- `event_transformation` to an existing event stream.
-* An `event_statistic` summarizes an event stream into a single value (usually
- a number, but other types of value are possible).
-* A `score` uses such a statistic to define an in-game score which players can
- see.
+
+- `character`, with the id of the character gaining the level.
+- `skill`, with the id of the skill gained.
+- `new_level`, with the integer level newly acquired in that skill.
+
+Events are generated by the game when in-game circumstances dictate. These events can be transformed
+and summarized in various ways. There are three concepts involved: event streams, event statistics,
+and scores.
+
+- Each `event_type` defined by the game generates an event stream.
+- Further event streams can be defined in json by applying an `event_transformation` to an existing
+ event stream.
+- An `event_statistic` summarizes an event stream into a single value (usually a number, but other
+ types of value are possible).
+- A `score` uses such a statistic to define an in-game score which players can see.
#### `event_transformation`
-An `event_transformation` can modify an event stream, producing another event
-stream.
+An `event_transformation` can modify an event stream, producing another event stream.
-The input stream to be transformed is specified either as an `"event_type"`, to
-use one of the built-in event type streams, or an `"event_transformation"`,
-to use another json-defined transformed event stream.
+The input stream to be transformed is specified either as an `"event_type"`, to use one of the
+built-in event type streams, or an `"event_transformation"`, to use another json-defined transformed
+event stream.
Any or all of the following alterations can be made to the event stream:
-* Add new fields to each event based on event field transformations. The event
- field transformations can be found in
- [`event_field_transformation.cpp`](../src/event_field_transformation.cpp).
-* Filter events based on the values they contain to produce a stream containing
- some subset of the input stream.
-* Drop some fields which are not of interest in the output stream.
+- Add new fields to each event based on event field transformations. The event field transformations
+ can be found in [`event_field_transformation.cpp`](../src/event_field_transformation.cpp).
+- Filter events based on the values they contain to produce a stream containing some subset of the
+ input stream.
+- Drop some fields which are not of interest in the output stream.
Here are examples of each modification:
@@ -1275,49 +1322,54 @@ Here are examples of each modification:
#### `event_statistic`
-As with `event_transformation`, an `event_statistic` requires an input event
-stream. That input stream can be specified in the same was as for
-`event_transformation`, via one of the following two entries:
+As with `event_transformation`, an `event_statistic` requires an input event stream. That input
+stream can be specified in the same was as for `event_transformation`, via one of the following two
+entries:
```C++
"event_type" : "avatar_moves" // Events of this built-in type
"event_transformation" : "moves_on_horse" // Events resulting from this json-defined transformation
```
-Then it specifies a particular `stat_type` and potentially additional details
-as follows:
+Then it specifies a particular `stat_type` and potentially additional details as follows:
The number of events:
+
```C++
"stat_type" : "count"
```
The sum of the numeric value in the specified field across all events:
+
```C++
"stat_type" : "total"
"field" : "damage"
```
The maximum of the numeric value in the specified field across all events:
+
```C++
"stat_type" : "maximum"
"field" : "damage"
```
The minimum of the numeric value in the specified field across all events:
+
```C++
"stat_type" : "minimum"
"field" : "damage"
```
-Assume there is only a single event to consider, and take the value of the
-given field for that unique event:
+Assume there is only a single event to consider, and take the value of the given field for that
+unique event:
+
```C++
"stat_type": "unique_value",
"field": "avatar_id"
```
Regardless of `stat_type`, each `event_statistic` can also have:
+
```C++
// Intended for use in describing scores and achievement requirements.
"description": "Number of things"
@@ -1325,15 +1377,14 @@ Regardless of `stat_type`, each `event_statistic` can also have:
#### `score`
-Scores simply associate a description to an event for formatting in tabulations
-of scores. The `description` specifies a string which is expected to contain a
-`%s` format specifier where the value of the statistic will be inserted.
+Scores simply associate a description to an event for formatting in tabulations of scores. The
+`description` specifies a string which is expected to contain a `%s` format specifier where the
+value of the statistic will be inserted.
-Note that even though most statistics yield an integer, you should still use
-`%s`.
+Note that even though most statistics yield an integer, you should still use `%s`.
-If the underlying statistic has a description, then the score description is
-optional. It defaults to ": ".
+If the underlying statistic has a description, then the score description is optional. It defaults
+to ": ".
```C++
"id": "score_headshots",
@@ -1344,11 +1395,11 @@ optional. It defaults to ": ".
#### `achievement`
-Achievements are goals for the player to aspire to, in the usual sense of the
-term as popularised in other games.
+Achievements are goals for the player to aspire to, in the usual sense of the term as popularised in
+other games.
-An achievement is specified via requirements, each of which is a constraint on
-an `event_statistic`. For example:
+An achievement is specified via requirements, each of which is a constraint on an `event_statistic`.
+For example:
```C++
{
@@ -1366,8 +1417,8 @@ an `event_statistic`. For example:
},
```
-The `"is"` field must be `">="`, `"<="` or `"anything"`. When it is not
-`"anything"` the `"target"` must be present, and must be an integer.
+The `"is"` field must be `">="`, `"<="` or `"anything"`. When it is not `"anything"` the `"target"`
+must be present, and must be an integer.
There are further optional fields:
@@ -1375,9 +1426,8 @@ There are further optional fields:
"hidden_by": [ "other_achievement_id" ]
```
-Give a list of other achievement ids. This achievement will be hidden (i.e.
-not appear in the achievements UI) until all of the achievements listed have
-been completed.
+Give a list of other achievement ids. This achievement will be hidden (i.e. not appear in the
+achievements UI) until all of the achievements listed have been completed.
Use this to prevent spoilers or to reduce clutter in the list of achievements.
@@ -1385,45 +1435,41 @@ Use this to prevent spoilers or to reduce clutter in the list of achievements.
"skill_requirements": [ { "skill": "archery", "is": ">=", "level": 5 } ]
```
-This allows a skill level requirement (either an upper or lower bound) on when
-the achievement can be claimed. The `"skill"` field uses the id of a skill.
-
-Note that like `"time_constraint"` below achievements can only be captured when
-a statistic listed in `"requirements"` changes.
+This allows a skill level requirement (either an upper or lower bound) on when the achievement can
+be claimed. The `"skill"` field uses the id of a skill.
+Note that like `"time_constraint"` below achievements can only be captured when a statistic listed
+in `"requirements"` changes.
```C++
"kill_requirements": [ { "faction": "ZOMBIE", "is": ">=", "count": 1 }, { "monster": "mon_sludge_crawler", "is": ">=", "count": 1 } ],
```
-This allows a kill requirement (either an upper or lower bound) on when
-the achievement can be claimed. Can be defined with either `"faction"` or
-`"monster"` as a target, using species ids from `species.json` or a
-specific monster id.
+This allows a kill requirement (either an upper or lower bound) on when the achievement can be
+claimed. Can be defined with either `"faction"` or `"monster"` as a target, using species ids from
+`species.json` or a specific monster id.
-Only one of the `"monster"`/`"faction"` fields can be used per entry. If neither
-are used, any monster will fulfill the requirements.
+Only one of the `"monster"`/`"faction"` fields can be used per entry. If neither are used, any
+monster will fulfill the requirements.
NPCs cannot currently be defined as a target.
-Note that like `"time_constraint"` below achievements can only be captured when
-a statistic listed in `"requirements"` changes.
+Note that like `"time_constraint"` below achievements can only be captured when a statistic listed
+in `"requirements"` changes.
```C++
"time_constraint": { "since": "game_start", "is": "<=", "target": "1 minute" }
```
-This allows putting a time limit (either a lower or upper bound) on when the
-achievement can be claimed. The `"since"` field can be either `"game_start"`
-or `"cataclysm"`. The `"target"` describes an amount of time since that
-reference point.
+This allows putting a time limit (either a lower or upper bound) on when the achievement can be
+claimed. The `"since"` field can be either `"game_start"` or `"cataclysm"`. The `"target"` describes
+an amount of time since that reference point.
-Note that achievements can only be captured when a statistic listed in their
-requirements changes. So, if you want an achievement which would normally be
-triggered by reaching some time threshold (such as "survived a certain amount
-of time") then you must place some requirement alongside it to trigger it after
-that time has passed. Pick some statistic which is likely to change often, and
-add an `"anything"` constraint on it. For example:
+Note that achievements can only be captured when a statistic listed in their requirements changes.
+So, if you want an achievement which would normally be triggered by reaching some time threshold
+(such as "survived a certain amount of time") then you must place some requirement alongside it to
+trigger it after that time has passed. Pick some statistic which is likely to change often, and add
+an `"anything"` constraint on it. For example:
```C++
{
@@ -1435,8 +1481,8 @@ add an `"anything"` constraint on it. For example:
},
```
-This is a simple "survive a day" but is triggered by waking up, so it will be
-completed when you wake up for the first time after 24 hours into the game.
+This is a simple "survive a day" but is triggered by waking up, so it will be completed when you
+wake up for the first time after 24 hours into the game.
### Skills
@@ -1552,7 +1598,6 @@ completed when you wake up for the first time after 24 hours into the game.
### Vehicle Groups
-
```C++
"id":"city_parked", // Unique ID. Must be one continuous word, use underscores if necessary
"vehicles":[ // List of potential vehicle ID's. Chance of a vehicle spawning is X/T, where
@@ -1635,6 +1680,7 @@ Vehicle components when installed on a vehicle.
```
### Vehicle Placement
+
```C++
"id":"road_straight_wrecks", // Unique ID. Must be one continuous word, use underscores if necessary
"locations":[ { // List of potential vehicle locations. When this placement is used, one of those locations will be chosen at random.
@@ -1665,6 +1711,7 @@ Vehicle components when installed on a vehicle.
```
### Vehicles
+
See also VEHICLE_JSON.md
```C++
@@ -1801,7 +1848,9 @@ Armor can be defined like this:
"power_armor" : false, // If this is a power armor item (those are special).
"valid_mods" : ["steel_padded"] // List of valid clothing mods. Note that if the clothing mod doesn't have "restricted" listed, this isn't needed.
```
+
Alternately, every item (book, tool, gun, even food) can be used as armor if it has armor_data:
+
```C++
"type" : "TOOL", // Or any other item type
... // same entries as for the type (e.g. same entries as for any tool),
@@ -1818,6 +1867,7 @@ Alternately, every item (book, tool, gun, even food) can be used as armor if it
```
### Pet Armor
+
Pet armor can be defined like this:
```C++
@@ -1832,7 +1882,9 @@ Pet armor can be defined like this:
"min_pet_vol: // the minimum volume of the pet that will fit into this armor. Volume in ml and L can be used - "50 ml" or "2 L".
"power_armor" : false, // If this is a power armor item (those are special).
```
+
Alternately, every item (book, tool, gun, even food) can be used as armor if it has armor_data:
+
```C++
"type" : "TOOL", // Or any other item type
... // same entries as for the type (e.g. same entries as for any tool),
@@ -1863,7 +1915,9 @@ Books can be defined like this:
"chapters" : 4, // Number of chapters (for fun only books), each reading "consumes" a chapter. Books with no chapters left are less fun (because the content is already known to the character).
"required_level" : 2 // Minimum skill level required to learn
```
+
Alternately, every item (tool, gun, even food) can be used as book if it has book_data:
+
```C++
"type" : "TOOL", // Or any other item type
... // same entries as for the type (e.g. same entries as for any tool),
@@ -1879,22 +1933,32 @@ Alternately, every item (tool, gun, even food) can be used as book if it has boo
}
```
-Since many book names are proper names, it's often necessary to explicitly specify
-the plural forms. The following is the game's convention on plural names of books:
+Since many book names are proper names, it's often necessary to explicitly specify the plural forms.
+The following is the game's convention on plural names of books:
1. For non-periodical books (textbooks, manuals, spellbooks, etc.),
- 1. If the book's singular name is a proper name, then the plural name is `copies of (singular name)`. For example, the plural name of `Lessons for the Novice Bowhunter` is `copies of Lessons for the Novice Bowhunter`.
- 2. Otherwise, the plural name is the usual plural of the singular name. For example, the plural name of `tactical baton defense manual` is `tactical baton defense manuals`
+ 1. If the book's singular name is a proper name, then the plural name is
+ `copies of (singular name)`. For example, the plural name of
+ `Lessons for the Novice Bowhunter` is `copies of Lessons for the Novice Bowhunter`.
+ 2. Otherwise, the plural name is the usual plural of the singular name. For example, the plural
+ name of `tactical baton defense manual` is `tactical baton defense manuals`
2. For periodicals (magazines and journals),
- 1. If the periodical's singular name is a proper name, and doesn't end with "Magazine", "Weekly", "Monthly", etc., the plural name is `issues of (singular name)`. For example, the plural name of `Archery for Kids` is `issues of Archery for Kids`.
- 2. Otherwise, the periodical's plural name is the usual plural of the singular name. For example, the plural name of `Crafty Crafter's Quarterly` is `Crafty Crafter's Quarterlies`.
+ 1. If the periodical's singular name is a proper name, and doesn't end with "Magazine", "Weekly",
+ "Monthly", etc., the plural name is `issues of (singular name)`. For example, the plural name
+ of `Archery for Kids` is `issues of Archery for Kids`.
+ 2. Otherwise, the periodical's plural name is the usual plural of the singular name. For example,
+ the plural name of `Crafty Crafter's Quarterly` is `Crafty Crafter's Quarterlies`.
3. For board games (represented internally as book items),
- 1. If the board game's singular name is a proper name, the plural is `sets of (singular name)`. For example, the plural name of `Picturesque` is `sets of Picturesque`.
- 2. Otherwise the plural name is the usual plural. For example, the plural of `deck of cards` is `decks of cards`.
+ 1. If the board game's singular name is a proper name, the plural is `sets of (singular name)`.
+ For example, the plural name of `Picturesque` is `sets of Picturesque`.
+ 2. Otherwise the plural name is the usual plural. For example, the plural of `deck of cards` is
+ `decks of cards`.
#### Conditional Naming
-The `conditional_names` field allows defining alternate names for items that will be displayed instead of (or in addition to) the default name, when specific conditions are met. Take the following (incomplete) definition for `sausage` as an example of the syntax:
+The `conditional_names` field allows defining alternate names for items that will be displayed
+instead of (or in addition to) the default name, when specific conditions are met. Take the
+following (incomplete) definition for `sausage` as an example of the syntax:
```json
{
@@ -1914,35 +1978,49 @@ The `conditional_names` field allows defining alternate names for items that wil
}
```
-You can list as many conditional names for a given item as you want. Each conditional name must consist of 3 elements:
+You can list as many conditional names for a given item as you want. Each conditional name must
+consist of 3 elements:
+
1. The condition type:
- - `COMPONENT_ID` searches all the components of the item (and all of *their* components, and so on) for an item with the condition string in their ID. The ID only needs to *contain* the condition, not match it perfectly (though it is case sensitive). For example, supplying a condition `mutant` would match `mutant_meat`.
- - `FLAG` which checks if an item has the specified flag (exact match).
+ - `COMPONENT_ID` searches all the components of the item (and all of _their_ components, and so
+ on) for an item with the condition string in their ID. The ID only needs to _contain_ the
+ condition, not match it perfectly (though it is case sensitive). For example, supplying a
+ condition `mutant` would match `mutant_meat`.
+ - `FLAG` which checks if an item has the specified flag (exact match).
2. The condition you want to look for.
-3. The name to use if a match is found. Follows all the rules of a standard `name` field, with valid keys being `str`, `str_pl`, and `ctxt`. You may use %s here, which will be replaced by the name of the item. Conditional names defined prior to this one are taken into account.
+3. The name to use if a match is found. Follows all the rules of a standard `name` field, with valid
+ keys being `str`, `str_pl`, and `ctxt`. You may use %s here, which will be replaced by the name
+ of the item. Conditional names defined prior to this one are taken into account.
-So, in the above example, if the sausage is made from mutant humanoid meat, and therefore both has the `CANNIBALISM` flag, *and* has a component with `mutant` in its ID:
-1. First, the item name is entirely replaced with "Mannwurst" if singular, or "Mannwursts" if plural.
-2. Next, it is replaced by "sinister %s", but %s is replaced with the name as it was before this step, resulting in "sinister Mannwurst" or "sinister Mannwursts".
+So, in the above example, if the sausage is made from mutant humanoid meat, and therefore both has
+the `CANNIBALISM` flag, _and_ has a component with `mutant` in its ID:
-NB: If `"str": "sinister %s"` was specified instead of `"str_sp": "sinister %s"`, the plural form would be automatically created as "sinister %ss", which would become "sinister Mannwurstss" which is of course one S too far. Rule of thumb: If you are using %s in the name, always specify an identical plural form unless you know exactly what you're doing!
+1. First, the item name is entirely replaced with "Mannwurst" if singular, or "Mannwursts" if
+ plural.
+2. Next, it is replaced by "sinister %s", but %s is replaced with the name as it was before this
+ step, resulting in "sinister Mannwurst" or "sinister Mannwursts".
+NB: If `"str": "sinister %s"` was specified instead of `"str_sp": "sinister %s"`, the plural form
+would be automatically created as "sinister %ss", which would become "sinister Mannwurstss" which is
+of course one S too far. Rule of thumb: If you are using %s in the name, always specify an identical
+plural form unless you know exactly what you're doing!
#### Color Key
When adding a new book, please use this color key:
-* Magazines: `pink`
-* “Paperbacks” Short enjoyment books (including novels): `light_cyan`
-* “Hardbacks” Long enjoyment books (including novels): `light_blue`
-* “Small textbook” Beginner level textbooks, guides and martial arts books: `green`
-* “Large textbook” Advanced level textbooks and advanced guides: `blue`
-* Religious books: `dark_gray`
-* “Printouts” (including spiral-bound, binders, and similar) Technical documents, (technical?) protocols, (lab) journals, personal diaries: `light_green`
-* Other reading material/non-books (use only if every other category does not apply): `light_gray`
+- Magazines: `pink`
+- “Paperbacks” Short enjoyment books (including novels): `light_cyan`
+- “Hardbacks” Long enjoyment books (including novels): `light_blue`
+- “Small textbook” Beginner level textbooks, guides and martial arts books: `green`
+- “Large textbook” Advanced level textbooks and advanced guides: `blue`
+- Religious books: `dark_gray`
+- “Printouts” (including spiral-bound, binders, and similar) Technical documents, (technical?)
+ protocols, (lab) journals, personal diaries: `light_green`
+- Other reading material/non-books (use only if every other category does not apply): `light_gray`
-A few exceptions to this color key may apply, for example for books that don’t are what they seem to be.
-Never use `yellow` and `red`, those colors are reserved for sounds and infrared vision.
+A few exceptions to this color key may apply, for example for books that don’t are what they seem to
+be. Never use `yellow` and `red`, those colors are reserved for sounds and infrared vision.
####CBMs
@@ -1998,7 +2076,9 @@ CBMs can be defined like this:
"watertight": false, // Can hold liquids, this is a required for it to be used for liquids. (optional, default: false)
"preserves": false, // Contents do not spoil. (optional, default: false)
```
+
Alternately, every item can be used as container:
+
```C++
"type": "ARMOR", // Any type is allowed here
... // same data as for the type
@@ -2006,8 +2086,10 @@ Alternately, every item can be used as container:
"contains": 200,
}
```
-This defines a armor (you need to add all the armor specific entries), but makes it usable as container.
-It could also be written as a generic item ("type": "GENERIC") with "armor_data" and "container_data" entries.
+
+This defines a armor (you need to add all the armor specific entries), but makes it usable as
+container. It could also be written as a generic item ("type": "GENERIC") with "armor_data" and
+"container_data" entries.
### Melee
@@ -2029,44 +2111,44 @@ It could also be written as a generic item ("type": "GENERIC") with "armor_data"
#### Melee `Weapon_category`
-| Weapon Category | Description
-| --- | ---
-| FIST_WEAPONS | Handheld weapons used to supplement fists in martial arts.
-| --- | ---
-| KNIVES | Short blade fixed onto a handle, for cutting or as weapon.
-| SHORT_SWORDS | One handed sword of length between a large knife and a 'proper' sword.
-| 1H_SWORDS | Sword meant to be wielded with one hand.
-| 2H_SWORDS | Sword meant to be wielded with both hands.
-| DUELING_SWORDS | Swords with thin profiles typically meant for stabbing.
-| BIONIC_SWORDS | Swords integrated into the body via bionics.
-| --- | ---
-| SAPS | Very short length of typically flexible material, with a weighted tip.
-| BATONS | Thin, balanced rod of strong material.
-| TONFAS | Unique looking T-shaped baton believed to originate from China.
-| CLUBS | Rod with a thicker striking head. May be one or two-handed.
-| QUARTERSTAVES | Long pole wielded with both hands.
-| MACES | Short one-handed weapon with a striking head firmly attached to a short handle.
-| MORNINGSTARS | Typically two-handed weapon with a striking head firmly attached to a long handle.
-| FLAILS | Striking head attached to handle by flexible rope/chain.
-| 1H_HAMMERS | Hammers meant to be wielded with a single hand.
-| 2H_HAMMERS | Hammers meant to be wielded with both hands.
-| --- | ---
-| HAND_AXES | Axe with a short handle, typically wielded in one hand, ocassionally thrown.
-| 1H_AXES | Axes meant to be wielded with one hand, typically with a handle longer than the handaxe.
-| 2H_AXES | Axes meant to be wielded with two hands.
-| --- | ---
-| WHIPS | Flexible tool used to strike at range.
-| --- | ---
-| HOOKED_POLES | Polearm with hooked end (Like a shepherd's crook)
-| SPEARS | Polearm with a long shaft and a sharp tip made of hard material.
-| PIKES | Very long spear that can only be wielded in two hands, very unwieldy.
-| GLAIVES | Polearm with a single-edged blade mounted on the end.
-
-| Weapon Styles | Description
-| --- | ---
-| MEDIEVAL_SWORDS | Swords associated with European culture.
-| JAPANESE_SWORDS | Swords associated with Japanese culture.
-| BIONIC_WEAPONRY | Weapons integrated into the body via bionics.
+| Weapon Category | Description |
+| --------------- | ---------------------------------------------------------------------------------------- |
+| FIST_WEAPONS | Handheld weapons used to supplement fists in martial arts. |
+| --- | --- |
+| KNIVES | Short blade fixed onto a handle, for cutting or as weapon. |
+| SHORT_SWORDS | One handed sword of length between a large knife and a 'proper' sword. |
+| 1H_SWORDS | Sword meant to be wielded with one hand. |
+| 2H_SWORDS | Sword meant to be wielded with both hands. |
+| DUELING_SWORDS | Swords with thin profiles typically meant for stabbing. |
+| BIONIC_SWORDS | Swords integrated into the body via bionics. |
+| --- | --- |
+| SAPS | Very short length of typically flexible material, with a weighted tip. |
+| BATONS | Thin, balanced rod of strong material. |
+| TONFAS | Unique looking T-shaped baton believed to originate from China. |
+| CLUBS | Rod with a thicker striking head. May be one or two-handed. |
+| QUARTERSTAVES | Long pole wielded with both hands. |
+| MACES | Short one-handed weapon with a striking head firmly attached to a short handle. |
+| MORNINGSTARS | Typically two-handed weapon with a striking head firmly attached to a long handle. |
+| FLAILS | Striking head attached to handle by flexible rope/chain. |
+| 1H_HAMMERS | Hammers meant to be wielded with a single hand. |
+| 2H_HAMMERS | Hammers meant to be wielded with both hands. |
+| --- | --- |
+| HAND_AXES | Axe with a short handle, typically wielded in one hand, ocassionally thrown. |
+| 1H_AXES | Axes meant to be wielded with one hand, typically with a handle longer than the handaxe. |
+| 2H_AXES | Axes meant to be wielded with two hands. |
+| --- | --- |
+| WHIPS | Flexible tool used to strike at range. |
+| --- | --- |
+| HOOKED_POLES | Polearm with hooked end (Like a shepherd's crook) |
+| SPEARS | Polearm with a long shaft and a sharp tip made of hard material. |
+| PIKES | Very long spear that can only be wielded in two hands, very unwieldy. |
+| GLAIVES | Polearm with a single-edged blade mounted on the end. |
+
+| Weapon Styles | Description |
+| --------------- | --------------------------------------------- |
+| MEDIEVAL_SWORDS | Swords associated with European culture. |
+| JAPANESE_SWORDS | Swords associated with Japanese culture. |
+| BIONIC_WEAPONRY | Weapons integrated into the body via bionics. |
### Gun
@@ -2099,7 +2181,9 @@ Guns can be defined like this:
"barrel_length": "30 mL", // Amount of volume lost when the barrel is sawn. Approximately 250 ml per inch is a decent approximation.
"valid_mod_locations": [ [ "accessories", 4 ], [ "grip", 1 ] ], // The valid locations for gunmods and the mount of slots for that location.
```
+
Alternately, every item (book, tool, armor, even food) can be used as gun if it has gun_data:
+
```json
"type": "TOOL", // Or any other item type
... // same entries as for the type (e.g. same entries as for any tool),
@@ -2112,38 +2196,38 @@ Alternately, every item (book, tool, armor, even food) can be used as gun if it
#### Ranged `Weapon_category`
-| Weapon Category | Description
-| --- | ---
-| BOWS | Elastic launching device for long-shafted projectiles.
-| S_XBOWS | Elastic launching device mounted on a frame to be triggered, pistol sized.
-| M_XBOWS | Elastic launching device mounted on a frame to be triggered.
-| SLINGSHOTS | Elastic, handheld launching device typically used for small round projectiles.
-| SLINGS | Projectile weapon using a cradle connected to two retention cords, used to fling blunt projectiles.
-| --- | ---
-| PISTOLS | Handgun with a chamber integral to the gun barrel. In-game, any handgun that isn't a revolver goes here.
-| REVOLVERS | Repeating handgun with a revolving cylinder containing multiple chambers.
-| SUBMACHINE_GUNS | Magazine fed automatic carbine designed to fire handgun cartridges.
-| RIFLES | Long barrelled firearms designed for more accurate shooting.
-| MACHINE_GUNS | Fully automatic autoloading firearm designed for sustained fire.
-| GATLING_GUNS | Rapid firing multi barrel firearm.
-| SHOTGUNS | Long barreled firearm generally designed to fire shotshells.
-| --- | ---
-| GRENADE_LAUNCHERS | Firearm designed to propel large caliber projectile typically loaded with warhead of some kind (smoke, gas, explosive, etc)
-| ROCKET_LAUNCHERS | Firearm that propels unguided, rocket-propelled projectile.
-| --- | ---
-| FLAMETHROWERS | Ranged incendiary device designed to propel a controllable jet of fire.
-| WATER_CANNONS | It fires water at your enemies.
-| SPRAY_GUNS | It spews chemicals at your enemies.
-
-| Action category | Description
-| --- | ---
-| 1SHOT | Ranged weapon with at least one barrel but no loading system/magazine.
-| AUTOLOADING | Ranged weapon with autoloading mechanisms like blowback, gas-operated, or recoil operated systems.
-| MANUAL_ACTION | Ranged weapon using manual actions like bolt/pump/lever.
-| ENERGY_WEAPONS | Weapon designed to utilize focused energy (sonic, electromagnetic waves, particle beams, etc). Both Ranged/Melee.
-| MAGNETIC | Weapon that propels payload via electromagnetism.
-| PNEUMATIC | Ranged weapon that propels payload via compressed air.
-| ELASTIC | Ranged weapon that propels payload via elastic band.
+| Weapon Category | Description |
+| ----------------- | --------------------------------------------------------------------------------------------------------------------------- |
+| BOWS | Elastic launching device for long-shafted projectiles. |
+| S_XBOWS | Elastic launching device mounted on a frame to be triggered, pistol sized. |
+| M_XBOWS | Elastic launching device mounted on a frame to be triggered. |
+| SLINGSHOTS | Elastic, handheld launching device typically used for small round projectiles. |
+| SLINGS | Projectile weapon using a cradle connected to two retention cords, used to fling blunt projectiles. |
+| --- | --- |
+| PISTOLS | Handgun with a chamber integral to the gun barrel. In-game, any handgun that isn't a revolver goes here. |
+| REVOLVERS | Repeating handgun with a revolving cylinder containing multiple chambers. |
+| SUBMACHINE_GUNS | Magazine fed automatic carbine designed to fire handgun cartridges. |
+| RIFLES | Long barrelled firearms designed for more accurate shooting. |
+| MACHINE_GUNS | Fully automatic autoloading firearm designed for sustained fire. |
+| GATLING_GUNS | Rapid firing multi barrel firearm. |
+| SHOTGUNS | Long barreled firearm generally designed to fire shotshells. |
+| --- | --- |
+| GRENADE_LAUNCHERS | Firearm designed to propel large caliber projectile typically loaded with warhead of some kind (smoke, gas, explosive, etc) |
+| ROCKET_LAUNCHERS | Firearm that propels unguided, rocket-propelled projectile. |
+| --- | --- |
+| FLAMETHROWERS | Ranged incendiary device designed to propel a controllable jet of fire. |
+| WATER_CANNONS | It fires water at your enemies. |
+| SPRAY_GUNS | It spews chemicals at your enemies. |
+
+| Action category | Description |
+| --------------- | ----------------------------------------------------------------------------------------------------------------- |
+| 1SHOT | Ranged weapon with at least one barrel but no loading system/magazine. |
+| AUTOLOADING | Ranged weapon with autoloading mechanisms like blowback, gas-operated, or recoil operated systems. |
+| MANUAL_ACTION | Ranged weapon using manual actions like bolt/pump/lever. |
+| ENERGY_WEAPONS | Weapon designed to utilize focused energy (sonic, electromagnetic waves, particle beams, etc). Both Ranged/Melee. |
+| MAGNETIC | Weapon that propels payload via electromagnetism. |
+| PNEUMATIC | Ranged weapon that propels payload via compressed air. |
+| ELASTIC | Ranged weapon that propels payload via elastic band. |
### Gunmod
@@ -2170,7 +2254,10 @@ Gun mods can be defined like this:
"reload_modifier": -10, // Optional field increasing or decreasing base gun reload time in percent
"min_str_required_mod": 14, // Optional field increasing or decreasing minimum strength required to use gun
```
-Alternately, every item (book, tool, armor, even food) can be used as a gunmod if it has gunmod_data:
+
+Alternately, every item (book, tool, armor, even food) can be used as a gunmod if it has
+gunmod_data:
+
```
"type": "TOOL", // Or any other item type
... // same entries as for the type (e.g. same entries as for any tool),
@@ -2182,6 +2269,7 @@ Alternately, every item (book, tool, armor, even food) can be used as a gunmod i
```
### Batteries
+
```C++
"type": "BATTERY", // Defines this as a BATTERY
... // Same entries as above for the generic item
@@ -2222,7 +2310,8 @@ Alternately, every item (book, tool, armor, even food) can be used as a gunmod i
### Seed Data
-Every item type can have optional seed data, if the item has seed data, it's considered a seed and can be planted:
+Every item type can have optional seed data, if the item has seed data, it's considered a seed and
+can be planted:
```C++
"seed_data" : {
@@ -2239,7 +2328,8 @@ Every item type can have optional seed data, if the item has seed data, it's con
### Brewing Data
-Every item type can have optional brewing data, if the item has brewing data, it can be placed in a vat and will ferment into a different item type.
+Every item type can have optional brewing data, if the item has brewing data, it can be placed in a
+vat and will ferment into a different item type.
Currently only vats can only accept and produce liquid items.
@@ -2252,8 +2342,7 @@ Currently only vats can only accept and produce liquid items.
### Relic Data
-Defines various (semi-)magical properties of items.
-See RELICS.md for
+Defines various (semi-)magical properties of items. See RELICS.md for
### Artifact Data
@@ -2305,6 +2394,7 @@ Possible values (see src/enums.h for an up-to-date list):
- `AEP_SICK` Decreases health over time
#### `effects_worn`
+
(optional, default: empty list)
Effects of the artifact when it's worn (it must be an armor item to be worn).
@@ -2323,7 +2413,8 @@ Possible values are the same as for effects_carried.
(optional, default: empty list)
-Effects of the artifact when it's activated (which require it to have a `"use_action": "ARTIFACT"` and it must have a non-zero max_charges value).
+Effects of the artifact when it's activated (which require it to have a `"use_action": "ARTIFACT"`
+and it must have a non-zero max_charges value).
Possible values (see src/artifact.h for an up-to-date list):
@@ -2373,9 +2464,12 @@ Every item type can have software data, it does not have any behavior:
### Fuel data
-Every item type can have fuel data that determines how much horse power it produces per unit consumed. Currently, gasses and plasmas cannot really be fuels.
+Every item type can have fuel data that determines how much horse power it produces per unit
+consumed. Currently, gasses and plasmas cannot really be fuels.
-If a fuel has the PERPETUAL flag, engines powered by it never use any fuel. This is primarily intended for the muscle pseudo-fuel, but mods may take advantage of it to make perpetual motion machines.
+If a fuel has the PERPETUAL flag, engines powered by it never use any fuel. This is primarily
+intended for the muscle pseudo-fuel, but mods may take advantage of it to make perpetual motion
+machines.
```C++
"fuel" : {
@@ -2397,7 +2491,9 @@ If a fuel has the PERPETUAL flag, engines powered by it never use any fuel. Thi
### Use Actions
-The contents of use_action fields can either be a string indicating a built-in function to call when the item is activated (defined in iuse.cpp), or one of several special definitions that invoke a more structured function.
+The contents of use_action fields can either be a string indicating a built-in function to call when
+the item is activated (defined in iuse.cpp), or one of several special definitions that invoke a
+more structured function.
```C++
"use_action": {
@@ -2650,11 +2746,16 @@ The contents of use_action fields can either be a string indicating a built-in f
### Random Descriptions
-Any item with a "snippet_category" entry will have random descriptions, based on that snippet category:
+Any item with a "snippet_category" entry will have random descriptions, based on that snippet
+category:
+
```
"snippet_category": "newspaper",
```
-The item descriptions are taken from snippets, which can be specified like this (the value of category must match the snippet_category in the item definition):
+
+The item descriptions are taken from snippets, which can be specified like this (the value of
+category must match the snippet_category in the item definition):
+
```C++
{
"type" : "snippet",
@@ -2663,7 +2764,9 @@ The item descriptions are taken from snippets, which can be specified like this
"text": "your flavor text"
}
```
+
or several snippets at once:
+
```C++
{
"type" : "snippet",
@@ -2677,14 +2780,18 @@ or several snippets at once:
"text": [ "your flavor text", "another flavor text", "more flavor" ]
}
```
-Multiple snippets for the same category are possible and actually recommended. The game will select a random one for each item of that type.
+
+Multiple snippets for the same category are possible and actually recommended. The game will select
+a random one for each item of that type.
One can also put the snippets directly in the item definition:
+
```
"snippet_category": [ "text 1", "text 2", "text 3" ],
```
-This will automatically create a snippet category specific to that item and populate that category with the given snippets.
-The format also support snippet ids like above.
+
+This will automatically create a snippet category specific to that item and populate that category
+with the given snippets. The format also support snippet ids like above.
# `json/` JSONs
@@ -2747,35 +2854,41 @@ Should always be `harvest` to mark the object as a harvest definition.
#### `message`
-Optional message to be printed when a creature using the harvest definition is butchered. May be omitted from definition.
+Optional message to be printed when a creature using the harvest definition is butchered. May be
+omitted from definition.
#### `entries`
-Array of dictionaries defining possible items produced on butchering and their likelihood of being produced.
-`drop` value should be the `id` string of the item to be produced.
+Array of dictionaries defining possible items produced on butchering and their likelihood of being
+produced. `drop` value should be the `id` string of the item to be produced.
-`type` value should be a string with the associated body part the item comes from.
- Acceptable values are as follows:
- `flesh`: the "meat" of the creature.
- `offal`: the "organs" of the creature. these are removed when field dressing.
- `skin`: the "skin" of the creature. this is what is ruined while quartering.
- `bone`: the "bones" of the creature. you will get some amount of these from field dressing, and the rest of them from butchering the carcass.
- `bionic`: an item gained by dissecting the creature. not restricted to CBMs.
- `bionic_group`: an item group that will give an item by dissecting a creature. not restricted to groups containing CBMs.
+`type` value should be a string with the associated body part the item comes from. Acceptable values
+are as follows: `flesh`: the "meat" of the creature. `offal`: the "organs" of the creature. these
+are removed when field dressing. `skin`: the "skin" of the creature. this is what is ruined while
+quartering. `bone`: the "bones" of the creature. you will get some amount of these from field
+dressing, and the rest of them from butchering the carcass. `bionic`: an item gained by dissecting
+the creature. not restricted to CBMs. `bionic_group`: an item group that will give an item by
+dissecting a creature. not restricted to groups containing CBMs.
-`flags` value should be an array of strings. It's the flags that will be added to te items of that entry upon harvesting.
+`flags` value should be an array of strings. It's the flags that will be added to te items of that
+entry upon harvesting.
-`faults` value should be an array of `fault_id` strings. It's the faults that will be added to te items of that entry upon harvesting.
+`faults` value should be an array of `fault_id` strings. It's the faults that will be added to te
+items of that entry upon harvesting.
For every `type` other then `bionic` and `bionic_group` following entries scale the results:
- `base_num` value should be an array with two elements in which the first defines the minimum number of the corresponding item produced and the second defines the maximum number.
- `scale_num` value should be an array with two elements, increasing the minimum and maximum drop numbers respectively by element value * survival skill.
- `max` upper limit after `bas_num` and `scale_num` are calculated using
- `mass_ratio` value is a multiplier of how much of the monster's weight comprises the associated item. to conserve mass, keep between 0 and 1 combined with all drops. This overrides `base_num`, `scale_num` and `max`
-
-
-For `type`s: `bionic` and `bionic_group` following enrties can scale the results:
- `max` this value (in contrary to `max` for other `type`s) corresponds to maximum butchery roll that will be passed to check_butcher_cbm() in activity_handlers.cpp; view check_butcher_cbm() to see corresponding distribution chances for roll values passed to that function
+`base_num` value should be an array with two elements in which the first defines the minimum number
+of the corresponding item produced and the second defines the maximum number. `scale_num` value
+should be an array with two elements, increasing the minimum and maximum drop numbers respectively
+by element value * survival skill. `max` upper limit after `bas_num` and `scale_num` are calculated
+using `mass_ratio` value is a multiplier of how much of the monster's weight comprises the
+associated item. to conserve mass, keep between 0 and 1 combined with all drops. This overrides
+`base_num`, `scale_num` and `max`
+
+For `type`s: `bionic` and `bionic_group` following enrties can scale the results: `max` this value
+(in contrary to `max` for other `type`s) corresponds to maximum butchery roll that will be passed to
+check_butcher_cbm() in activity_handlers.cpp; view check_butcher_cbm() to see corresponding
+distribution chances for roll values passed to that function
### Weapon Category
@@ -2789,7 +2902,8 @@ Used to classify weapons (guns or melee) into groups, mainly for use in martial
}
```
-`"name"` is a translatable string used for UI display in martial arts UI, while ID is used for JSON entries.
+`"name"` is a translatable string used for UI display in martial arts UI, while ID is used for JSON
+entries.
### Furniture
@@ -2827,12 +2941,14 @@ Same as for terrain, see below in the chapter "Common to furniture and terrain".
#### `move_cost_mod`
-Movement cost modifier (`-10` = impassable, `0` = no change). This is added to the movecost of the underlying terrain.
+Movement cost modifier (`-10` = impassable, `0` = no change). This is added to the movecost of the
+underlying terrain.
#### `light_emitted`
-How much light the furniture produces. 10 will light the tile it's on brightly, 15 will light that tile and the tiles around it brightly, as well as slightly lighting the tiles two tiles away from the source.
-For examples: An overhead light is 120, a utility light, 240, and a console, 10.
+How much light the furniture produces. 10 will light the tile it's on brightly, 15 will light that
+tile and the tiles around it brightly, as well as slightly lighting the tiles two tiles away from
+the source. For examples: An overhead light is 120, a utility light, 240, and a console, 10.
#### `required_str`
@@ -2840,21 +2956,25 @@ Strength required to move the furniture around. Negative values indicate an unmo
#### `crafting_pseudo_item`
-(Optional) Id of an item (tool) that will be available for crafting when this furniture is range (the furniture acts as an item of that type). Can be made into an array.
-Example: `"crafting_pseudo_item": [ "fake_gridwelder", "fake_gridsolderingiron" ],`
+(Optional) Id of an item (tool) that will be available for crafting when this furniture is range
+(the furniture acts as an item of that type). Can be made into an array. Example:
+`"crafting_pseudo_item": [ "fake_gridwelder", "fake_gridsolderingiron" ],`
#### `workbench`
-(Optional) Can craft here. Must specify a speed multiplier, allowed mass, and allowed volume. Mass/volume over these limits incur a speed penalty. Must be paired with a `"workbench"` `examine_action` to function.
+(Optional) Can craft here. Must specify a speed multiplier, allowed mass, and allowed volume.
+Mass/volume over these limits incur a speed penalty. Must be paired with a `"workbench"`
+`examine_action` to function.
#### `plant_data`
-(Optional) This is a plant. Must specify a plant transform, and a base depending on context. You can also add a harvest or growth multiplier if it has the `GROWTH_HARVEST` flag.
+(Optional) This is a plant. Must specify a plant transform, and a base depending on context. You can
+also add a harvest or growth multiplier if it has the `GROWTH_HARVEST` flag.
#### `surgery_skill_multiplier`
-(Optional) Surgery skill multiplier (float) applied by this furniture to survivor standing next to it for the purpose of surgery.
-
+(Optional) Surgery skill multiplier (float) applied by this furniture to survivor standing next to
+it for the purpose of surgery.
### Terrain
@@ -2894,35 +3014,46 @@ Same as for furniture, see below in the chapter "Common to furniture and terrain
#### `move_cost`
-Move cost to move through. A value of 0 means it's impassable (e.g. wall). You should not use negative values. The positive value is multiple of 50 move points, e.g. value 2 means the player uses 2\*50 = 100 move points when moving across the terrain.
+Move cost to move through. A value of 0 means it's impassable (e.g. wall). You should not use
+negative values. The positive value is multiple of 50 move points, e.g. value 2 means the player
+uses 2\*50 = 100 move points when moving across the terrain.
#### `light_emitted`
-How much light the terrain emits. 10 will light the tile it's on brightly, 15 will light that tile and the tiles around it brightly, as well as slightly lighting the tiles two tiles away from the source.
-For examples: An overhead light is 120, a utility light, 240, and a console, 10.
+How much light the terrain emits. 10 will light the tile it's on brightly, 15 will light that tile
+and the tiles around it brightly, as well as slightly lighting the tiles two tiles away from the
+source. For examples: An overhead light is 120, a utility light, 240, and a console, 10.
#### `trap`
(Optional) Id of the build-in trap of that terrain.
-For example the terrain `t_pit` has the built-in trap `tr_pit`. Every tile in the game that has the terrain `t_pit` also has, therefore, an implicit trap `tr_pit` on it. The two are inseparable (the player can not deactivate the built-in trap, and changing the terrain will also deactivate the built-in trap).
+For example the terrain `t_pit` has the built-in trap `tr_pit`. Every tile in the game that has the
+terrain `t_pit` also has, therefore, an implicit trap `tr_pit` on it. The two are inseparable (the
+player can not deactivate the built-in trap, and changing the terrain will also deactivate the
+built-in trap).
A built-in trap prevents adding any other trap explicitly (by the player and through mapgen).
#### `harvestable`
-(Optional) If defined, the terrain is harvestable. This entry defines the item type of the harvested fruits (or similar). To make this work, you also have to set one of the `harvest_*` `examine_action` functions.
+(Optional) If defined, the terrain is harvestable. This entry defines the item type of the harvested
+fruits (or similar). To make this work, you also have to set one of the `harvest_*` `examine_action`
+functions.
#### `transforms_into`
-(Optional) Used for various transformation of the terrain. If defined, it must be a valid terrain id. Used for example:
+(Optional) Used for various transformation of the terrain. If defined, it must be a valid terrain
+id. Used for example:
- When harvesting fruits (to change into the harvested form of the terrain).
-- In combination with the `HARVESTED` flag and `harvest_season` to change the harvested terrain back into a terrain with fruits.
+- In combination with the `HARVESTED` flag and `harvest_season` to change the harvested terrain back
+ into a terrain with fruits.
#### `harvest_season`
-(Optional) On of "SUMMER", "AUTUMN", "WINTER", "SPRING", used in combination with the "HARVESTED" flag to revert the terrain back into a terrain that can be harvested.
+(Optional) On of "SUMMER", "AUTUMN", "WINTER", "SPRING", used in combination with the "HARVESTED"
+flag to revert the terrain back into a terrain that can be harvested.
#### `roof`
@@ -2934,7 +3065,10 @@ Some values can/must be set for terrain and furniture. They have the same meanin
#### `id`
-Id of the object, this should be unique among all object of that type (all terrain or all furniture types). By convention (but technically not needed), the id should have the "f_" prefix for furniture and the "t_" prefix for terrain. This is not translated. It must not be changed later as that would break save compatibility.
+Id of the object, this should be unique among all object of that type (all terrain or all furniture
+types). By convention (but technically not needed), the id should have the "f_" prefix for furniture
+and the "t_" prefix for terrain. This is not translated. It must not be changed later as that would
+break save compatibility.
#### `name`
@@ -2946,16 +3080,21 @@ Displayed name of the object. This will be translated.
#### `connects_to`
-(Optional) The group of terrains to which this terrain connects. This affects tile rotation and connections, and the ASCII symbol drawn by terrain with the flag "AUTO_WALL_SYMBOL".
+(Optional) The group of terrains to which this terrain connects. This affects tile rotation and
+connections, and the ASCII symbol drawn by terrain with the flag "AUTO_WALL_SYMBOL".
Current values:
+
- `CHAINFENCE`
- `RAILING`
- `WALL`
- `WATER`
- `WOODFENCE`
-Example: `-` , `|` , `X` and `Y` are terrain which share the same `connects_to` value. `O` does not have it. `X` and `Y` also have the `AUTO_WALL_SYMBOL` flag. `X` will be drawn as a T-intersection (connected to west, south and east), `Y` will be drawn as a horizontal line (going from west to east, no connection to south).
+Example: `-` , `|` , `X` and `Y` are terrain which share the same `connects_to` value. `O` does not
+have it. `X` and `Y` also have the `AUTO_WALL_SYMBOL` flag. `X` will be drawn as a T-intersection
+(connected to west, south and east), `Y` will be drawn as a horizontal line (going from west to
+east, no connection to south).
```
-X- -Y-
@@ -2964,16 +3103,15 @@ Example: `-` , `|` , `X` and `Y` are terrain which share the same `connects_to`
#### `symbol`
-ASCII symbol of the object as it appears in the game. The symbol string must be exactly one character long. This can also be an array of 4 strings, which define the symbol during the different seasons. The first entry defines the symbol during spring. If it's not an array, the same symbol is used all year round.
+ASCII symbol of the object as it appears in the game. The symbol string must be exactly one
+character long. This can also be an array of 4 strings, which define the symbol during the different
+seasons. The first entry defines the symbol during spring. If it's not an array, the same symbol is
+used all year round.
#### `comfort`
-How comfortable this terrain/furniture is. Impact ability to fall asleep on it.
- uncomfortable = -999,
- neutral = 0,
- slightly_comfortable = 3,
- comfortable = 5,
- very_comfortable = 10
+How comfortable this terrain/furniture is. Impact ability to fall asleep on it. uncomfortable =
+-999, neutral = 0, slightly_comfortable = 3, comfortable = 5, very_comfortable = 10
#### `floor_bedding_warmth`
@@ -2981,21 +3119,28 @@ Bonus warmth offered by this terrain/furniture when used to sleep.
#### `bonus_fire_warmth_feet`
-Increase warmth received on feet from nearby fire (default = 300)
+Increase warmth received on feet from nearby fire (default = 300)
#### `looks_like`
-id of a similar item that this item looks like. The tileset loader will try to load the tile for that item if this item doesn't have a tile. looks_like entries are implicitly chained, so if 'throne' has looks_like 'big_chair' and 'big_chair' has looks_like 'chair', a throne will be displayed using the chair tile if tiles for throne and big_chair do not exist. If a tileset can't find a tile for any item in the looks_like chain, it will default to the ascii symbol.
+id of a similar item that this item looks like. The tileset loader will try to load the tile for
+that item if this item doesn't have a tile. looks_like entries are implicitly chained, so if
+'throne' has looks_like 'big_chair' and 'big_chair' has looks_like 'chair', a throne will be
+displayed using the chair tile if tiles for throne and big_chair do not exist. If a tileset can't
+find a tile for any item in the looks_like chain, it will default to the ascii symbol.
#### `color` or `bgcolor`
-Color of the object as it appears in the game. "color" defines the foreground color (no background color), "bgcolor" defines a solid background color. As with the "symbol" value, this can be an array with 4 entries, each entry being the color during the different seasons.
+Color of the object as it appears in the game. "color" defines the foreground color (no background
+color), "bgcolor" defines a solid background color. As with the "symbol" value, this can be an array
+with 4 entries, each entry being the color during the different seasons.
> **NOTE**: You must use ONLY ONE of "color" or "bgcolor"
#### `max_volume`
-(Optional) Maximal volume that can be used to store items here. Volume in ml and L can be used - "50 ml" or "2 L"
+(Optional) Maximal volume that can be used to store items here. Volume in ml and L can be used - "50
+ml" or "2 L"
#### `examine_action`
@@ -3003,7 +3148,11 @@ Color of the object as it appears in the game. "color" defines the foreground co
#### `close" And "open`
-(Optional) The value should be a terrain id (inside a terrain entry) or a furniture id (in a furniture entry). If either is defined, the player can open / close the object. Opening / closing will change the object at the affected tile to the given one. For example one could have object "safe_c", which "open"s to "safe_o" and "safe_o" in turn "close"s to "safe_c". Here "safe_c" and "safe_o" are two different terrain (or furniture) types that have different properties.
+(Optional) The value should be a terrain id (inside a terrain entry) or a furniture id (in a
+furniture entry). If either is defined, the player can open / close the object. Opening / closing
+will change the object at the affected tile to the given one. For example one could have object
+"safe_c", which "open"s to "safe_o" and "safe_o" in turn "close"s to "safe_c". Here "safe_c" and
+"safe_o" are two different terrain (or furniture) types that have different properties.
#### `bash`
@@ -3011,15 +3160,18 @@ Color of the object as it appears in the game. "color" defines the foreground co
#### `deconstruct`
-(Optional) Defines whether the object can be deconstructed and if so, what the results shall be. See "map_deconstruct_info".
+(Optional) Defines whether the object can be deconstructed and if so, what the results shall be. See
+"map_deconstruct_info".
#### `pry`
-(Optional) Defines whether the object can be pried open and if so, what happens. See "prying_result".
+(Optional) Defines whether the object can be pried open and if so, what happens. See
+"prying_result".
#### `map_bash_info`
-Defines the various things that happen when the player or something else bashes terrain or furniture.
+Defines the various things that happen when the player or something else bashes terrain or
+furniture.
```C++
{
@@ -3049,27 +3201,38 @@ Defines the various things that happen when the player or something else bashes
TODO
#### `sound`, `sound_fail`, `sound_vol`, `sound_fail_vol`
-(Optional) Sound and volume of the sound that appears upon destroying the bashed object or upon unsuccessfully bashing it (failing). The sound strings are translated (and displayed to the player).
+
+(Optional) Sound and volume of the sound that appears upon destroying the bashed object or upon
+unsuccessfully bashing it (failing). The sound strings are translated (and displayed to the player).
#### `furn_set`, `ter_set`
-The terrain / furniture that will be set when the original is destroyed. This is mandatory for bash entries in terrain, but optional for entries in furniture (it defaults to no furniture).
+The terrain / furniture that will be set when the original is destroyed. This is mandatory for bash
+entries in terrain, but optional for entries in furniture (it defaults to no furniture).
#### `explosive`
-(Optional) If greater than 0, destroying the object causes an explosion with this strength (see `game::explosion`).
+
+(Optional) If greater than 0, destroying the object causes an explosion with this strength (see
+`game::explosion`).
#### `destroy_only`
+
TODO
#### `bash_below`
+
TODO
#### `tent_centers`, `collapse_radius`
-(Optional) For furniture that is part of tents, this defines the id of the center part, which will be destroyed as well when other parts of the tent get bashed. The center is searched for in the given "collapse_radius" radius, it should match the size of the tent.
+
+(Optional) For furniture that is part of tents, this defines the id of the center part, which will
+be destroyed as well when other parts of the tent get bashed. The center is searched for in the
+given "collapse_radius" radius, it should match the size of the tent.
#### `items`
-(Optional) An item group (inline) or an id of an item group, see doc/ITEM_SPAWN.md. The default subtype is "collection". Upon successful bashing, items from that group will be spawned.
+(Optional) An item group (inline) or an id of an item group, see doc/ITEM_SPAWN.md. The default
+subtype is "collection". Upon successful bashing, items from that group will be spawned.
#### `map_deconstruct_info`
@@ -3083,77 +3246,97 @@ TODO
#### `furn_set`, `ter_set`
-The terrain / furniture that will be set after the original has been deconstructed. "furn_set" is optional (it defaults to no furniture), "ter_set" is only used upon "deconstruct" entries in terrain and is mandatory there.
+The terrain / furniture that will be set after the original has been deconstructed. "furn_set" is
+optional (it defaults to no furniture), "ter_set" is only used upon "deconstruct" entries in terrain
+and is mandatory there.
#### `items`
-(Optional) An item group (inline) or an id of an item group, see doc/ITEM_SPAWN.md. The default subtype is "collection". Upon deconstruction the object, items from that group will be spawned.
+(Optional) An item group (inline) or an id of an item group, see doc/ITEM_SPAWN.md. The default
+subtype is "collection". Upon deconstruction the object, items from that group will be spawned.
#### `prying_result`
```JSON
{
- "success_message": "You pry open the door.",
- "fail_message": "You pry, but cannot pry open the door.",
- "break_message": "You damage the door!",
- "pry_quality": 2,
- "pry_bonus_mult": 3,
- "noise": 12,
- "break_noise": 10,
- "sound": "crunch!",
- "break_sound": "crack!",
- "breakable": true,
- "difficulty": 8,
- "new_ter_type": "t_door_o",
- "new_furn_type": "f_crate_o",
- "break_ter_type": "t_door_b",
- "break_furn_type": "f_null",
- "break_items": [
- { "item": "2x4", "prob": 25 },
- { "item": "wood_panel", "prob": 10 },
- { "item": "splinter", "count": [ 1, 2 ] },
- { "item": "nail", "charges": [ 0, 2 ] }
- ]
+ "success_message": "You pry open the door.",
+ "fail_message": "You pry, but cannot pry open the door.",
+ "break_message": "You damage the door!",
+ "pry_quality": 2,
+ "pry_bonus_mult": 3,
+ "noise": 12,
+ "break_noise": 10,
+ "sound": "crunch!",
+ "break_sound": "crack!",
+ "breakable": true,
+ "difficulty": 8,
+ "new_ter_type": "t_door_o",
+ "new_furn_type": "f_crate_o",
+ "break_ter_type": "t_door_b",
+ "break_furn_type": "f_null",
+ "break_items": [
+ { "item": "2x4", "prob": 25 },
+ { "item": "wood_panel", "prob": 10 },
+ { "item": "splinter", "count": [1, 2] },
+ { "item": "nail", "charges": [0, 2] }
+ ]
}
```
#### `new_ter_type`, `new_furn_type`
-The terrain / furniture that will be set after the original has been pried open. "furn_set" is optional (it defaults to no furniture), "ter_set" is only used upon "pry" entries in terrain and is mandatory there.
+The terrain / furniture that will be set after the original has been pried open. "furn_set" is
+optional (it defaults to no furniture), "ter_set" is only used upon "pry" entries in terrain and is
+mandatory there.
#### `success_message`, `fail_message`, `break_message`
-Messages displayed on successfully prying open the terrain / furniture, on failure, or should the terrain / furniture break into something else from a failed pry attempt. `break_message` is only required if `breakable` is set to true and `break_ter_type` is defined.
+Messages displayed on successfully prying open the terrain / furniture, on failure, or should the
+terrain / furniture break into something else from a failed pry attempt. `break_message` is only
+required if `breakable` is set to true and `break_ter_type` is defined.
#### `pry_quality`, `pry_bonus_mult`, `difficulty`
-This determines the minimum prying quality needed to attempt to pry open the terrain / furniture, and the chance of successfully prying it open. From iuse.cpp:
+This determines the minimum prying quality needed to attempt to pry open the terrain / furniture,
+and the chance of successfully prying it open. From iuse.cpp:
```C++
- int diff = pry->difficulty;
- diff -= ( ( pry_level - pry->pry_quality ) * pry->pry_bonus_mult );
+int diff = pry->difficulty;
+diff -= ( ( pry_level - pry->pry_quality ) * pry->pry_bonus_mult );
```
```C++
- if( dice( 4, diff ) < dice( 4, p->str_cur ) ) {
- p->add_msg_if_player( m_good, pry->success_message );
+if( dice( 4, diff ) < dice( 4, p->str_cur ) ) {
+ p->add_msg_if_player( m_good, pry->success_message );
```
-`difficulty` is compared to the character's current strength during the prying attempt. If the available prying quality of your tool is above the required `pry_quality`, effective difficulty is reduced at a rate determined by `pry_bonus_mult`. `pry_bonus_mult` is optional, and defaults to 1 (meaning for every level of prying quality your tool has beyond the minimum, `diff` will be reduced by 1).
+`difficulty` is compared to the character's current strength during the prying attempt. If the
+available prying quality of your tool is above the required `pry_quality`, effective difficulty is
+reduced at a rate determined by `pry_bonus_mult`. `pry_bonus_mult` is optional, and defaults to 1
+(meaning for every level of prying quality your tool has beyond the minimum, `diff` will be reduced
+by 1).
#### `noise`, `break_noise`, 'sound', 'break_sound'
-If `noise` is specified, successfully prying the terrain / furniture open will play `sound` at the specified volume. If `breakable` is true and `break_noise` is specified, breaking the terrain / furniture on a failed prying attempt will play `break_noise` at the specified volume.
+If `noise` is specified, successfully prying the terrain / furniture open will play `sound` at the
+specified volume. If `breakable` is true and `break_noise` is specified, breaking the terrain /
+furniture on a failed prying attempt will play `break_noise` at the specified volume.
-If `noise` or `break_noise` are not specified then prying or breaking the terrain / furniture in question will simply be silent and make no sound, making them optional. `sound` and `break_sound` are also optional, with default messages of "crunch!" and "crack!" respectively.
+If `noise` or `break_noise` are not specified then prying or breaking the terrain / furniture in
+question will simply be silent and make no sound, making them optional. `sound` and `break_sound`
+are also optional, with default messages of "crunch!" and "crack!" respectively.
#### `breakable`, `break_ter_type`, `break_furn_type`
-If `breakable` is set to true, then failed pry attempts have a chance of breaking the terrain / furniture instead. For terrain, `break_ter_type` is mandatory if `breakable` is set to true, `break_furn_type` is optional and defaults to null.
+If `breakable` is set to true, then failed pry attempts have a chance of breaking the terrain /
+furniture instead. For terrain, `break_ter_type` is mandatory if `breakable` is set to true,
+`break_furn_type` is optional and defaults to null.
#### `break_items`
-(Optional) An item group (inline) or an id of an item group, see doc/ITEM_SPAWN.md. The default subtype is "collection". If `breakable` is set to true, breaking the object from a failed pry attempt will spawn items from that group.
+(Optional) An item group (inline) or an id of an item group, see doc/ITEM_SPAWN.md. The default
+subtype is "collection". If `breakable` is set to true, breaking the object from a failed pry
+attempt will spawn items from that group.
### `plant_data`
@@ -3168,19 +3351,23 @@ If `breakable` is set to true, then failed pry attempts have a chance of breakin
#### `transform`
-What the `PLANT` furniture turn into when it grows a stage, or what a `PLANTABLE` furniture turns into when it is planted on.
+What the `PLANT` furniture turn into when it grows a stage, or what a `PLANTABLE` furniture turns
+into when it is planted on.
#### `base`
-What the 'base' furniture of the `PLANT` furniture is - what it would be if there was not a plant growing there. Used when monsters 'eat' the plant to preserve what furniture it is.
+What the 'base' furniture of the `PLANT` furniture is - what it would be if there was not a plant
+growing there. Used when monsters 'eat' the plant to preserve what furniture it is.
#### `growth_multiplier`
-A flat multiplier on the growth speed on the plant. For numbers greater than one, it will take longer to grow, and for numbers less than one it will take less time to grow.
+A flat multiplier on the growth speed on the plant. For numbers greater than one, it will take
+longer to grow, and for numbers less than one it will take less time to grow.
#### `harvest_multiplier`
-A flat multiplier on the harvest count of the plant. For numbers greater than one, the plant will give more produce from harvest, for numbers less than one it will give less produce from harvest.
+A flat multiplier on the harvest count of the plant. For numbers greater than one, the plant will
+give more produce from harvest, for numbers less than one it will give less produce from harvest.
### clothing_mod
@@ -3221,16 +3408,19 @@ The id member should be the unique id of the scenario.
The following properties (mandatory, except if noted otherwise) are supported:
-
## `description`
+
(string)
The in-game description.
## `name`
+
(string or object with members "male" and "female")
-The in-game name, either one gender-neutral string, or an object with gender specific names. Example:
+The in-game name, either one gender-neutral string, or an object with gender specific names.
+Example:
+
```C++
"name": {
"male": "Runaway groom",
@@ -3239,16 +3429,21 @@ The in-game name, either one gender-neutral string, or an object with gender spe
```
## `points`
+
(integer)
Point cost of scenario. Positive values cost points and negative values grant points.
## `items`
+
(optional, object with optional members "both", "male" and "female")
-Items the player starts with when selecting this scenario. One can specify different items based on the gender of the character. Each lists of items should be an array of items ids. Ids may appear multiple times, in which case the item is created multiple times.
+Items the player starts with when selecting this scenario. One can specify different items based on
+the gender of the character. Each lists of items should be an array of items ids. Ids may appear
+multiple times, in which case the item is created multiple times.
Example:
+
```C++
"items": {
"both": [
@@ -3260,11 +3455,14 @@ Example:
"female": [ "panties" ]
}
```
+
This gives the player pants, two rocks and (depending on the gender) briefs or panties.
-Mods can modify the lists of an existing scenario via "add:both" / "add:male" / "add:female" and "remove:both" / "remove:male" / "remove:female".
+Mods can modify the lists of an existing scenario via "add:both" / "add:male" / "add:female" and
+"remove:both" / "remove:male" / "remove:female".
Example for mods:
+
```C++
{
"type": "scenario",
@@ -3278,6 +3476,7 @@ Example for mods:
```
## `flags`
+
(optional, array of strings)
A list of flags. TODO: document those flags here.
@@ -3285,6 +3484,7 @@ A list of flags. TODO: document those flags here.
Mods can modify this via "add:flags" and "remove:flags".
## `cbms`
+
(optional, array of strings)
A list of CBM ids that are implanted in the character.
@@ -3292,40 +3492,57 @@ A list of CBM ids that are implanted in the character.
Mods can modify this via "add:CBMs" and "remove:CBMs".
## `traits", "forced_traits", "forbidden_traits`
+
(optional, array of strings)
-Lists of trait/mutation ids. Traits in "forbidden_traits" are forbidden and can't be selected during the character creation. Traits in "forced_traits" are automatically added to character. Traits in "traits" enables them to be chosen, even if they are not starting traits.
+Lists of trait/mutation ids. Traits in "forbidden_traits" are forbidden and can't be selected during
+the character creation. Traits in "forced_traits" are automatically added to character. Traits in
+"traits" enables them to be chosen, even if they are not starting traits.
-Mods can modify this via "add:traits" / "add:forced_traits" / "add:forbidden_traits" and "remove:traits" / "remove:forced_traits" / "remove:forbidden_traits".
+Mods can modify this via "add:traits" / "add:forced_traits" / "add:forbidden_traits" and
+"remove:traits" / "remove:forced_traits" / "remove:forbidden_traits".
## `allowed_locs`
+
(optional, array of strings)
-A list of starting location ids (see start_locations.json) that can be chosen when using this scenario.
+A list of starting location ids (see start_locations.json) that can be chosen when using this
+scenario.
## `start_name`
+
(string)
-The name that is shown for the starting location. This is useful if the scenario allows several starting locations, but the game can not list them all at once in the scenario description. Example: if the scenario allows to start somewhere in the wilderness, the starting locations would contain forest and fields, but its "start_name" may simply be "wilderness".
+The name that is shown for the starting location. This is useful if the scenario allows several
+starting locations, but the game can not list them all at once in the scenario description. Example:
+if the scenario allows to start somewhere in the wilderness, the starting locations would contain
+forest and fields, but its "start_name" may simply be "wilderness".
## `professions`
+
(optional, array of strings)
-A list of allowed professions that can be chosen when using this scenario. The first entry is the default profession. If this is empty, all professions are allowed.
+A list of allowed professions that can be chosen when using this scenario. The first entry is the
+default profession. If this is empty, all professions are allowed.
## `map_special`
+
(optional, string)
Add a map special to the starting location, see JSON_FLAGS for the possible specials.
## `missions`
+
(optional, array of strings)
-A list of mission ids that will be started and assigned to the player at the start of the game. Only missions with the ORIGIN_GAME_START origin are allowed. The last mission in the list will be the active mission, if multiple missions are assigned.
+A list of mission ids that will be started and assigned to the player at the start of the game. Only
+missions with the ORIGIN_GAME_START origin are allowed. The last mission in the list will be the
+active mission, if multiple missions are assigned.
# Starting locations
Starting locations are specified as JSON object with "type" member set to "start_location":
+
```C++
{
"type": "start_location",
@@ -3341,138 +3558,148 @@ The id member should be the unique id of the location.
The following properties (mandatory, except if noted otherwise) are supported:
## `name`
+
(string)
The in-game name of the location.
## `target`
+
(string)
-The id of an overmap terrain type (see overmap_terrain.json) of the starting location. The game will chose a random place with that terrain.
+The id of an overmap terrain type (see overmap_terrain.json) of the starting location. The game will
+chose a random place with that terrain.
## `flags`
+
(optional, array of strings)
Arbitrary flags. Mods can modify this via "add:flags" / "remove:flags". TODO: document them.
### `tile_config`
-Each tileset has a tile_config.json describing how to map the contents of a sprite sheet to various tile identifiers, different orientations, etc. The ordering of the overlays used for displaying mutations can be controlled as well. The ordering can be used to override the default ordering provided in `mutation_ordering.json`. Example:
+
+Each tileset has a tile_config.json describing how to map the contents of a sprite sheet to various
+tile identifiers, different orientations, etc. The ordering of the overlays used for displaying
+mutations can be controlled as well. The ordering can be used to override the default ordering
+provided in `mutation_ordering.json`. Example:
```C++
- { // whole file is a single object
- "tile_info": [ // tile_info is mandatory
- {
- "height": 32,
- "width": 32,
- "iso" : true, // Optional. Indicates an isometric tileset. Defaults to false.
- "pixelscale" : 2 // Optional. Sets a multiplier for resizing a tileset. Defaults to 1.
- }
- ],
- "tiles-new": [ // tiles-new is an array of sprite sheets
- { // alternately, just one "tiles" array
- "file": "tiles.png", // file containing sprites in a grid
- "tiles": [ // array with one entry per tile
- {
- "id": "10mm", // id is how the game maps things to sprites
- "fg": 1, // lack of prefix mostly indicates items
- "bg": 632, // fg and bg can be sprite indexes in the image
- "rotates": false
- },
- {
- "id": "t_wall", // "t_" indicates terrain
- "fg": [2918, 2919, 2918, 2919], // 2 or 4 sprite numbers indicates pre-rotated
- "bg": 633,
- "rotates": true,
- "multitile": true,
- "additional_tiles": [ // connected/combined versions of sprite
- { // or variations, see below
- "id": "center",
- "fg": [2919, 2918, 2919, 2918]
- },
- {
- "id": "corner",
- "fg": [2924, 2922, 2922, 2923]
- },
- {
- "id": "end_piece",
- "fg": [2918, 2919, 2918, 2919]
- },
- {
- "id": "t_connection",
- "fg": [2919, 2918, 2919, 2918]
- },
- {
- "id": "unconnected",
- "fg": 2235
- }
- ]
- },
- {
- "id": "vp_atomic_lamp", // "vp_" vehicle part
- "fg": 3019,
- "bg": 632,
- "rotates": false,
- "multitile": true,
- "additional_tiles": [
- {
- "id": "broken", // variant sprite
- "fg": 3021
- }
- ]
- },
- {
- "id": "t_dirt",
- "rotates": false,
- "fg": [
- { "weight":50, "sprite":640}, // weighted random variants
- { "weight":1, "sprite":3620},
- { "weight":1, "sprite":3621},
- { "weight":1, "sprite":3622}
- ]
- },
- {
- "id": [
- "overlay_mutation_GOURMAND", // character overlay for mutation
- "overlay_mutation_male_GOURMAND", // overlay for specified gender
- "overlay_mutation_active_GOURMAND" // overlay for activated mutation
- ],
- "fg": 4040
- }
- ]
- },
- { // second entry in tiles-new
- "file": "moretiles.png", // another sprite sheet
- "tiles": [
- {
- "id": ["xxx","yyy"], // define two ids at once
- "fg": 1,
- "bg": 234
- }
- ]
- }
- ],
- "overlay_ordering": [
- {
- "id" : "WINGS_BAT", // mutation name, in a string or array of strings
- "order" : 1000 // range from 0 - 9999, 9999 being the topmost layer
- },
- {
- "id" : [ "PLANTSKIN", "BARK" ], // mutation name, in a string or array of strings
- "order" : 3500 // order is applied to all items in the array
- },
- {
- "id" : "bio_armor_torso", // Overlay order of bionics is controlled in the same way
- "order" : 500
- }
- ]
- }
+{ // whole file is a single object
+ "tile_info": [ // tile_info is mandatory
+ {
+ "height": 32,
+ "width": 32,
+ "iso" : true, // Optional. Indicates an isometric tileset. Defaults to false.
+ "pixelscale" : 2 // Optional. Sets a multiplier for resizing a tileset. Defaults to 1.
+ }
+ ],
+ "tiles-new": [ // tiles-new is an array of sprite sheets
+ { // alternately, just one "tiles" array
+ "file": "tiles.png", // file containing sprites in a grid
+ "tiles": [ // array with one entry per tile
+ {
+ "id": "10mm", // id is how the game maps things to sprites
+ "fg": 1, // lack of prefix mostly indicates items
+ "bg": 632, // fg and bg can be sprite indexes in the image
+ "rotates": false
+ },
+ {
+ "id": "t_wall", // "t_" indicates terrain
+ "fg": [2918, 2919, 2918, 2919], // 2 or 4 sprite numbers indicates pre-rotated
+ "bg": 633,
+ "rotates": true,
+ "multitile": true,
+ "additional_tiles": [ // connected/combined versions of sprite
+ { // or variations, see below
+ "id": "center",
+ "fg": [2919, 2918, 2919, 2918]
+ },
+ {
+ "id": "corner",
+ "fg": [2924, 2922, 2922, 2923]
+ },
+ {
+ "id": "end_piece",
+ "fg": [2918, 2919, 2918, 2919]
+ },
+ {
+ "id": "t_connection",
+ "fg": [2919, 2918, 2919, 2918]
+ },
+ {
+ "id": "unconnected",
+ "fg": 2235
+ }
+ ]
+ },
+ {
+ "id": "vp_atomic_lamp", // "vp_" vehicle part
+ "fg": 3019,
+ "bg": 632,
+ "rotates": false,
+ "multitile": true,
+ "additional_tiles": [
+ {
+ "id": "broken", // variant sprite
+ "fg": 3021
+ }
+ ]
+ },
+ {
+ "id": "t_dirt",
+ "rotates": false,
+ "fg": [
+ { "weight":50, "sprite":640}, // weighted random variants
+ { "weight":1, "sprite":3620},
+ { "weight":1, "sprite":3621},
+ { "weight":1, "sprite":3622}
+ ]
+ },
+ {
+ "id": [
+ "overlay_mutation_GOURMAND", // character overlay for mutation
+ "overlay_mutation_male_GOURMAND", // overlay for specified gender
+ "overlay_mutation_active_GOURMAND" // overlay for activated mutation
+ ],
+ "fg": 4040
+ }
+ ]
+ },
+ { // second entry in tiles-new
+ "file": "moretiles.png", // another sprite sheet
+ "tiles": [
+ {
+ "id": ["xxx","yyy"], // define two ids at once
+ "fg": 1,
+ "bg": 234
+ }
+ ]
+ }
+ ],
+ "overlay_ordering": [
+ {
+ "id" : "WINGS_BAT", // mutation name, in a string or array of strings
+ "order" : 1000 // range from 0 - 9999, 9999 being the topmost layer
+ },
+ {
+ "id" : [ "PLANTSKIN", "BARK" ], // mutation name, in a string or array of strings
+ "order" : 3500 // order is applied to all items in the array
+ },
+ {
+ "id" : "bio_armor_torso", // Overlay order of bionics is controlled in the same way
+ "order" : 500
+ }
+ ]
+}
```
# Mutation overlay ordering
-The file `mutation_ordering.json` defines the order that visual mutation and bionic overlays are rendered on a character ingame. The layering value from 0 (bottom) - 9999 (top) sets the order.
+The file `mutation_ordering.json` defines the order that visual mutation and bionic overlays are
+rendered on a character ingame. The layering value from 0 (bottom) - 9999 (top) sets the order.
Example:
+
```C++
[
{
@@ -3501,26 +3728,32 @@ Example:
```
## `id`
+
(string)
-The internal ID of the mutation. Can be provided as a single string, or an array of strings. The order value provided will be applied to all items in the array.
+The internal ID of the mutation. Can be provided as a single string, or an array of strings. The
+order value provided will be applied to all items in the array.
## `order`
+
(integer)
-The ordering value of the mutation overlay. Values range from 0 - 9999, 9999 being the topmost drawn layer. Mutations that are not in any list will default to 9999.
+The ordering value of the mutation overlay. Values range from 0 - 9999, 9999 being the topmost drawn
+layer. Mutations that are not in any list will default to 9999.
# MOD_INFO
Also see [MODDING.md](MODDING.md).
-Object with `MOD_INFO` type describes the mod itself.
-Each mod must have exactly one `MOD_INFO`, and unlike other types of objects from mods it is loaded on game launch,
-before the title screen shows up. As such, any and all errors related to it will show up before the title screen shows up.
+Object with `MOD_INFO` type describes the mod itself. Each mod must have exactly one `MOD_INFO`, and
+unlike other types of objects from mods it is loaded on game launch, before the title screen shows
+up. As such, any and all errors related to it will show up before the title screen shows up.
-Current convention is to put your `MOD_INFO` in `mod_info.json` file within the root directory of the mod.
+Current convention is to put your `MOD_INFO` in `mod_info.json` file within the root directory of
+the mod.
Example:
+
```C++
[
{
@@ -3559,9 +3792,11 @@ Example:
# MOD tileset
-MOD tileset defines additional sprite sheets. It is specified as JSON object with `type` member set to `mod_tileset`.
+MOD tileset defines additional sprite sheets. It is specified as JSON object with `type` member set
+to `mod_tileset`.
Example:
+
```C++
[
{
@@ -3589,39 +3824,28 @@ Example:
```
## `compatibility`
+
(string)
-The internal ID of the compatible tilesets. MOD tileset is only applied when base tileset's ID exists in this field.
+The internal ID of the compatible tilesets. MOD tileset is only applied when base tileset's ID
+exists in this field.
## `tiles-new`
-Setting of sprite sheets. Same as `tiles-new` field in `tile_config`. Sprite files are loaded from the same folder json file exists.
+Setting of sprite sheets. Same as `tiles-new` field in `tile_config`. Sprite files are loaded from
+the same folder json file exists.
# Field types
- {
- "type": "field_type", // this is a field type
- "id": "fd_gum_web", // id of the field
- "immune_mtypes": [ "mon_spider_gum" ], // list of monster immune to this field
- "intensity_levels": [
- { "name": "shadow", // name of this level of intensity
- "light_override": 3.7 } //light level on the tile occupied by this field will be set at 3.7 not matter the ambient light.
- ],
- "bash": {
- "str_min": 1, // lower bracket of bashing damage required to bash
- "str_max": 3, // higher bracket
- "sound_vol": 2, // noise made when succesfully bashing the field
- "sound_fail_vol": 2, // noise made when failing to bash the field
- "sound": "shwip", // sound on success
- "sound_fail": "shwomp", // sound on failure
- "msg_success": "You brush the gum web aside.", // message on success
- "move_cost": 120, // how many moves it costs to succesfully bash that field (default: 100)
- "items": [ // item dropped upon succesful bashing
- { "item": "2x4", "count": [ 5, 8 ] },
- { "item": "nail", "charges": [ 6, 8 ] },
- { "item": "splinter", "count": [ 3, 6 ] },
- { "item": "rag", "count": [ 40, 55 ] },
- { "item": "scrap", "count": [ 10, 20 ] }
- ]
- }
- }
+{ "type": "field_type", // this is a field type "id": "fd_gum_web", // id of the field
+"immune_mtypes": [ "mon_spider_gum" ], // list of monster immune to this field "intensity_levels": [
+{ "name": "shadow", // name of this level of intensity "light_override": 3.7 } //light level on the
+tile occupied by this field will be set at 3.7 not matter the ambient light. ], "bash": { "str_min":
+1, // lower bracket of bashing damage required to bash "str_max": 3, // higher bracket "sound_vol":
+2, // noise made when succesfully bashing the field "sound_fail_vol": 2, // noise made when failing
+to bash the field "sound": "shwip", // sound on success "sound_fail": "shwomp", // sound on failure
+"msg_success": "You brush the gum web aside.", // message on success "move_cost": 120, // how many
+moves it costs to succesfully bash that field (default: 100) "items": [ // item dropped upon
+succesful bashing { "item": "2x4", "count": [ 5, 8 ] }, { "item": "nail", "charges": [ 6, 8 ] }, {
+"item": "splinter", "count": [ 3, 6 ] }, { "item": "rag", "count": [ 40, 55 ] }, { "item": "scrap",
+"count": [ 10, 20 ] } ] } }
diff --git a/doc/JSON_INHERITANCE.md b/doc/JSON_INHERITANCE.md
index ee53b72e8d6b..4cc6919492a7 100644
--- a/doc/JSON_INHERITANCE.md
+++ b/doc/JSON_INHERITANCE.md
@@ -1,8 +1,12 @@
# JSON Inheritance
-To reduce duplication in the JSON data it is possible for some types to inherit from an existing type.
+
+To reduce duplication in the JSON data it is possible for some types to inherit from an existing
+type.
## Examples
-In the following condensed example ```556``` ammo is derived from ```223``` ammo via ```copy-from```:
+
+In the following condensed example `556` ammo is derived from `223` ammo via `copy-from`:
+
```
"id": "556",
"copy-from": "223",
@@ -16,19 +20,26 @@ In the following condensed example ```556``` ammo is derived from ```223``` ammo
},
"extend": { "effects": [ "NEVER_MISFIRES" ] }
```
+
The following rules apply to the above example:
-* Missing fields have the same value as the parent
+- Missing fields have the same value as the parent
-* Fields explicitly specified replace those of the parent type. The above example replaces ```name```, ```description``` and ```price```.
+- Fields explicitly specified replace those of the parent type. The above example replaces `name`,
+ `description` and `price`.
-* Numeric values may be specified ```relative``` to the parent. For example ```556``` has less ```damage``` but more ```pierce``` than ```223``` and will maintain this relationship if the definition for ```223``` is changed.
+- Numeric values may be specified `relative` to the parent. For example `556` has less `damage` but
+ more `pierce` than `223` and will maintain this relationship if the definition for `223` is
+ changed.
-* Flags can be added via ```extend```. For example ```556``` is military ammo and gains the ```NEVER_MISFIRES``` ammo effect. Any existing flags specified from ```223``` are preserved.
+- Flags can be added via `extend`. For example `556` is military ammo and gains the `NEVER_MISFIRES`
+ ammo effect. Any existing flags specified from `223` are preserved.
-* The entry you copied from must be of the same ```type``` as the item you added or changed (not all types are supported, see 'support' below)
+- The entry you copied from must be of the same `type` as the item you added or changed (not all
+ types are supported, see 'support' below)
-Reloaded ammo is derived from the factory equivalent but with a 10% penalty to ```damage``` and ```dispersion``` and a chance to misfire:
+Reloaded ammo is derived from the factory equivalent but with a 10% penalty to `damage` and
+`dispersion` and a chance to misfire:
```
"id": "reloaded_556",
@@ -42,15 +53,22 @@ Reloaded ammo is derived from the factory equivalent but with a 10% penalty to `
"extend": { "effects": [ "RECYCLED" ] },
"delete": { "effects": [ "NEVER_MISFIRES" ] }
```
+
The following additional rules apply to the above example:
-Chained inheritance is possible; for example ```reloaded_556``` inherits from ```556``` which is itself derived from ```223```
+Chained inheritance is possible; for example `reloaded_556` inherits from `556` which is itself
+derived from `223`
-Numeric values may be specified ```proportional``` to the parent by via a decimal factor where ```0.5``` is 50% and ```2.0``` is 200%.
+Numeric values may be specified `proportional` to the parent by via a decimal factor where `0.5` is
+50% and `2.0` is 200%.
-Flags can be deleted via ```delete```. It is not an error if the deleted flag does not exist in the parent.
+Flags can be deleted via `delete`. It is not an error if the deleted flag does not exist in the
+parent.
+
+It is possible to define an `abstract` type that exists only for other types to inherit from and
+cannot itself be used in game. In the following condensed example `magazine_belt` provides values
+common to all implemented ammo belts:
-It is possible to define an ```abstract``` type that exists only for other types to inherit from and cannot itself be used in game. In the following condensed example ```magazine_belt``` provides values common to all implemented ammo belts:
```
"abstract": "magazine_belt",
"type": "MAGAZINE",
@@ -63,14 +81,18 @@ It is possible to define an ```abstract``` type that exists only for other types
},
"flags": [ "MAG_BELT", "MAG_DESTROY" ]
```
+
The following additional rules apply to the above example:
-Missing mandatory fields do not result in errors as the ```abstract``` type is discarded after JSON loading completes
+Missing mandatory fields do not result in errors as the `abstract` type is discarded after JSON
+loading completes
Missing optional fields are set to the usual defaults for that type
## Support
+
The following types currently support inheritance:
+
```
GENERIC
AMMO
@@ -83,12 +105,21 @@ BOOK
ENGINE
```
-To find out if a types supports copy-from, you need to know if it has implemented generic_factory. To find out if this is the case, do the following:
-* Open [init.cpp](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/src/init.cpp)
-* Find the line that mentions your type, for example `add( "gate", &gates::load );`
-* Copy the load function, in this case it would be *gates::load*
-* Use this in [the search bar on github](https://github.com/CleverRaven/Cataclysm-DDA/search?q=%22gates%3A%3Aload%22&unscoped_q=%22gates%3A%3Aload%22&type=Code) to find the file that contains *gates::load*
-* In the search results you find [gates.cpp](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/src/gates.cpp). open it.
-* In gates.cpp, find the generic_factory line, it looks like this: `generic_factory gates_data( "gate type", "handle", "other_handles" );`
-* Since the generic_factory line is present, you can now conclude that it supports copy-from.
-* If you don't find generic_factoy present, it does not support copy-from, as is the case for type vitamin (repeat the above steps and find that [vitamin.cpp](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/src/vitamin.cpp) does not contain generic_factoy)
+To find out if a types supports copy-from, you need to know if it has implemented generic_factory.
+To find out if this is the case, do the following:
+
+- Open [init.cpp](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/src/init.cpp)
+- Find the line that mentions your type, for example `add( "gate", &gates::load );`
+- Copy the load function, in this case it would be _gates::load_
+- Use this in
+ [the search bar on github](https://github.com/CleverRaven/Cataclysm-DDA/search?q=%22gates%3A%3Aload%22&unscoped_q=%22gates%3A%3Aload%22&type=Code)
+ to find the file that contains _gates::load_
+- In the search results you find
+ [gates.cpp](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/src/gates.cpp). open it.
+- In gates.cpp, find the generic_factory line, it looks like this:
+ `generic_factory gates_data( "gate type", "handle", "other_handles" );`
+- Since the generic_factory line is present, you can now conclude that it supports copy-from.
+- If you don't find generic_factoy present, it does not support copy-from, as is the case for type
+ vitamin (repeat the above steps and find that
+ [vitamin.cpp](https://github.com/CleverRaven/Cataclysm-DDA/tree/master/src/vitamin.cpp) does not
+ contain generic_factoy)
diff --git a/doc/JSON_STYLE.md b/doc/JSON_STYLE.md
index 9cd950bb341f..3799f6668823 100644
--- a/doc/JSON_STYLE.md
+++ b/doc/JSON_STYLE.md
@@ -1,8 +1,10 @@
# JSON Style Guide
-Like in `doc/CODE_STYLE.md`, the JSON styling policy is to update JSON as it is added or edited, and in relatively small chunks otherwise in order to prevent undue disruption to development.
+Like in `doc/CODE_STYLE.md`, the JSON styling policy is to update JSON as it is added or edited, and
+in relatively small chunks otherwise in order to prevent undue disruption to development.
-We haven't been able to find a decent JSON styling tool, so we wrote our own. It lives in tools/format/format.cpp and it leverages src/json.cpp to parse and emit JSON.
+We haven't been able to find a decent JSON styling tool, so we wrote our own. It lives in
+tools/format/format.cpp and it leverages src/json.cpp to parse and emit JSON.
## JSON Example
@@ -13,7 +15,7 @@ This example outlines most of the styling features:
{
"type": "foo",
"id": "example",
- "short_array": [ 1, 2, 3, 4, 5 ],
+ "short_array": [1, 2, 3, 4, 5],
"short_object": {
"item_a": "a",
"item_b": "b"
@@ -24,46 +26,45 @@ This example outlines most of the styling features:
],
"nested_array": [
[
- [ "item1", "value1" ],
- [ "item2", "value2" ],
- [ "item3", "value3" ],
- [ "item4", "value4" ],
- [ "item5", "value5" ],
- [ "item6", "value6" ]
+ ["item1", "value1"],
+ ["item2", "value2"],
+ ["item3", "value3"],
+ ["item4", "value4"],
+ ["item5", "value5"],
+ ["item6", "value6"]
]
]
}
]
```
-Indention is two spaces.
-All JSON delimiters except comma and colon are surrounded by whitespace (either a space or a newline).
-Comma and colon are followed by whitespace.
-Object entries are always newline-separated.
-Array entries are newline-separated if the resulting array would exceed 120 characters otherwise (including indention).
-Line breaks occur after open brackets, close brackets, or entries.
+
+Indention is two spaces. All JSON delimiters except comma and colon are surrounded by whitespace
+(either a space or a newline). Comma and colon are followed by whitespace. Object entries are always
+newline-separated. Array entries are newline-separated if the resulting array would exceed 120
+characters otherwise (including indention). Line breaks occur after open brackets, close brackets,
+or entries.
## Formatting tool
-The formatting tool can be invoked via the Makefile, directly as `tools/format/json_formatter.cgi` (built via `make style-json`), or via cgi at http://dev.narc.ro/cataclysm/format.html
+The formatting tool can be invoked via the Makefile, directly as `tools/format/json_formatter.cgi`
+(built via `make style-json`), or via cgi at http://dev.narc.ro/cataclysm/format.html
-If you're using the Visual Studio solution, you can configure Visual Studio with
-commands to format all of the JSON in the project.
+If you're using the Visual Studio solution, you can configure Visual Studio with commands to format
+all of the JSON in the project.
-1. Build the JsonFormatter project by either building the entire solution or
- just that project. This will create a `tools/format/json_formatter.exe`
- binary.
-2. Add a new external tool entry ( `Tools` > `External Tools..` > `Add` ) and
- configure it as follows:
- * Title: `Lint All JSON`
- * Command: `C:\windows\system32\windowspowershell\v1.0\powershell.exe`
- * Arguments: `-file $(SolutionDir)\style-json.ps1`
- * Initial Directory: `$(SolutionDir)`
- * Use Output window: *checked*
+1. Build the JsonFormatter project by either building the entire solution or just that project. This
+ will create a `tools/format/json_formatter.exe` binary.
+2. Add a new external tool entry ( `Tools` > `External Tools..` > `Add` ) and configure it as
+ follows:
+ - Title: `Lint All JSON`
+ - Command: `C:\windows\system32\windowspowershell\v1.0\powershell.exe`
+ - Arguments: `-file $(SolutionDir)\style-json.ps1`
+ - Initial Directory: `$(SolutionDir)`
+ - Use Output window: _checked_
-At this point, you can use the menu ( `Tools` > `Lint All JSON` ) to invoke the
-command and can look in the Output Window for the output of running it.
-Additionally, you can configure a keybinding for this command by navigating to
-`Tools` > `Options` > `Environment` > `Keyboard`, searching for commands
-containing `Tools.ExternalCommand` and pick the one that corresponds to the
-position of your command in the list (e.g. `Tools.ExternalCommand1` if it's the
-top item in the list) and then assign shortcut keys to it.
+At this point, you can use the menu ( `Tools` > `Lint All JSON` ) to invoke the command and can look
+in the Output Window for the output of running it. Additionally, you can configure a keybinding for
+this command by navigating to `Tools` > `Options` > `Environment` > `Keyboard`, searching for
+commands containing `Tools.ExternalCommand` and pick the one that corresponds to the position of
+your command in the list (e.g. `Tools.ExternalCommand1` if it's the top item in the list) and then
+assign shortcut keys to it.
diff --git a/doc/LUA_SUPPORT.md b/doc/LUA_SUPPORT.md
index e7bc9477350d..4134a306b723 100644
--- a/doc/LUA_SUPPORT.md
+++ b/doc/LUA_SUPPORT.md
@@ -29,7 +29,6 @@ Use the `Home` key to return to the top.
- [Binding new enum to Lua](#binding-new-enum-to-lua)
- [Binding new string\_id or int\_id to Lua](#binding-new-string_id-or-int_id-to-lua)
-
## Introduction
This document describes implementation details behind Lua integration in Cataclysm: Bright Nights.
@@ -46,32 +45,31 @@ Programming in Lua (first edition): https://www.lua.org/pil/contents.html
## Example mods
-There are a couple heavily-commented example mods in `data/mods/` that make use
-of Lua API described here:
-* `smart_house_remotes` - Add remotes for controlling garage doors and window curtains.
-* `saveload_lua_test` - Mod for testing Lua save/load API.
+There are a couple heavily-commented example mods in `data/mods/` that make use of Lua API described
+here:
+
+- `smart_house_remotes` - Add remotes for controlling garage doors and window curtains.
+- `saveload_lua_test` - Mod for testing Lua save/load API.
## Ingame Lua console
-In-game Lua console is available through the debug menu or via `Lua Console`
-hotkey (unbound by default).
+In-game Lua console is available through the debug menu or via `Lua Console` hotkey (unbound by
+default).
-It is rather simple, but is capable of keeping input history, showing output and
-errors from Lua scripts as well as running Lua snippets and printing the
-returned values.
+It is rather simple, but is capable of keeping input history, showing output and errors from Lua
+scripts as well as running Lua snippets and printing the returned values.
-You can adjust console log capacity by running `gdebug.set_log_capacity( num )`
-(default is 100 entries), or clear it by running `gdebug.clear_lua_log()`.
+You can adjust console log capacity by running `gdebug.set_log_capacity( num )` (default is 100
+entries), or clear it by running `gdebug.clear_lua_log()`.
## Lua hot-reload
To speed up mod development process, BN supports Lua hot-reload functionality.
-There is no filesystem watcher, so hot-reload must be triggered manually via a
-corresponding `Reload Lua Code` hotkey (unbound by default). The hot-reload can
-also be triggered from console window by pressing the corresponding hotkey, or
-by running `gdebug.reload_lua_code()` command. Running the command from regular Lua
-scripts may have unintended consequences, use at your own risk!
+There is no filesystem watcher, so hot-reload must be triggered manually via a corresponding
+`Reload Lua Code` hotkey (unbound by default). The hot-reload can also be triggered from console
+window by pressing the corresponding hotkey, or by running `gdebug.reload_lua_code()` command.
+Running the command from regular Lua scripts may have unintended consequences, use at your own risk!
Note that not all code can be hot-reloaded, it'll be explained in later sections.
@@ -80,47 +78,54 @@ Note that not all code can be hot-reloaded, it'll be explained in later sections
When a world is being loaded, game does it in roughly these steps:
1. Initializes world-related internal state, sets the world as active
-2. Loads world's artifact item types (artifacts are hacky and will likely be removed soon in favor of relics + Lua)
+2. Loads world's artifact item types (artifacts are hacky and will likely be removed soon in favor
+ of relics + Lua)
3. Retrieves list of mods used by the world
4. Loads the mods according to the list
5. Initializes avatar-related internal state
6. Loads from save dir actual overmap data, avatar data and reality bubble data
What we care about here is the mod loading stage. It has a number of sub-steps:
+
1. Loading function receives list of world mods
2. It discards the missing ones and prints debug message for each
-3. It checks remaining mods on the list, and throws error if a mod needs Lua, but the game build does NOT support Lua
+3. It checks remaining mods on the list, and throws error if a mod needs Lua, but the game build
+ does NOT support Lua
4. It also throws a warning if game's Lua API version differs from the one used by the mod
5. For every mod on the list that uses Lua, it runs the mod's `preload.lua` script (if present)
-6. It goes over all mods in same order as in the list, and loads JSON definitions from each mod's folder
+6. It goes over all mods in same order as in the list, and loads JSON definitions from each mod's
+ folder
7. It finalizes loaded data (resolves copy-from, prepares some types with complex state for use)
8. For every mod on the list that uses Lua, it runs the mod's `finalize.lua` script (if present)
-9. It checks consistency of loaded data (validates values, warns about iffy combinations of values, etc.)
+9. It checks consistency of loaded data (validates values, warns about iffy combinations of values,
+ etc.)
10. (R) For every mod on the list that uses Lua, it runs the mod's `main.lua` script (if present)
-As such, we only have 3 scipts to place a mod's Lua code into: `preload.lua`, `finalize.lua` and `main.lua`.
-The differences between the 3 and their intended use cases will be explained below.
+As such, we only have 3 scipts to place a mod's Lua code into: `preload.lua`, `finalize.lua` and
+`main.lua`. The differences between the 3 and their intended use cases will be explained below.
You can use only one script, two or all three, depending on your needs.
-When executing hot-reload, the game repeats the step marked with (R).
-That means if you want the code you're working on to be hot-reloadable, put it into `main.lua`.
+When executing hot-reload, the game repeats the step marked with (R). That means if you want the
+code you're working on to be hot-reloadable, put it into `main.lua`.
### preload.lua
-This script is supposed to register event hooks and set up definitions that will
-then be referred by game JSON loading system (e.g. item use actions).
-Note that you can registers them here, and define them in some later stage
-(e.g. in `main.lua` to allow hot-reload to affect your hooks).
+
+This script is supposed to register event hooks and set up definitions that will then be referred by
+game JSON loading system (e.g. item use actions). Note that you can registers them here, and define
+them in some later stage (e.g. in `main.lua` to allow hot-reload to affect your hooks).
### finalize.lua
-This script is supposed to allow mods to modify definitions loaded from JSON
-after copy-from has been resolved, but there is no API for this yet.
+
+This script is supposed to allow mods to modify definitions loaded from JSON after copy-from has
+been resolved, but there is no API for this yet.
TODO: api for finalization
### main.lua
-This script is supposed to implement the main logic of the mod.
-This includes, but not limited, to:
+
+This script is supposed to implement the main logic of the mod. This includes, but not limited, to:
+
1. Mod runtime state
2. Mod initialization on game start
3. Mod save/load code, if required
@@ -128,68 +133,77 @@ This includes, but not limited, to:
## Lua API details
-While you can do a lot of interesting stuff with vanilla Lua, the integration
-imposes some limits to prevent potential bugs:
+While you can do a lot of interesting stuff with vanilla Lua, the integration imposes some limits to
+prevent potential bugs:
+
- Loading packages (or Lua modules) is disabled.
- Your current mod id is stored in `game.current_mod` variable
-- Your mod's runtime state should live in `game.mod_runtime[ game.current_mod ]` table.
- You can also interface with other mods if you know their id, by accessing their runtime state
- in a similar way with `game.mod_runtime[ that_other_mod_id ]`
-- Changes to global state are not available between scripts.
- This is to prevent accidental collisions between function names and variable names.
- You still can define global variables and functions, but they will be visible to your mod only.
+- Your mod's runtime state should live in `game.mod_runtime[ game.current_mod ]` table. You can also
+ interface with other mods if you know their id, by accessing their runtime state in a similar way
+ with `game.mod_runtime[ that_other_mod_id ]`
+- Changes to global state are not available between scripts. This is to prevent accidental
+ collisions between function names and variable names. You still can define global variables and
+ functions, but they will be visible to your mod only.
### Lua libraries and functions
+
When script is called, it comes with some standard Lua libraries pre-loaded:
-Library | Description
---------------|------------
- `base` | print, assert, and other base functions
- `math` | all things math
- `string` | string library
- `table` | the table manipulator and observer functions
+| Library | Description |
+| -------- | -------------------------------------------- |
+| `base` | print, assert, and other base functions |
+| `math` | all things math |
+| `string` | string library |
+| `table` | the table manipulator and observer functions |
See `Standard Libraries` section in Lua manual for details.
-Some of the functions here are overloaded by BN, see [Global overrides](#global-overrides) for details.
+Some of the functions here are overloaded by BN, see [Global overrides](#global-overrides) for
+details.
### Global state
-Most of necessary data and game runtime state is available through global `game` table.
-It has the following members:
-game.current_mod Id of mod that's being loaded (available only when script is executed)
-game.active_mods List of active world mods, in load order
-game.mod_runtime. Runtime data for mods (each mod gets its own table named after its id)
-game.mod_storage. Per-mod storage that gets automatically saved/loaded on game save/load.
-game.cata_internal For internal game purposes, please don't use this
-game.hooks. Hooks exposed to Lua scripts, will be called on corresponding events
-game.iuse. Item use functions that will be recognized by the item factory and called on item use
+Most of necessary data and game runtime state is available through global `game` table. It has the
+following members:
+
+game.current_mod Id of mod that's being loaded (available only when script is executed)
+game.active_mods List of active world mods, in load order game.mod_runtime. Runtime data for
+mods (each mod gets its own table named after its id) game.mod_storage. Per-mod storage that
+gets automatically saved/loaded on game save/load. game.cata_internal For internal game purposes,
+please don't use this game.hooks. Hooks exposed to Lua scripts, will be called on
+corresponding events game.iuse. Item use functions that will be recognized by the item
+factory and called on item use
### Game Bindings
-The game exposes various functions, constants and types to Lua.
-Functions and constants are organized into "libraries" for organizational purposes.
-Types are available globally, and may have member functions and fields.
-To see the full list of functions, constants and types, run the game with `--lua-doc` command line argument.
-This will generate documentation file `lua_doc.md` that will be placed in your `config` folder.
+The game exposes various functions, constants and types to Lua. Functions and constants are
+organized into "libraries" for organizational purposes. Types are available globally, and may have
+member functions and fields.
+
+To see the full list of functions, constants and types, run the game with `--lua-doc` command line
+argument. This will generate documentation file `lua_doc.md` that will be placed in your `config`
+folder.
#### Global overrides
+
Some functions have been globally overriden to improve integration with the game.
-Function | Description
-------------------------|-------------
-print | Print as `INFO LUA` to debug.log (overrides default Lua print)
-dofile | Disabled
-loadfile | Disabled
-load | Disabled
-loadstring | Disabled
+| Function | Description |
+| ---------- | -------------------------------------------------------------- |
+| print | Print as `INFO LUA` to debug.log (overrides default Lua print) |
+| dofile | Disabled |
+| loadfile | Disabled |
+| load | Disabled |
+| loadstring | Disabled |
TODO: alternatives for dofile and such
#### Hooks
-To see the list of hooks, check `hooks_doc` section of the autogenerated documentation file.
-There, you will see the list of hook ids as well as function signatures that they expect.
-You can register new hooks by appending to the hooks table like so:
+
+To see the list of hooks, check `hooks_doc` section of the autogenerated documentation file. There,
+you will see the list of hook ids as well as function signatures that they expect. You can register
+new hooks by appending to the hooks table like so:
+
```lua
-- In preload.lua
local mod = game.mod_runtime[ game.current_mod ]
@@ -209,8 +223,9 @@ end
```
#### Item use function
-Item use functions use unique id to register themselves in item factory.
-On item activation, they receive multiple arguments that will be described in the example below.
+
+Item use functions use unique id to register themselves in item factory. On item activation, they
+receive multiple arguments that will be described in the example below.
```lua
-- In preload.lua
@@ -232,9 +247,9 @@ end
```
#### Translation functions
-To make the mod translatable to other languages,
-get your text via functions bound in `locale` library.
-See [TRANSLATING.md](TRANSLATING.md) for detailed explanation of their C++ counterparts.
+
+To make the mod translatable to other languages, get your text via functions bound in `locale`
+library. See [TRANSLATING.md](TRANSLATING.md) for detailed explanation of their C++ counterparts.
Usage examples are shown below:
@@ -304,90 +319,89 @@ local ok = locale.gettext("Confusing text that needs explanation.")
--~ just use 2 or more single-line comments.
--~ They'll be concatenated and shown as a single multi-line comment.
local ok = locale.gettext("Confusing text that needs explanation.")
-
```
## C++ layout
-Lua build can be enabled by passing `LUA=1` to the Makefile, or enabling `LUA` build switch in CMake builds.
-Both msvc and android for simplicity always build with Lua **enabled**.
+
+Lua build can be enabled by passing `LUA=1` to the Makefile, or enabling `LUA` build switch in CMake
+builds. Both msvc and android for simplicity always build with Lua **enabled**.
### Lua source files
-To simplify build setup and improve portability we bundle `Lua 5.3.6` source
-code in `src/lua/` directory and have the build systems compile it and link into
-the game executable and library for tests.
+
+To simplify build setup and improve portability we bundle `Lua 5.3.6` source code in `src/lua/`
+directory and have the build systems compile it and link into the game executable and library for
+tests.
### Sol2 source files
-Sol2 makes it easy to bundle, we have `sol2 v3.3.0` single-header amalgamated
-version in `src/sol/` and just include it as needed. The header is quite large,
-so the less source files include it the better.
-* `sol/config.hpp` - Configuration header, we have a few options defined there
-* `sol/forward.hpp` - Forward declarations, a lightweight header that should be
- included in game headers instead of `sol/sol.hpp`
-* `sol/sol.hpp` - Main sol2 header file, quite large, avoid including in game
- headers
+
+Sol2 makes it easy to bundle, we have `sol2 v3.3.0` single-header amalgamated version in `src/sol/`
+and just include it as needed. The header is quite large, so the less source files include it the
+better.
+
+- `sol/config.hpp` - Configuration header, we have a few options defined there
+- `sol/forward.hpp` - Forward declarations, a lightweight header that should be included in game
+ headers instead of `sol/sol.hpp`
+- `sol/sol.hpp` - Main sol2 header file, quite large, avoid including in game headers
### Game source files
-All Lua-related game source files have the `catalua` prefix.
-If you want to add new bindings, consider looking at existing examples in
-`src/catalua_bindings.cpp` and reading relevant part of Sol2 docs.
-
-* `catalua.h` (and `catalua.cpp`) - Main Lua interface. It's the only header
- most of the codebase will have to include, and it provides a public interface
- that works in both `LUA=1` and `LUA=0` builds ( in builds without Lua, most of
- the functions there are no-op ).
-* `catalua_sol.h` and `catalua_sol_fwd.h` - Wrappers for `sol/sol.hpp` and
- `sol/forward.hpp` with custom pragmas to make them compile.
-* `catalua_bindings*` - Game Lua bindings live here.
-* `catalua_console.h`(`.cpp`) - Ingame Lua console.
-* `catalua_impl.h`(`.cpp`) - Implementation details for `catalua.h`(`.cpp`).
-* `catalua_iuse_actor.h`(`.cpp`) - Lua-driven `iuse_actor`.
-* `catalua_log.h`(`.cpp`) - In-memory logging for the console.
-* `catalua_luna.h` - Usertype registration interface with automatic doc generation, aka `luna`.
-* `catalua_luna_doc.h` - List of types registration through `luna` or exposed to its doc generator.
-* `catalua_readonly.h`(`.cpp`) - Functions for marking Lua tables as read-only.
-* `catalua_serde.h`(`.cpp`) - Lua table to/from JSON (de-)serialization.
-* `catalua_type_operators.h` - Macro that helps with implementing bindings for string_ids
+All Lua-related game source files have the `catalua` prefix.
+If you want to add new bindings, consider looking at existing examples in `src/catalua_bindings.cpp`
+and reading relevant part of Sol2 docs.
+
+- `catalua.h` (and `catalua.cpp`) - Main Lua interface. It's the only header most of the codebase
+ will have to include, and it provides a public interface that works in both `LUA=1` and `LUA=0`
+ builds ( in builds without Lua, most of the functions there are no-op ).
+- `catalua_sol.h` and `catalua_sol_fwd.h` - Wrappers for `sol/sol.hpp` and `sol/forward.hpp` with
+ custom pragmas to make them compile.
+- `catalua_bindings*` - Game Lua bindings live here.
+- `catalua_console.h`(`.cpp`) - Ingame Lua console.
+- `catalua_impl.h`(`.cpp`) - Implementation details for `catalua.h`(`.cpp`).
+- `catalua_iuse_actor.h`(`.cpp`) - Lua-driven `iuse_actor`.
+- `catalua_log.h`(`.cpp`) - In-memory logging for the console.
+- `catalua_luna.h` - Usertype registration interface with automatic doc generation, aka `luna`.
+- `catalua_luna_doc.h` - List of types registration through `luna` or exposed to its doc generator.
+- `catalua_readonly.h`(`.cpp`) - Functions for marking Lua tables as read-only.
+- `catalua_serde.h`(`.cpp`) - Lua table to/from JSON (de-)serialization.
+- `catalua_type_operators.h` - Macro that helps with implementing bindings for string_ids
### Adding new type to the doc generator without binding internals
-If a C++ type has not been registered in the doc generator,
-it will show up as ``.
-To mitigate this problem, you can add `LUNA_VAL( your_type, "YourType" )` in `catalua_luna_doc.h`,
-and the generator will use `YourType` string for argument type.
+If a C++ type has not been registered in the doc generator, it will show up as
+``. To mitigate this problem, you can add
+`LUNA_VAL( your_type, "YourType" )` in `catalua_luna_doc.h`, and the generator will use `YourType`
+string for argument type.
### Binding new type to Lua
-First, we need to register the new type with the bindings system.
-It needs to be done for many reasons, including so that the doc generator understands it,
-and the runtime can deserialize from JSON any Lua table that contains that type.
-If you don't you'll get a compile error saying
+
+First, we need to register the new type with the bindings system. It needs to be done for many
+reasons, including so that the doc generator understands it, and the runtime can deserialize from
+JSON any Lua table that contains that type. If you don't you'll get a compile error saying
`Type must implement luna_traits`.
-1. In `catala_luna_doc.h`, add declaration for your type.
- For example, if we're binding an imaginary `horde` type (which is a `struct`),
- it will be a single line near the top of the file:
+1. In `catala_luna_doc.h`, add declaration for your type. For example, if we're binding an imaginary
+ `horde` type (which is a `struct`), it will be a single line near the top of the file:
```c++
struct horde;
```
- Complex templated types may need to actually pull in the relevant header,
- but please avoid it as it heavily impacts compilation times.
+ Complex templated types may need to actually pull in the relevant header, but please avoid it as
+ it heavily impacts compilation times.
-2. In the same file, register your type with the doc generator.
- Continuing with the `horde` example, it's done like this:
+2. In the same file, register your type with the doc generator. Continuing with the `horde` example,
+ it's done like this:
```c++
LUNA_VAL( horde, "Horde" );
```
- While C++ types use all kinds of style for their names,
- on Lua side they all should be in `CamelCase`.
-
-Now we can actually get to the details.
-The bindings are implemented in `catalua_bindings*.cpp` files.
-They are spread out into multiple `.cpp` files to speed up compilation
-and make it easy to navigate, so you can put yours into any existing
-`catalua_bindings*.cpp` file or make your own similar file.
-They are also spread out into functions, for the same reasons.
-Let's register our `horde` type, and put it in a new file and a new function:
+ While C++ types use all kinds of style for their names, on Lua side they all should be in
+ `CamelCase`.
+
+Now we can actually get to the details. The bindings are implemented in `catalua_bindings*.cpp`
+files. They are spread out into multiple `.cpp` files to speed up compilation and make it easy to
+navigate, so you can put yours into any existing `catalua_bindings*.cpp` file or make your own
+similar file. They are also spread out into functions, for the same reasons. Let's register our
+`horde` type, and put it in a new file and a new function:
+
1. Add a new function declaration in `catalua_bindings.h`:
```c++
void reg_horde( sol::state &lua );
@@ -397,52 +411,51 @@ Let's register our `horde` type, and put it in a new file and a new function:
reg_horde( lua );
```
3. Make a new file, `catalua_bindings_horde.cpp`, with the following contents:
- ```c++
- #ifdef LUA
- #include "catalua_bindings.h"
-
- #include "horde.h" // Replace with the header where your type is defined
-
- void cata::detail::reg_horde( sol::state &lua )
- {
- sol::usertype ut =
- luna::new_usertype(
- lua,
- luna::no_bases,
- luna::constructors <
- // Define your actual constructors here
- horde(),
- horde( const point & ),
- horde( int, int )
- > ()
- );
-
- // Register all needed members
- luna::set( ut, "pos", &horde::pos );
- luna::set( ut, "size", &horde::size );
-
- // Register all needed methods
- luna::set_fx( ut, "move_to", &horde::move_to );
- luna::set_fx( ut, "update", &horde::update );
- luna::set_fx( ut, "get_printable_name", &horde::get_printable_name );
-
- // Add (de-)serialization functions so we can carry
- // our horde over the save/load boundary
- reg_serde_functions( ut );
-
- // Add more stuff like arithmetic operators, to_string operator, etc.
- }
- ```
-1. That's it. Your type is now visible in Lua under name `Horde`,
- and you can use the binded methods and members.
-
+ ```c++
+ #ifdef LUA
+ #include "catalua_bindings.h"
+
+ #include "horde.h" // Replace with the header where your type is defined
+
+ void cata::detail::reg_horde( sol::state &lua )
+ {
+ sol::usertype ut =
+ luna::new_usertype(
+ lua,
+ luna::no_bases,
+ luna::constructors <
+ // Define your actual constructors here
+ horde(),
+ horde( const point & ),
+ horde( int, int )
+ > ()
+ );
+
+ // Register all needed members
+ luna::set( ut, "pos", &horde::pos );
+ luna::set( ut, "size", &horde::size );
+
+ // Register all needed methods
+ luna::set_fx( ut, "move_to", &horde::move_to );
+ luna::set_fx( ut, "update", &horde::update );
+ luna::set_fx( ut, "get_printable_name", &horde::get_printable_name );
+
+ // Add (de-)serialization functions so we can carry
+ // our horde over the save/load boundary
+ reg_serde_functions( ut );
+
+ // Add more stuff like arithmetic operators, to_string operator, etc.
+ }
+ ```
+4. That's it. Your type is now visible in Lua under name `Horde`, and you can use the binded methods
+ and members.
### Binding new enum to Lua
-Binding enums is similar to binding types.
-Let's bind an imaginary `horde_type` enum here:
-1. If enum does not have an explicitly defined container
- (the `: type` part after `enum name` in the header where it's defined),
- you'll have to specify the container first, for example:
+
+Binding enums is similar to binding types. Let's bind an imaginary `horde_type` enum here:
+
+1. If enum does not have an explicitly defined container (the `: type` part after `enum name` in the
+ header where it's defined), you'll have to specify the container first, for example:
```diff
// hordes.h
- enum class horde_type {
@@ -460,45 +473,40 @@ Let's bind an imaginary `horde_type` enum here:
```c++
LUNA_ENUM( horde_type, "HordeType" )
```
-4. Ensure the enum implements the automatic conversion
- to/from `std::string`, see `enum_conversions.h` for details.
- Some enums will already have it, but most won't.
- Usually it's just a matter of specializing `enum_traits` for your enum `T`
- in the header, then defining `io::enum_to_string` in the `.cpp`
- file with enum -> string conversion.
- Some enums won't have the "last" value required for `enum_traits`.
- In that case, you'd have to add one:
- ```diff
- enum class horde_type : int {
- animals,
- robots,
- - zombies
- + zombies,
- + num_horde_types
- }
- ```
- Note that this only works for "monotonic" enums, i.e. ones that start with 0
- and don't skip any values. In the example above, `animals` has implicit
- value of `0`, robots has implicit value of `1` and `zombies` has implicit
- value of `2`, so we can easily add `num_horde_types`, which will have
- correct and expected implicit value of `3`.
+4. Ensure the enum implements the automatic conversion to/from `std::string`, see
+ `enum_conversions.h` for details. Some enums will already have it, but most won't. Usually it's
+ just a matter of specializing `enum_traits` for your enum `T` in the header, then defining
+ `io::enum_to_string` in the `.cpp` file with enum -> string conversion. Some enums won't have
+ the "last" value required for `enum_traits`. In that case, you'd have to add one:
+ ```diff
+ enum class horde_type : int {
+ animals,
+ robots,
+ - zombies
+ + zombies,
+ + num_horde_types
+ }
+ ```
+ Note that this only works for "monotonic" enums, i.e. ones that start with 0 and don't skip any
+ values. In the example above, `animals` has implicit value of `0`, robots has implicit value of
+ `1` and `zombies` has implicit value of `2`, so we can easily add `num_horde_types`, which will
+ have correct and expected implicit value of `3`.
5. Bind enum fields in `reg_enums` function in `catalua_bindings.cpp`:
```c++
reg_enum( lua );
```
- This uses the automatic convertion from step 4,
- so we have equal names between JSON and Lua.
-
-
+ This uses the automatic convertion from step 4, so we have equal names between JSON and Lua.
### Binding new string_id or int_id to Lua
+
Binding these can be done separately from binding `T` itself.
-1. Register your type `T` with the doc generator if you haven't already
- (see [relevant docs](#adding-new-type-to-the-doc-generator-without-binding-internals)).
+
+1. Register your type `T` with the doc generator if you haven't already (see
+ [relevant docs](#adding-new-type-to-the-doc-generator-without-binding-internals)).
2. Replace `LUNA_VAL` from step 1 with `LUNA_ID`.
-3. Ensure your type `T` implements operators `<` and `==`.
- It's usually easy implement them manually, and can be done semi-automatically with macro
- `LUA_TYPE_OPS` found in `catalua_type_operators.h`.
+3. Ensure your type `T` implements operators `<` and `==`. It's usually easy implement them
+ manually, and can be done semi-automatically with macro `LUA_TYPE_OPS` found in
+ `catalua_type_operators.h`.
4. In `catalua_bindings_ids.cpp`, add the header where your type T is defined:
```c++
#include "your_type_definition.h"
@@ -507,19 +515,19 @@ Binding these can be done separately from binding `T` itself.
```c++
reg_id( lua );
```
- That `true` can be replaced with `false` if you only want to bind
- `string_id` and don't care about (or can't implement) `int_id`.
-
- You may get linker errors at this stage, e.g. about `is_valid()` or
- `NULL_ID()` methods, which are for various reasons not implemented
- forall string or int ids. In this case, you'll have to define these manually,
- see relevant docs on `string_id` and `int_id` for more info.
-
-And that's it. Now, your type `T` will show up in Lua with `Raw` postfix,
-`string_id` will have `Id` postfix, and `int_id` will have `IntId` postfix.
-As example, for `LUNA_ID( horde, "Horde" )`, we'll get:
-* `horde` -> `HordeRaw`
-* `string_id` -> `HordeId`
-* `int_id` -> `HordeIntId`
-All type conversions between the 3 are implemented automatically by the system.
-Actual fields and methods of `T` can be binded to Lua same way as usual.
+
+That `true` can be replaced with `false` if you only want to bind `string_id` and don't care
+about (or can't implement) `int_id`.
+
+You may get linker errors at this stage, e.g. about `is_valid()` or `NULL_ID()` methods, which are
+for various reasons not implemented forall string or int ids. In this case, you'll have to define
+these manually, see relevant docs on `string_id` and `int_id` for more info.
+
+And that's it. Now, your type `T` will show up in Lua with `Raw` postfix, `string_id` will have
+`Id` postfix, and `int_id` will have `IntId` postfix. As example, for
+`LUNA_ID( horde, "Horde" )`, we'll get:
+
+- `horde` -> `HordeRaw`
+- `string_id` -> `HordeId`
+- `int_id` -> `HordeIntId` All type conversions between the 3 are implemented automatically
+ by the system. Actual fields and methods of `T` can be binded to Lua same way as usual.
diff --git a/doc/MAGIC.md b/doc/MAGIC.md
index a150b1df2540..3033dc18b5f6 100644
--- a/doc/MAGIC.md
+++ b/doc/MAGIC.md
@@ -1,4 +1,5 @@
# Spells, enchantments and other custom effects
+
- [Spells](#spells)
- [Currently Implemented Effects and special rules](#currently-implemented-effects-and-special-rules)
- [Spells that level up](#spells-that-level-up)
@@ -40,7 +41,6 @@
- [ITEM\_ARMOR\_X](#item_armor_x)
- [Examples](#examples)
-
# Spells
In `data/mods/Magiclysm` there is a template spell, copied here for your perusal:
@@ -94,86 +94,122 @@ In `data/mods/Magiclysm` there is a template spell, copied here for your perusal
"sound_variant": "shockwave" // the sound variant
}
```
-Most of the default values for the above are either 0 or "NONE", so you may leave out most of the values if they do not pertain to your spell.
-When deciding values for some of these, it is important to note that some of the formulae are not linear.
-For example, this is the formula for spell failure chance:
+Most of the default values for the above are either 0 or "NONE", so you may leave out most of the
+values if they do not pertain to your spell.
-```( ( ( ( spell_level - spell_difficulty ) * 2 + intelligence + spellcraft_skill ) - 30 ) / 30 ) ^ 2```
+When deciding values for some of these, it is important to note that some of the formulae are not
+linear. For example, this is the formula for spell failure chance:
-Meaning a spell with difficulty 0 cast by a player with 8 intelligence, 0 spellcraft, and level 0 in the spell will have a 53% spell failure chance.
-On the other hand, a player with 12 intelligence, 6 spellcraft, and level 6 in the same spell will have a 0% spell failure chance.
+`( ( ( ( spell_level - spell_difficulty ) * 2 + intelligence + spellcraft_skill ) - 30 ) / 30 ) ^ 2`
-However, experience gain is a little more complicated to calculate. The formula for how much experience you need to get to a level is below:
+Meaning a spell with difficulty 0 cast by a player with 8 intelligence, 0 spellcraft, and level 0 in
+the spell will have a 53% spell failure chance. On the other hand, a player with 12 intelligence, 6
+spellcraft, and level 6 in the same spell will have a 0% spell failure chance.
-```e ^ ( ( level + 62.5 ) * 0.146661 ) ) - 6200```
+However, experience gain is a little more complicated to calculate. The formula for how much
+experience you need to get to a level is below:
+
+`e ^ ( ( level + 62.5 ) * 0.146661 ) ) - 6200`
#### Currently Implemented Effects and special rules
-* "pain_split" - makes all of your limbs' damage even out.
+- "pain_split" - makes all of your limbs' damage even out.
+
+- "move_earth" - "digs" at the target location. some terrain is not diggable this way.
-* "move_earth" - "digs" at the target location. some terrain is not diggable this way.
+- "target_attack" - deals damage to a target (ignores walls). Negative damage heals the target. If
+ "effect_str" is included, it will add that effect (defined elsewhere in json) to the targets if
+ able, to the body parts defined in affected_body_parts. Any aoe will manifest as a circular area
+ centered on the target, and will only deal damage to valid_targets. (aoe does not ignore walls)
-* "target_attack" - deals damage to a target (ignores walls). Negative damage heals the target. If "effect_str" is included, it will add that effect (defined elsewhere in json) to the targets if able, to the body parts defined in affected_body_parts.
-Any aoe will manifest as a circular area centered on the target, and will only deal damage to valid_targets. (aoe does not ignore walls)
+- "projectile_attack" - similar to target_attack, except the projectile you shoot will stop short at
+ impassable terrain. If "effect_str" is included, it will add that effect (defined elsewhere in
+ json) to the targets if able, to the body parts defined in affected_body_parts.
-* "projectile_attack" - similar to target_attack, except the projectile you shoot will stop short at impassable terrain. If "effect_str" is included, it will add that effect (defined elsewhere in json) to the targets if able, to the body parts defined in affected_body_parts.
+- "cone_attack" - fires a cone toward the target up to your range. The arc of the cone in degrees is
+ aoe. Stops at walls. If "effect_str" is included, it will add that effect (defined elsewhere in
+ json) to the targets if able, to the body parts defined in affected_body_parts.
-* "cone_attack" - fires a cone toward the target up to your range. The arc of the cone in degrees is aoe. Stops at walls. If "effect_str" is included, it will add that effect (defined elsewhere in json) to the targets if able, to the body parts defined in affected_body_parts.
+- "line_attack" - fires a line with width aoe toward the target, being blocked by walls on the way.
+ If "effect_str" is included, it will add that effect (defined elsewhere in json) to the targets if
+ able, to the body parts defined in affected_body_parts.
-* "line_attack" - fires a line with width aoe toward the target, being blocked by walls on the way. If "effect_str" is included, it will add that effect (defined elsewhere in json) to the targets if able, to the body parts defined in affected_body_parts.
+- "spawn_item" - spawns an item that will disappear at the end of its duration. Default duration
+ is 0. Damage determines quantity.
-* "spawn_item" - spawns an item that will disappear at the end of its duration. Default duration is 0. Damage determines quantity.
+- "teleport_random" - teleports the player randomly range spaces with aoe variation
-* "teleport_random" - teleports the player randomly range spaces with aoe variation
+- "recover_energy" - recovers an energy source (defined in the effect_str, shown below) equal to
+ damage of the spell
-* "recover_energy" - recovers an energy source (defined in the effect_str, shown below) equal to damage of the spell
-- "MANA"
-- "STAMINA"
-- "FATIGUE"
-- "PAIN"
-- "BIONIC"
+* "MANA"
+* "STAMINA"
+* "FATIGUE"
+* "PAIN"
+* "BIONIC"
-* "ter_transform" - transform the terrain and furniture in an area centered at the target. The chance of any one of the points in the area of effect changing is one_in( damage ). The effect_str is the id of a ter_furn_transform.
+- "ter_transform" - transform the terrain and furniture in an area centered at the target. The
+ chance of any one of the points in the area of effect changing is one_in( damage ). The effect_str
+ is the id of a ter_furn_transform.
-* "vomit" - any creature within its area of effect will instantly vomit, if it's able to do so.
+- "vomit" - any creature within its area of effect will instantly vomit, if it's able to do so.
-* "timed_event" - adds a timed event to the player only. valid timed events: "help", "wanted", "robot_attack", "spawn_wyrms", "amigara", "roots_die", "temple_open", "temple_flood", "temple_spawn", "dim", "artifact_light" NOTE: This was added only for artifact active effects. support is limited, use at your own risk.
+- "timed_event" - adds a timed event to the player only. valid timed events: "help", "wanted",
+ "robot_attack", "spawn_wyrms", "amigara", "roots_die", "temple_open", "temple_flood",
+ "temple_spawn", "dim", "artifact_light" NOTE: This was added only for artifact active effects.
+ support is limited, use at your own risk.
-* "explosion" - an explosion is centered on the target, with power damage() and factor aoe()/10
+- "explosion" - an explosion is centered on the target, with power damage() and factor aoe()/10
-* "flashbang" - a flashbang effect is centered on the target, with poewr damage() and factor aoe()/10
+- "flashbang" - a flashbang effect is centered on the target, with poewr damage() and factor
+ aoe()/10
-* "mod_moves" - adds damage() moves to the target. can be negative to "freeze" the target for that amount of time
+- "mod_moves" - adds damage() moves to the target. can be negative to "freeze" the target for that
+ amount of time
-* "map" - maps the overmap centered on the player out to a radius of aoe()
+- "map" - maps the overmap centered on the player out to a radius of aoe()
-* "morale" - gives a morale effect to all npcs or avatar within aoe, with value damage(). decay_start is duration() / 10.
+- "morale" - gives a morale effect to all npcs or avatar within aoe, with value damage().
+ decay_start is duration() / 10.
-* "charm_monster" - charms a monster that has less hp than damage() for approximately duration()
+- "charm_monster" - charms a monster that has less hp than damage() for approximately duration()
-* "mutate" - mutates the target(s). if effect_str is defined, mutates toward that category instead of picking at random. the "MUTATE_TRAIT" flag allows effect_str to be a specific trait instead of a category. damage() / 100 is the percent chance the mutation will be successful (a value of 10000 represents 100.00%)
+- "mutate" - mutates the target(s). if effect_str is defined, mutates toward that category instead
+ of picking at random. the "MUTATE_TRAIT" flag allows effect_str to be a specific trait instead of
+ a category. damage() / 100 is the percent chance the mutation will be successful (a value of 10000
+ represents 100.00%)
-* "bash" - bashes the terrain at the target. uses damage() as the strength of the bash.
+- "bash" - bashes the terrain at the target. uses damage() as the strength of the bash.
-* "WONDER" - Unlike the above, this is not an "effect" but a "flag". This alters the behavior of the parent spell drastically: The spell itself doesn't cast, but its damage and range information is used in order to cast the extra_effects. N of the extra_effects will be chosen at random to be cast, where N is the current damage of the spell (stacks with RANDOM_DAMAGE flag) and the message of the spell cast by this spell will also be displayed. If this spell's message is not wanted to be displayed, make sure the message is an empty string.
+- "WONDER" - Unlike the above, this is not an "effect" but a "flag". This alters the behavior of the
+ parent spell drastically: The spell itself doesn't cast, but its damage and range information is
+ used in order to cast the extra_effects. N of the extra_effects will be chosen at random to be
+ cast, where N is the current damage of the spell (stacks with RANDOM_DAMAGE flag) and the message
+ of the spell cast by this spell will also be displayed. If this spell's message is not wanted to
+ be displayed, make sure the message is an empty string.
-* "RANDOM_TARGET" - A special spell flag (like wonder) that forces the spell to choose a random valid target within range instead of the caster choosing the target. This also affects extra_effects.
+- "RANDOM_TARGET" - A special spell flag (like wonder) that forces the spell to choose a random
+ valid target within range instead of the caster choosing the target. This also affects
+ extra_effects.
##### For Spells that have an attack type, these are the available damage types:
-* "fire"
-* "acid"
-* "bash"
-* "bio" - internal damage such as poison
-* "cold"
-* "cut"
-* "electric"
-* "stab"
-* "none" - this damage type goes through armor altogether. it is the default.
+
+- "fire"
+- "acid"
+- "bash"
+- "bio" - internal damage such as poison
+- "cold"
+- "cut"
+- "electric"
+- "stab"
+- "none" - this damage type goes through armor altogether. it is the default.
#### Spells that level up
-Spells that change effects as they level up must have a min and max effect and an increment. The min effect is what the spell will do at level 0, and the max effect is where it stops growing. The increment is how much it changes per level. For example:
+Spells that change effects as they level up must have a min and max effect and an increment. The min
+effect is what the spell will do at level 0, and the max effect is where it stops growing. The
+increment is how much it changes per level. For example:
```json
"min_range": 1,
@@ -181,27 +217,32 @@ Spells that change effects as they level up must have a min and max effect and a
"range_increment": 5,
```
-Min and max values must always have the same sign, but it can be negative eg. in the case of spells that use a negative 'recover' effect to cause pain or stamina damage. For example:
+Min and max values must always have the same sign, but it can be negative eg. in the case of spells
+that use a negative 'recover' effect to cause pain or stamina damage. For example:
```json
- {
- "id": "stamina_damage",
- "type": "SPELL",
- "name": "Tired",
- "description": "decreases stamina",
- "valid_targets": [ "hostile" ],
- "min_damage": -2000,
- "max_damage": -10000,
- "damage_increment": -3000,
- "max_level": 10,
- "effect": "recover_energy",
- "effect_str": "STAMINA"
- }
+{
+ "id": "stamina_damage",
+ "type": "SPELL",
+ "name": "Tired",
+ "description": "decreases stamina",
+ "valid_targets": ["hostile"],
+ "min_damage": -2000,
+ "max_damage": -10000,
+ "damage_increment": -3000,
+ "max_level": 10,
+ "effect": "recover_energy",
+ "effect_str": "STAMINA"
+}
```
### Learning Spells
-There are two ways of granting spells that is implemented: Mutating can grant a spell with the "spells_learned" field which also lets you specify the level granted. Otherwise you can learn a spell from an item through a use_action, which is also the only way to train a spell other than using it. Examples of both are shown below:
+There are two ways of granting spells that is implemented: Mutating can grant a spell with the
+"spells_learned" field which also lets you specify the level granted. Otherwise you can learn a
+spell from an item through a use_action, which is also the only way to train a spell other than
+using it. Examples of both are shown below:
+
```C++
{
"id": "DEBUG_spellbook",
@@ -218,7 +259,9 @@ There are two ways of granting spells that is implemented: Mutating can grant a
}
},
```
-You can study this spellbook for a rate of ~1 experience per turn depending on intelligence, spellcraft, and focus.
+
+You can study this spellbook for a rate of ~1 experience per turn depending on intelligence,
+spellcraft, and focus.
```json
"spells_learned": [ [ "debug_hp", 1 ], [ "debug_stamina", 1 ], [ "example_template", 1 ], [ "pain_split", 1 ] ],
@@ -227,56 +270,74 @@ You can study this spellbook for a rate of ~1 experience per turn depending on i
#### Spells in professions and NPC classes
You can add a "spell" member to professions or an NPC class definition like so:
+
```json
"spells": [ { "id": "summon_zombie", "level": 0 }, { "id": "magic_missile", "level": 10 } ]
```
-NOTE: This makes it possible to learn spells that conflict with a class. It also does not give the prompt to gain the class. Be judicious upon adding this to a profession!
+NOTE: This makes it possible to learn spells that conflict with a class. It also does not give the
+prompt to gain the class. Be judicious upon adding this to a profession!
#### Spells in monsters
You can assign a spell as a special attack for a monster.
+
```json
{ "type": "spell", "spell_id": "burning_hands", "spell_level": 10, "cooldown": 10 }
```
-* spell_id: the id for the spell being cast.
-* spell_level: the level at which the spell is cast. Spells cast by monsters do not gain levels like player spells.
-* cooldown: how often the monster can cast this spell
+
+- spell_id: the id for the spell being cast.
+- spell_level: the level at which the spell is cast. Spells cast by monsters do not gain levels like
+ player spells.
+- cooldown: how often the monster can cast this spell
# Enchantments
+
Enchantments make it possible to specify custom effects provided by item, bionic or mutation.
## Fields
+
### id
+
(string) Unique identifier for this enchantment.
### has
-(string) How an enchantment determines if it is in the right location in order to qualify for being active.
+
+(string) How an enchantment determines if it is in the right location in order to qualify for being
+active.
This field is relevant only for items.
Values:
-* `HELD` (default) - when in your inventory
-* `WIELD` - when wielded in your hand
-* `WORN` - when worn as armor
+
+- `HELD` (default) - when in your inventory
+- `WIELD` - when wielded in your hand
+- `WORN` - when worn as armor
### condition
-(string) How an enchantment determines if you are in the right environments in order for the enchantment to qualify for being active.
+
+(string) How an enchantment determines if you are in the right environments in order for the
+enchantment to qualify for being active.
Values:
-* `ALWAYS` (default) - Always active
-* `UNDERGROUND` - When the owner of the item is below Z-level 0
-* `UNDERWATER` - When the owner is in swimmable terrain
-* `ACTIVE` - whenever the item, mutation, bionic, or whatever the enchantment is attached to is active.
+
+- `ALWAYS` (default) - Always active
+- `UNDERGROUND` - When the owner of the item is below Z-level 0
+- `UNDERWATER` - When the owner is in swimmable terrain
+- `ACTIVE` - whenever the item, mutation, bionic, or whatever the enchantment is attached to is
+ active.
### emitter
-(string) Identifier of an emitter that's active as long as this enchantment is active.
-Default: no emitter.
+
+(string) Identifier of an emitter that's active as long as this enchantment is active. Default: no
+emitter.
### ench_effects
+
(array) Grants effects of specified intensity as long as this enchantment is active.
Syntax for single entry:
+
```C++
{
// (required) Identifier of the effect
@@ -288,9 +349,12 @@ Syntax for single entry:
```
### hit_you_effect
-(array) List of spells that may be cast when enchantment is active and character melee attacks a creature.
+
+(array) List of spells that may be cast when enchantment is active and character melee attacks a
+creature.
Syntax for single entry:
+
```c++
{
// (required) Identifier of the spell
@@ -324,17 +388,22 @@ Syntax for single entry:
```
### hit_me_effect
-(array) List of spells that may be cast when enchantment is active and character gets melee attacked by a creature.
+
+(array) List of spells that may be cast when enchantment is active and character gets melee attacked
+by a creature.
Same syntax as for `hit_you_effect`.
### mutations
+
(array) List of mutations temporarily granted while enchantment is active.
### intermittent_activation
+
(object) Rules that specify random effects which occur while enchantment is active.
Syntax:
+
```c++
{
// List of checks to run on every turn while enchantment is active.
@@ -365,9 +434,11 @@ Syntax:
```
### values
+
(array) List of miscellaneous character/item values to modify.
Syntax for single entry:
+
```c++
{
// (required) Value ID to modify, refer to list below.
@@ -383,156 +454,150 @@ Syntax for single entry:
```
Additive bonus is applied separately from multiplicative, like so:
+
```c++
bonus = add + base_value * multiply
```
-Thus, a `multiply` value of -0.8 is -80%, and a `multiply` of 2.5 is +250%.
-When modifying integer values, final bonus is rounded towards 0 (truncated).
+Thus, a `multiply` value of -0.8 is -80%, and a `multiply` of 2.5 is +250%. When modifying integer
+values, final bonus is rounded towards 0 (truncated).
When multiple enchantments (e.g. one from an item and one from a bionic) modify the same value,
-their bonuses are added together without rounding, then the sum is rounded (if necessary)
-before being applied to the base value.
+their bonuses are added together without rounding, then the sum is rounded (if necessary) before
+being applied to the base value.
-Since there's no limit on number of enchantments the character can have at a time,
-the final calculated values have hardcoded bounds to prevent unintended behavior.
+Since there's no limit on number of enchantments the character can have at a time, the final
+calculated values have hardcoded bounds to prevent unintended behavior.
#### IDs of modifiable values
#### Character values
##### STRENGTH
-Strength stat.
-`base_value` here is the base stat value.
-The final value cannot go below 0.
+
+Strength stat. `base_value` here is the base stat value. The final value cannot go below 0.
##### DEXTERITY
-Dexterity stat.
-`base_value` here is the base stat value.
-The final value cannot go below 0.
+
+Dexterity stat. `base_value` here is the base stat value. The final value cannot go below 0.
##### PERCEPTION
-Perception stat.
-`base_value` here is the base stat value.
-The final value cannot go below 0.
+
+Perception stat. `base_value` here is the base stat value. The final value cannot go below 0.
##### INTELLIGENCE
-Intelligence stat.
-`base_value` here is the base stat value.
-The final value cannot go below 0.
+
+Intelligence stat. `base_value` here is the base stat value. The final value cannot go below 0.
##### SPEED
-Character speed.
-`base_value` here is character speed including pain/hunger/weight penalties.
-Final speed value cannot go below 25% of base speed.
+
+Character speed. `base_value` here is character speed including pain/hunger/weight penalties. Final
+speed value cannot go below 25% of base speed.
##### ATTACK_COST
-Melee attack cost. The lower, the better.
-`base_value` here is attack cost for given weapon including modifiers from stats and skills.
-The final value cannot go below 25.
+
+Melee attack cost. The lower, the better. `base_value` here is attack cost for given weapon
+including modifiers from stats and skills. The final value cannot go below 25.
##### MOVE_COST
-Movement cost.
-`base_value` here is tile movement cost including modifiers from clothing and traits.
+
+Movement cost. `base_value` here is tile movement cost including modifiers from clothing and traits.
The final value cannot go below 20.
##### METABOLISM
-Metabolic rate.
-This modifier ignores `add` field.
-`base_value` here is `PLAYER_HUNGER_RATE` modified by traits.
-The final value cannot go below 0.
+
+Metabolic rate. This modifier ignores `add` field. `base_value` here is `PLAYER_HUNGER_RATE`
+modified by traits. The final value cannot go below 0.
##### MANA_CAP
-Mana capacity.
-`base_value` here is character's base mana capacity modified by traits.
-The final value cannot go below 0.
+
+Mana capacity. `base_value` here is character's base mana capacity modified by traits. The final
+value cannot go below 0.
##### MANA_REGEN
-Mana regeneration rate.
-This modifier ignores `add` field.
-`base_value` here is character's base mana gain rate modified by traits.
-The final value cannot go below 0.
+
+Mana regeneration rate. This modifier ignores `add` field. `base_value` here is character's base
+mana gain rate modified by traits. The final value cannot go below 0.
##### STAMINA_CAP
-Stamina capacity.
-This modifier ignores `add` field.
-`base_value` here is character's base stamina capacity modified by traits.
-The final value cannot go below 10% of `PLAYER_MAX_STAMINA`.
+
+Stamina capacity. This modifier ignores `add` field. `base_value` here is character's base stamina
+capacity modified by traits. The final value cannot go below 10% of `PLAYER_MAX_STAMINA`.
##### STAMINA_REGEN
-Stamina regeneration rate.
-This modifier ignores `add` field.
-`base_value` here is character's base stamina gain rate modified by mouth encumbrance.
-The final value cannot go below 0.
+
+Stamina regeneration rate. This modifier ignores `add` field. `base_value` here is character's base
+stamina gain rate modified by mouth encumbrance. The final value cannot go below 0.
##### THIRST
-Thirst gain rate.
-This modifier ignores `add` field.
-`base_value` here is character's base thirst gain rate.
-The final value cannot go below 0.
+
+Thirst gain rate. This modifier ignores `add` field. `base_value` here is character's base thirst
+gain rate. The final value cannot go below 0.
##### FATIGUE
-Fatigue gain rate.
-This modifier ignores `add` field.
-`base_value` here is character's base fatigue gain rate.
-The final value cannot go below 0.
+
+Fatigue gain rate. This modifier ignores `add` field. `base_value` here is character's base fatigue
+gain rate. The final value cannot go below 0.
##### BONUS_DODGE
-Additional dodges per turn before dodge penalty kicks in.
-`base_value` here is character's base dodges per turn before penalty (usually 1).
-The final value can go below 0, which results in penalty to dodge roll.
+
+Additional dodges per turn before dodge penalty kicks in. `base_value` here is character's base
+dodges per turn before penalty (usually 1). The final value can go below 0, which results in penalty
+to dodge roll.
##### ARMOR_X
-Incoming damage modifier.
-Applied after Active Defense System bionic but before the damage is absorbed by items.
-Note that `base_value` here is incoming damage value of corresponding type,
-so positive `add` and greater than 1 `mul` will **increase** damage received by the character.
-Each damage type has its own enchant value:
-* `ARMOR_ACID`
-* `ARMOR_BASH`
-* `ARMOR_BIO`
-* `ARMOR_BULLET`
-* `ARMOR_COLD`
-* `ARMOR_CUT`
-* `ARMOR_ELEC`
-* `ARMOR_HEAT`
-* `ARMOR_STAB`
+
+Incoming damage modifier. Applied after Active Defense System bionic but before the damage is
+absorbed by items. Note that `base_value` here is incoming damage value of corresponding type, so
+positive `add` and greater than 1 `mul` will **increase** damage received by the character. Each
+damage type has its own enchant value:
+
+- `ARMOR_ACID`
+- `ARMOR_BASH`
+- `ARMOR_BIO`
+- `ARMOR_BULLET`
+- `ARMOR_COLD`
+- `ARMOR_CUT`
+- `ARMOR_ELEC`
+- `ARMOR_HEAT`
+- `ARMOR_STAB`
#### Item values
##### ITEM_ATTACK_COST
-Attack cost (melee or throwing) for this item.
-Ignores condition / location, and is always active.
-`base_value` here is base item attack cost.
-Note that the final value cannot go below 0.
+
+Attack cost (melee or throwing) for this item. Ignores condition / location, and is always active.
+`base_value` here is base item attack cost. Note that the final value cannot go below 0.
##### ITEM_DAMAGE_X
-Melee damage of this item.
-Ignores condition / location, and is always active.
-`base_value` here is base item damage of corresponding type.
-Note that the final value cannot go below 0.
-Only some damage types are supported:
-* `ITEM_DAMAGE_BASH`
-* `ITEM_DAMAGE_CUT`
-* `ITEM_DAMAGE_STAB`
+
+Melee damage of this item. Ignores condition / location, and is always active. `base_value` here is
+base item damage of corresponding type. Note that the final value cannot go below 0. Only some
+damage types are supported:
+
+- `ITEM_DAMAGE_BASH`
+- `ITEM_DAMAGE_CUT`
+- `ITEM_DAMAGE_STAB`
##### ITEM_ARMOR_X
-Incoming damage modifier for this item, applied before the damage is absorbed by the item.
-Note that `base_value` here is incoming damage value of corresponding type,
-so positive `add` and greater than 1 `mul` will **increase** damage received by the character.
-Each damage type has its own enchant value:
-* `ITEM_ARMOR_ACID`
-* `ITEM_ARMOR_BASH`
-* `ITEM_ARMOR_BIO`
-* `ITEM_ARMOR_BULLET`
-* `ITEM_ARMOR_COLD`
-* `ITEM_ARMOR_CUT`
-* `ITEM_ARMOR_ELEC`
-* `ITEM_ARMOR_HEAT`
-* `ITEM_ARMOR_STAB`
+Incoming damage modifier for this item, applied before the damage is absorbed by the item. Note that
+`base_value` here is incoming damage value of corresponding type, so positive `add` and greater than
+1 `mul` will **increase** damage received by the character. Each damage type has its own enchant
+value:
+
+- `ITEM_ARMOR_ACID`
+- `ITEM_ARMOR_BASH`
+- `ITEM_ARMOR_BIO`
+- `ITEM_ARMOR_BULLET`
+- `ITEM_ARMOR_COLD`
+- `ITEM_ARMOR_CUT`
+- `ITEM_ARMOR_ELEC`
+- `ITEM_ARMOR_HEAT`
+- `ITEM_ARMOR_STAB`
## Examples
+
```json
[
{
@@ -555,13 +620,13 @@ Each damage type has its own enchant value:
"id": "ENCH_ULTIMATE_ASSKICK",
"has": "WIELD",
"condition": "ALWAYS",
- "ench_effects": [ { "effect": "invisibility", "intensity": 1 } ],
- "hit_you_effect": [ { "id": "AEA_FIREBALL" } ],
- "hit_me_effect": [ { "id": "AEA_HEAL" } ],
- "mutations": [ "KILLER", "PARKOUR" ],
- "values": [ { "value": "STRENGTH", "multiply": 1.1, "add": -5 } ],
+ "ench_effects": [{ "effect": "invisibility", "intensity": 1 }],
+ "hit_you_effect": [{ "id": "AEA_FIREBALL" }],
+ "hit_me_effect": [{ "id": "AEA_HEAL" }],
+ "mutations": ["KILLER", "PARKOUR"],
+ "values": [{ "value": "STRENGTH", "multiply": 1.1, "add": -5 }],
"intermittent_activation": {
- "effects": [
+ "effects": [
{
"frequency": "1 hour",
"spell_effects": [
diff --git a/doc/MANUAL_OF_STYLE.md b/doc/MANUAL_OF_STYLE.md
index 35d07fd30ab5..85b84c04a15a 100644
--- a/doc/MANUAL_OF_STYLE.md
+++ b/doc/MANUAL_OF_STYLE.md
@@ -1,10 +1,15 @@
Follow these conventions when adding or editing in-game text:
1. Use US English spelling.
-2. Use double sentence spacing after periods. This means that a period that ends a sentence should be followed by two spaces. If the sentence is the last in the block of text, there should be no spaces following it.
+2. Use double sentence spacing after periods. This means that a period that ends a sentence should
+ be followed by two spaces. If the sentence is the last in the block of text, there should be no
+ spaces following it.
3. Use second person point of view (eg. "you").
-4. The names of traits, martial arts, and Compact Bionics Modules (CBM's) should be in title case. This means that each word should be capitalized unless it is an article, preposition or conjunction.
+4. The names of traits, martial arts, and Compact Bionics Modules (CBM's) should be in title case.
+ This means that each word should be capitalized unless it is an article, preposition or
+ conjunction.
5. Items and entities with proper noun names should also be in title case.
6. All other item and entity names should be in all lower-case letters.
7. Use the serial comma (Oxford comma).
-8. Use ellipsis character (…) instead of three dots (...). Replace instances of three periods with the dedicated Unicode character for ellipsis, namely U+2026.
+8. Use ellipsis character (…) instead of three dots (...). Replace instances of three periods with
+ the dedicated Unicode character for ellipsis, namely U+2026.
diff --git a/doc/MAPGEN.md b/doc/MAPGEN.md
index 4dc9d8008bc6..490dc56a27fd 100644
--- a/doc/MAPGEN.md
+++ b/doc/MAPGEN.md
@@ -1,81 +1,82 @@
# MAPGEN
-* [How buildings and terrain are generated](#how-buildings-and-terrain-are-generated)
-* [Adding mapgen entries](#adding-mapgen-entries)
- * [Methods](#methods)
- * [Mapgen definition Placement](#mapgen-definition-placement)
- * [Embedded mapgen](#embedded-mapgen)
- * [Standalone mapgen](#standalone-mapgen)
- * [Format and variables](#format-and-variables)
- * [Define mapgen "method"](#define-mapgen-method)
- * [Define overmap terrain with "om_terrain" value, array, or nested array](#define-overmap-terrain-with-om_terrain-value-array-or-nested-array)
- * [Define mapgen "weight"](#define-mapgen-weight)
- * [How "overmap_terrain" variables affect mapgen](#how-overmap_terrain-variables-affect-mapgen)
- * [Limitations / TODO](#limitations--todo)
-* [JSON object definition](#json-object-definition)
- * [Fill terrain using "fill_ter"](#fill-terrain-using-fill_ter)
- * [ASCII map using "rows" array](#ascii-map-using-rows-array)
- * [Row terrains in "terrain"](#row-terrains-in-terrain)
- * [Furniture symbols in "furniture" array](#furniture-symbols-in-furniture-array)
- * [Set terrain, furniture, or traps with a "set" array](#set-terrain-furniture-or-traps-with-a-set-array)
- * [Set things at a "point"](#set-things-at-a-point)
- * [Set things in a "line"](#set-things-in-a-line)
- * [Set things in a "square"](#set-things-in-a-square)
- * [Spawn item or monster groups with "place_groups"](#spawn-item-or-monster-groups-with-place_groups)
- * [Spawn monsters from a group with "monster"](#spawn-monsters-from-a-group-with-monster)
- * [Spawn items from a group with "item"](#spawn-items-from-a-group-with-item)
- * [Spawn a single monster with "place_monster"](#spawn-a-single-monster-with-place_monster)
- * [Spawn an entire group of monsters with "place_monsters"](#spawn-an-entire-group-of-monsters-with-place_monsters)
- * [Spawn specific items with a "place_item" array](#spawn-specific-items-with-a-place_item-array)
- * [Extra map features with specials](#extra-map-features-with-specials)
- * [Place smoke, gas, or blood with "fields"](#place-smoke-gas-or-blood-with-fields)
- * [Place NPCs with "npcs"](#place-npcs-with-npcs)
- * [Place signs with "signs"](#place-signs-with-signs)
- * [Place a vending machine and items with "vendingmachines"](#place-a-vending-machine-and-items-with-vendingmachines)
- * [Place a toilet with some amount of water with "toilets"](#place-a-toilet-with-some-amount-of-water-with-toilets)
- * [Place a gas or diesel pump with some fuel with "gaspumps"](#place-a-gas-or-diesel-pump-with-some-fuel-with-gaspumps)
- * [Place items from an item group with "items"](#place-items-from-an-item-group-with-items)
- * [Place monsters from a monster group with "monsters"](#place-monsters-from-a-monster-group-with-monsters)
- * [Place a vehicle by type or group with "vehicles"](#place-a-vehicle-by-type-or-group-with-vehicles)
- * [Place a specific item with "item"](#place-a-specific-item-with-item)
- * [Place a specific monster with "monster"](#place-a-specific-monster-with-monster)
- * [Place a trap with "traps"](#place-a-trap-with-traps)
- * [Place furniture with "furniture"](#place-furniture-with-furniture)
- * [Place terrain with "terrain"](#place-terrain-with-terrain)
- * [Place rubble and smash existing terrain with "rubble"](#place-rubble-and-smash-existing-terrain-with-rubble)
- * [Place spilled liquids with "place_liquids"](#place-spilled-liquids-with-place_liquids)
- * [Place a specific item or an item from a group with "loot"](#place-a-specific-item-or-an-item-from-a-group-with-loot)
- * [Plant seeds in a planter with "sealed_item"](#plant-seeds-in-a-planter-with-sealed_item)
- * [Place messages with "graffiti"](#place-messages-with-graffiti)
- * [Place a zone for an NPC faction with "zones"](#place-a-zone-for-an-npc-faction-with-zones)
- * [Translate terrain type with "translate_ter"](#translate-terrain-type-with-translate_ter)
- * [Apply mapgen transformation with "ter_furn_transforms"](#apply-mapgen-transformation-with-ter_furn_transforms)
- * [Rotate the map with "rotation"](#rotate-the-map-with-rotation)
- * [Pre-load a base mapgen with "predecessor_mapgen"](#pre-load-a-base-mapgen-with-predecessor_mapgen)
-* [Using update_mapgen](#using-update_mapgen)
- * [Overmap tile specification](#overmap-tile-specification)
- * ["assign_mission_target"](#assign_mission_target)
- * ["om_terrain"](#om_terrain)
-* [Mission specials](#mission-specials)
- * ["target"](#target)
+- [How buildings and terrain are generated](#how-buildings-and-terrain-are-generated)
+- [Adding mapgen entries](#adding-mapgen-entries)
+ - [Methods](#methods)
+ - [Mapgen definition Placement](#mapgen-definition-placement)
+ - [Embedded mapgen](#embedded-mapgen)
+ - [Standalone mapgen](#standalone-mapgen)
+ - [Format and variables](#format-and-variables)
+ - [Define mapgen "method"](#define-mapgen-method)
+ - [Define overmap terrain with "om_terrain" value, array, or nested array](#define-overmap-terrain-with-om_terrain-value-array-or-nested-array)
+ - [Define mapgen "weight"](#define-mapgen-weight)
+ - [How "overmap_terrain" variables affect mapgen](#how-overmap_terrain-variables-affect-mapgen)
+ - [Limitations / TODO](#limitations--todo)
+- [JSON object definition](#json-object-definition)
+ - [Fill terrain using "fill_ter"](#fill-terrain-using-fill_ter)
+ - [ASCII map using "rows" array](#ascii-map-using-rows-array)
+ - [Row terrains in "terrain"](#row-terrains-in-terrain)
+ - [Furniture symbols in "furniture" array](#furniture-symbols-in-furniture-array)
+ - [Set terrain, furniture, or traps with a "set" array](#set-terrain-furniture-or-traps-with-a-set-array)
+ - [Set things at a "point"](#set-things-at-a-point)
+ - [Set things in a "line"](#set-things-in-a-line)
+ - [Set things in a "square"](#set-things-in-a-square)
+ - [Spawn item or monster groups with "place_groups"](#spawn-item-or-monster-groups-with-place_groups)
+ - [Spawn monsters from a group with "monster"](#spawn-monsters-from-a-group-with-monster)
+ - [Spawn items from a group with "item"](#spawn-items-from-a-group-with-item)
+ - [Spawn a single monster with "place_monster"](#spawn-a-single-monster-with-place_monster)
+ - [Spawn an entire group of monsters with "place_monsters"](#spawn-an-entire-group-of-monsters-with-place_monsters)
+ - [Spawn specific items with a "place_item" array](#spawn-specific-items-with-a-place_item-array)
+ - [Extra map features with specials](#extra-map-features-with-specials)
+ - [Place smoke, gas, or blood with "fields"](#place-smoke-gas-or-blood-with-fields)
+ - [Place NPCs with "npcs"](#place-npcs-with-npcs)
+ - [Place signs with "signs"](#place-signs-with-signs)
+ - [Place a vending machine and items with "vendingmachines"](#place-a-vending-machine-and-items-with-vendingmachines)
+ - [Place a toilet with some amount of water with "toilets"](#place-a-toilet-with-some-amount-of-water-with-toilets)
+ - [Place a gas or diesel pump with some fuel with "gaspumps"](#place-a-gas-or-diesel-pump-with-some-fuel-with-gaspumps)
+ - [Place items from an item group with "items"](#place-items-from-an-item-group-with-items)
+ - [Place monsters from a monster group with "monsters"](#place-monsters-from-a-monster-group-with-monsters)
+ - [Place a vehicle by type or group with "vehicles"](#place-a-vehicle-by-type-or-group-with-vehicles)
+ - [Place a specific item with "item"](#place-a-specific-item-with-item)
+ - [Place a specific monster with "monster"](#place-a-specific-monster-with-monster)
+ - [Place a trap with "traps"](#place-a-trap-with-traps)
+ - [Place furniture with "furniture"](#place-furniture-with-furniture)
+ - [Place terrain with "terrain"](#place-terrain-with-terrain)
+ - [Place rubble and smash existing terrain with "rubble"](#place-rubble-and-smash-existing-terrain-with-rubble)
+ - [Place spilled liquids with "place_liquids"](#place-spilled-liquids-with-place_liquids)
+ - [Place a specific item or an item from a group with "loot"](#place-a-specific-item-or-an-item-from-a-group-with-loot)
+ - [Plant seeds in a planter with "sealed_item"](#plant-seeds-in-a-planter-with-sealed_item)
+ - [Place messages with "graffiti"](#place-messages-with-graffiti)
+ - [Place a zone for an NPC faction with "zones"](#place-a-zone-for-an-npc-faction-with-zones)
+ - [Translate terrain type with "translate_ter"](#translate-terrain-type-with-translate_ter)
+ - [Apply mapgen transformation with "ter_furn_transforms"](#apply-mapgen-transformation-with-ter_furn_transforms)
+ - [Rotate the map with "rotation"](#rotate-the-map-with-rotation)
+ - [Pre-load a base mapgen with "predecessor_mapgen"](#pre-load-a-base-mapgen-with-predecessor_mapgen)
+- [Using update_mapgen](#using-update_mapgen)
+ - [Overmap tile specification](#overmap-tile-specification)
+ - ["assign_mission_target"](#assign_mission_target)
+ - ["om_terrain"](#om_terrain)
+- [Mission specials](#mission-specials)
+ - ["target"](#target)
# How buildings and terrain are generated
-Cataclysm creates buildings and terrain on discovery via 'mapgen'; functions specific to an overmap terrain (the tiles
-you see in `[m]`ap are also determined by overmap terrain). Overmap terrains ("oter") are defined in
-`overmap_terrain.json`.
+Cataclysm creates buildings and terrain on discovery via 'mapgen'; functions specific to an overmap
+terrain (the tiles you see in `[m]`ap are also determined by overmap terrain). Overmap terrains
+("oter") are defined in `overmap_terrain.json`.
-By default, an oter has a single built-in mapgen function which matches the '"id"' in it's json entry (examples:
-"house", "bank", etc). Multiple functions also possible. When a player moves into range of an area marked on the map as
-a house, the game chooses semi-randomly from a list of functions for "house", picks one, and runs it, laying down walls
-and adding items, monsters, rubber chickens and whatnot. This is all done in a fraction of a second (something to keep
-in mind for later).
+By default, an oter has a single built-in mapgen function which matches the '"id"' in it's json
+entry (examples: "house", "bank", etc). Multiple functions also possible. When a player moves into
+range of an area marked on the map as a house, the game chooses semi-randomly from a list of
+functions for "house", picks one, and runs it, laying down walls and adding items, monsters, rubber
+chickens and whatnot. This is all done in a fraction of a second (something to keep in mind for
+later).
-All mapgen functions build in a 24x24 tile area - even for large buildings; obtuse but surprisingly effective methods
-are used to assemble giant 3x3 hotels, etc..
+All mapgen functions build in a 24x24 tile area - even for large buildings; obtuse but surprisingly
+effective methods are used to assemble giant 3x3 hotels, etc..
-In order to make a world that's random and (somewhat) sensical, there are numerous rules and exceptions to them, which
-are clarified below.
+In order to make a world that's random and (somewhat) sensical, there are numerous rules and
+exceptions to them, which are clarified below.
There are three methods:
@@ -83,33 +84,33 @@ There are three methods:
- JSON object definition
- Using update_mapgen
-
# Adding mapgen entries
-One doesn't need to create a new `overmap_terrain` for a new variation of a building. For a custom gas station, defining a
-mapgen entry and adding it to the "s_gas" mapgen list will add it to the random variations of gas station in the world.
-
-If you use an existing `overmap_terrain` and it has a roof or other z-level linked to its file, the other levels will be
-generated with the ground floor. To avoid this, or add your own multiple z-levels, create an `overmap_terrain` with a
-similar name (`s_gas_1`).
+One doesn't need to create a new `overmap_terrain` for a new variation of a building. For a custom
+gas station, defining a mapgen entry and adding it to the "s_gas" mapgen list will add it to the
+random variations of gas station in the world.
+If you use an existing `overmap_terrain` and it has a roof or other z-level linked to its file, the
+other levels will be generated with the ground floor. To avoid this, or add your own multiple
+z-levels, create an `overmap_terrain` with a similar name (`s_gas_1`).
## Methods
-While adding mapgen as a c++ function is one of the fastest (and the most versatile) ways to generate procedural terrain
-on the fly, this requires recompiling the game.
-
-Most of the existing c++ buildings have been moved to json and currently json mapping is the preferred method of adding
-both content and mods.
+While adding mapgen as a c++ function is one of the fastest (and the most versatile) ways to
+generate procedural terrain on the fly, this requires recompiling the game.
-* JSON: A set of json arrays and objects for defining stuff and things. Pros: Fastest to apply, mostly complete. Cons:
- Not a programming language; no if statements or variables means instances of a particular json mapgen definition
- will all be similar. Third party map editors are currently out of date.
+Most of the existing c++ buildings have been moved to json and currently json mapping is the
+preferred method of adding both content and mods.
-* JSON support includes the use of nested mapgen, smaller mapgen chunks which override a portion of the linked mapgen.
- This allows for greater variety in furniture, terrain and spawns within a single mapgen file. You can also link
- mapgen files for multiple z-level buildings and multi-tile buildings.
+- JSON: A set of json arrays and objects for defining stuff and things. Pros: Fastest to apply,
+ mostly complete. Cons: Not a programming language; no if statements or variables means instances
+ of a particular json mapgen definition will all be similar. Third party map editors are currently
+ out of date.
+- JSON support includes the use of nested mapgen, smaller mapgen chunks which override a portion of
+ the linked mapgen. This allows for greater variety in furniture, terrain and spawns within a
+ single mapgen file. You can also link mapgen files for multiple z-level buildings and multi-tile
+ buildings.
## Mapgen definition Placement
@@ -125,42 +126,40 @@ As `"mapgen": { ... }` only used in combination with the 'builtin' method:
Do not use this, use standalone instead.
-
### Standalone mapgen
-As standalone `{ "type": "mapgen", ... }` objects in a .json inside data/json. Below is the fast food restaurant.
+As standalone `{ "type": "mapgen", ... }` objects in a .json inside data/json. Below is the fast
+food restaurant.
```json
[
- {
- "type": "mapgen",
- "om_terrain": "s_restaurant_fast",
- "weight": 250,
- "method": "json",
- "object": {
- "//": "(see below)"
- }
+ {
+ "type": "mapgen",
+ "om_terrain": "s_restaurant_fast",
+ "weight": 250,
+ "method": "json",
+ "object": {
+ "//": "(see below)"
}
+ }
]
```
-Note how "om_terrain" matches the overmap "id". om_terrain is **required** for standalone mapgen entries.
-
+Note how "om_terrain" matches the overmap "id". om_terrain is **required** for standalone mapgen
+entries.
## Format and variables
-The above example only illustrate the mapgen entries, not the actual format for building stuff. However, the following
-variables impact where and how often stuff gets applied:
+The above example only illustrate the mapgen entries, not the actual format for building stuff.
+However, the following variables impact where and how often stuff gets applied:
- method
- om_terrain
- weight
-
### Define mapgen "method"
-**required**
-Values: *json* - required
+**required** Values: _json_ - required
```
"object": { (more json here) }
@@ -170,8 +169,8 @@ Values: *json* - required
**required for standalone**
-The `om_terrain` value may be declared in one of three forms: with a single overmap terrain ID, with a list of IDs, or
-with a nested list (of lists) of IDs.
+The `om_terrain` value may be declared in one of three forms: with a single overmap terrain ID, with
+a list of IDs, or with a nested list (of lists) of IDs.
With the first form, simply give the ID of an overmap terrain from `overmap_terrain.json`:
@@ -185,7 +184,8 @@ In the second form, provide a list of IDs:
"om_terrain": [ "house", "house_base" ]
```
-This creates duplicate overmap terrains by applying the same json mapgen to each of the listed overmap terrain IDs.
+This creates duplicate overmap terrains by applying the same json mapgen to each of the listed
+overmap terrain IDs.
The third option is a nested list:
@@ -193,18 +193,20 @@ The third option is a nested list:
"om_terrain": [ [ "oter_id_1a", "oter_id_1b", ... ], [ "oter_id_2a", "oter_id_2b", ... ], ... ]
```
-This form allows for multiple overmap terrains to be defined using a single json object, with the "rows" property
-expanding in blocks of 24x24 characters to accommodate as many overmap terrains as are listed here. The terrain ids are
-specified using a nested array of strings which represent the rows and columns of overmap terrain ids (found in
-`overmap_terrain.json`) that are associated with the "rows" property described in section 2.1 of this document.
+This form allows for multiple overmap terrains to be defined using a single json object, with the
+"rows" property expanding in blocks of 24x24 characters to accommodate as many overmap terrains as
+are listed here. The terrain ids are specified using a nested array of strings which represent the
+rows and columns of overmap terrain ids (found in `overmap_terrain.json`) that are associated with
+the "rows" property described in section 2.1 of this document.
-Characters mapped using the "terrain", "furniture", or any of the special mappings ("items", "monsters", etc) will be
-applied universally to all of the listed overmap terrains.
+Characters mapped using the "terrain", "furniture", or any of the special mappings ("items",
+"monsters", etc) will be applied universally to all of the listed overmap terrains.
-Placing things using x/y coordinates ("place_monsters", "place_loot", "place_item", etc) works using the full extended
-coordinates beyond 24x24. An important limitation is that ranged random coordinates (such as "x": `[ 10, 18 ]`) must not
-cross the 24x24 terrain boundaries. Ranges such as `[ 0, 23 ]` and `[ 50, 70 ]` are valid, but `[ 0, 47 ]` and
-`[ 15, 35 ]` are not because they extend beyond a single 24x24 block.
+Placing things using x/y coordinates ("place_monsters", "place_loot", "place_item", etc) works using
+the full extended coordinates beyond 24x24. An important limitation is that ranged random
+coordinates (such as "x": `[ 10, 18 ]`) must not cross the 24x24 terrain boundaries. Ranges such as
+`[ 0, 23 ]` and `[ 50, 70 ]` are valid, but `[ 0, 47 ]` and `[ 15, 35 ]` are not because they extend
+beyond a single 24x24 block.
Example:
@@ -215,58 +217,56 @@ Example:
]
```
-In this example, the "rows" property should be 48x48, with each quadrant of 24x24 being associated with each of the four
-apartments_mod_tower overmap terrain ids specified.
-
+In this example, the "rows" property should be 48x48, with each quadrant of 24x24 being associated
+with each of the four apartments_mod_tower overmap terrain ids specified.
### Define mapgen "weight"
-(optional) When the game randomly picks mapgen functions, each function's weight value determines how rare it is. 1000
-is the default, so adding something with weight '500' will make it appear 1/3 times, unless more functions are added.
-(An insanely high value like 10000000 is useful for testing)
+(optional) When the game randomly picks mapgen functions, each function's weight value determines
+how rare it is. 1000 is the default, so adding something with weight '500' will make it appear 1/3
+times, unless more functions are added. (An insanely high value like 10000000 is useful for testing)
-Values: number - *0 disables*
+Values: number - _0 disables_
Default: 1000
-
## How "overmap_terrain" variables affect mapgen
-"id" is used to determine the required "om_terrain" id for standalone, *except* when the following variables are set in
-"overmap_terrain":
+"id" is used to determine the required "om_terrain" id for standalone, _except_ when the following
+variables are set in "overmap_terrain":
-* "extras" - applies rare, random scenes after mapgen; helicopter crashes, etc
-* "mondensity" - determines the default 'density' value for `"place_groups": [ { "monster": ...` (json). If this is not
- set then place_monsters will not work without its own explicitly set density argument.
+- "extras" - applies rare, random scenes after mapgen; helicopter crashes, etc
+- "mondensity" - determines the default 'density' value for `"place_groups": [ { "monster": ...`
+ (json). If this is not set then place_monsters will not work without its own explicitly set
+ density argument.
## Limitations / TODO
-* JSON: adding specific monster spawns are still WIP.
-* The old mapgen.cpp system involved *The Biggest "if / else if / else if / .." Statement Known to Man*(tm), and is only
- halfway converted to the "builtin" mapgen class. This means that while custom mapgen functions are allowed, the game
- will cheerfully forget the default if one is added.
-* TODO: Add to this list.
-
+- JSON: adding specific monster spawns are still WIP.
+- The old mapgen.cpp system involved _The Biggest "if / else if / else if / .." Statement Known to
+ Man_(tm), and is only halfway converted to the "builtin" mapgen class. This means that while
+ custom mapgen functions are allowed, the game will cheerfully forget the default if one is added.
+- TODO: Add to this list.
# JSON object definition
-The JSON object for a mapgen entry must include either `"fill_ter"`, or `"rows"` and `"terrain"`. All other fields are
-optional.
-
+The JSON object for a mapgen entry must include either `"fill_ter"`, or `"rows"` and `"terrain"`.
+All other fields are optional.
## Fill terrain using "fill_ter"
-*required if "rows" is unset* Fill with the given terrain.
+
+_required if "rows" is unset_ Fill with the given terrain.
Value: `"string"`: Valid terrain id from data/json/terrain.json
Example: `"fill_ter": "t_grass"`
-
## ASCII map using "rows" array
-*required if "fill_ter" is unset*
-Nested array of 24 (or 48) strings, each 24 (or 48) characters long, where each character is defined by "terrain" and
-optionally "furniture" or other entries below.
+_required if "fill_ter" is unset_
+
+Nested array of 24 (or 48) strings, each 24 (or 48) characters long, where each character is defined
+by "terrain" and optionally "furniture" or other entries below.
Usage:
@@ -274,22 +274,24 @@ Usage:
"rows": [ "row1...", "row2...", ..., "row24..." ]
```
-Other parts can be linked with this map, for example one can place things like a gaspump (with gasoline) or a toilet
-(with water) or items from an item group or fields at the square given by a character.
-
-Any character used here must have some definition elsewhere to indicate its purpose. Failing to do so is an error which
-will be caught by running the tests. The tests will run automatically when you make a pull request for adding new maps
-to the game. If you have defined `fill_ter` or you are writing nested mapgen, then there are a couple of exceptions.
-The space and period characters (` ` and `.`) are permitted to have no definition and be used for 'background' in the
-`rows`.
-
-As keys, you can use any Unicode characters which are not double-width. This includes for example most European
-alphabets but not Chinese characters. If you intend to take advantage of this, ensure that your editor is saving the
-file with a UTF-8 encoding. Accents are acceptable, even when using [combining
-characters](https://en.wikipedia.org/wiki/Combining_character). No normalization is performed; comparison is done at
-the raw bytes (code unit) level. Therefore, there are literally an infinite number of mapgen key characters available.
-Please don't abuse this by using distinct characters that are visually indistinguishable, or which are so rare as to be
-unlikely to render correctly for other developers.
+Other parts can be linked with this map, for example one can place things like a gaspump (with
+gasoline) or a toilet (with water) or items from an item group or fields at the square given by a
+character.
+
+Any character used here must have some definition elsewhere to indicate its purpose. Failing to do
+so is an error which will be caught by running the tests. The tests will run automatically when you
+make a pull request for adding new maps to the game. If you have defined `fill_ter` or you are
+writing nested mapgen, then there are a couple of exceptions. The space and period characters (``
+and `.`) are permitted to have no definition and be used for 'background' in the `rows`.
+
+As keys, you can use any Unicode characters which are not double-width. This includes for example
+most European alphabets but not Chinese characters. If you intend to take advantage of this, ensure
+that your editor is saving the file with a UTF-8 encoding. Accents are acceptable, even when using
+[combining characters](https://en.wikipedia.org/wiki/Combining_character). No normalization is
+performed; comparison is done at the raw bytes (code unit) level. Therefore, there are literally an
+infinite number of mapgen key characters available. Please don't abuse this by using distinct
+characters that are visually indistinguishable, or which are so rare as to be unlikely to render
+correctly for other developers.
Example:
@@ -320,10 +322,10 @@ Example:
" ,,,,,,,,,,,,,,,,,,,, ",
" dd "
],
-
```
### Row terrains in "terrain"
+
**required by "rows"**
Defines terrain ids for "rows", each key is a single character with a terrain id string
@@ -366,10 +368,11 @@ Example:
```
### Furniture symbols in "furniture" array
+
**optional**
-Defines furniture ids for "rows" ( each character in rows is a terrain -or- terrain/furniture combo ). "f_null" means no
-furniture but the entry can be left out
+Defines furniture ids for "rows" ( each character in rows is a terrain -or- terrain/furniture combo
+). "f_null" means no furniture but the entry can be left out
Example:
@@ -392,7 +395,9 @@ Example:
```
## Set terrain, furniture, or traps with a "set" array
-**optional** Specific commands to set terrain, furniture, traps, radiation, etc. Array is processed in order.
+
+**optional** Specific commands to set terrain, furniture, traps, radiation, etc. Array is processed
+in order.
Value: `[ array of {objects} ]: [ { "point": .. }, { "line": .. }, { "square": .. }, ... ]`
@@ -406,29 +411,28 @@ Example:
]
```
-All X and Y values may be either a single integer between `0` and `23`, or an array of two integers `[ n1, n2 ]` (each
-between `0` and `23`). If X or Y are set to an array, the result is a random number in that range (inclusive). In the above
-examples, the furniture `"f_chair"` is always at coordinates `"x": 5, "y": 10`, but the trap `"tr_beartrap"` is
-randomly repeated in the area `"x": [ 0, 23 ], "y": [ 5, 18 ]"`.
+All X and Y values may be either a single integer between `0` and `23`, or an array of two integers
+`[ n1, n2 ]` (each between `0` and `23`). If X or Y are set to an array, the result is a random
+number in that range (inclusive). In the above examples, the furniture `"f_chair"` is always at
+coordinates `"x": 5, "y": 10`, but the trap `"tr_beartrap"` is randomly repeated in the area
+`"x": [ 0, 23 ], "y": [ 5, 18 ]"`.
See terrain.json, furniture.json, and trap.json for "id" strings.
-
### Set things at a "point"
- Requires "point" type, and coordinates "x" and "y"
- For "point" type "radiation", requires "amount"
- For other types, requires "id" of terrain, furniture, or trap
-| Field | Description
-| --- | ---
-| point | Allowed values: `"terrain"`, `"furniture"`, `"trap"`, `"radiation"`
-| id | Terrain, furniture, or trap ID. Examples: `"id": "f_counter"`, `"id": "tr_beartrap"`. Omit for "radiation".
-| x, y | X, Y coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. Example: `"x": 12, "y": [ 5, 15 ]`
-| amount | Radiation amount. Value from `0-100`.
-| chance | (optional) One-in-N chance to apply
-| repeat | (optional) Value: `[ n1, n2 ]`. Spawn item randomly between `n1` and `n2` times. Only makes sense if the coordinates are random. Example: `[ 1, 3 ]` - repeat 1-3 times.
-
+| Field | Description |
+| ------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
+| point | Allowed values: `"terrain"`, `"furniture"`, `"trap"`, `"radiation"` |
+| id | Terrain, furniture, or trap ID. Examples: `"id": "f_counter"`, `"id": "tr_beartrap"`. Omit for "radiation". |
+| x, y | X, Y coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. Example: `"x": 12, "y": [ 5, 15 ]` |
+| amount | Radiation amount. Value from `0-100`. |
+| chance | (optional) One-in-N chance to apply |
+| repeat | (optional) Value: `[ n1, n2 ]`. Spawn item randomly between `n1` and `n2` times. Only makes sense if the coordinates are random. Example: `[ 1, 3 ]` - repeat 1-3 times. |
### Set things in a "line"
@@ -437,20 +441,20 @@ See terrain.json, furniture.json, and trap.json for "id" strings.
- For other types, requires "id" of terrain, furniture, or trap
Example:
+
```json
{ "line": "terrain", "id": "t_lava", "x": 5, "y": 5, "x2": 20, "y2": 20 }
```
-| Field | Description
-| --- | ---
-| line | Allowed values: `"terrain"`, `"furniture"`, `"trap"`, `"radiation"`
-| id | Terrain, furniture, or trap ID. Examples: `"id": "f_counter"`, `"id": "tr_beartrap"`. Omit for "radiation".
-| x, y | Start X, Y coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. Example: `"x": 12, "y": [ 5, 15 ]`
-| x2, y2 | End X, Y coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. Example: `"x": 22, "y": [ 15, 20 ]`
-| amount | Radiation amount. Value from `0-100`.
-| chance | (optional) One-in-N chance to apply
-| repeat | (optional) Value: `[ n1, n2 ]`. Spawn item randomly between `n1` and `n2` times. Only makes sense if the coordinates are random. Example: `[ 1, 3 ]` - repeat 1-3 times.
-
+| Field | Description |
+| ------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
+| line | Allowed values: `"terrain"`, `"furniture"`, `"trap"`, `"radiation"` |
+| id | Terrain, furniture, or trap ID. Examples: `"id": "f_counter"`, `"id": "tr_beartrap"`. Omit for "radiation". |
+| x, y | Start X, Y coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. Example: `"x": 12, "y": [ 5, 15 ]` |
+| x2, y2 | End X, Y coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. Example: `"x": 22, "y": [ 15, 20 ]` |
+| amount | Radiation amount. Value from `0-100`. |
+| chance | (optional) One-in-N chance to apply |
+| repeat | (optional) Value: `[ n1, n2 ]`. Spawn item randomly between `n1` and `n2` times. Only makes sense if the coordinates are random. Example: `[ 1, 3 ]` - repeat 1-3 times. |
### Set things in a "square"
@@ -458,160 +462,169 @@ Example:
- For "square" type "radiation", requires "amount"
- For other types, requires "id" of terrain, furniture, or trap
-The "square" arguments are the same as for "line", but "x", "y" and "x2", "y2" define opposite corners.
+The "square" arguments are the same as for "line", but "x", "y" and "x2", "y2" define opposite
+corners.
Example:
+
```json
-{ "square": "radiation", "amount": 10, "x": [ 0, 5 ], "y": [ 0, 5 ], "x2": [ 18, 23 ], "y2": [ 18, 23 ] }
+{ "square": "radiation", "amount": 10, "x": [0, 5], "y": [0, 5], "x2": [18, 23], "y2": [18, 23] }
```
-| Field | Description
-| --- | ---
-| square | Allowed values: `"terrain"`, `"furniture"`, `"trap"`, `"radiation"`
-| id | Terrain, furniture, or trap ID. Examples: `"id": "f_counter"`, `"id": "tr_beartrap"`. Omit for "radiation".
-| x, y | Top-left corner of square.
-| x2, y2 | Bottom-right corner of square.
-
+| Field | Description |
+| ------ | ----------------------------------------------------------------------------------------------------------- |
+| square | Allowed values: `"terrain"`, `"furniture"`, `"trap"`, `"radiation"` |
+| id | Terrain, furniture, or trap ID. Examples: `"id": "f_counter"`, `"id": "tr_beartrap"`. Omit for "radiation". |
+| x, y | Top-left corner of square. |
+| x2, y2 | Bottom-right corner of square. |
## Spawn item or monster groups with "place_groups"
+
**optional** Spawn items or monsters from item_groups.json and monster_groups.json
Value: `[ array of {objects} ]: [ { "monster": ... }, { "item": ... }, ... ]`
### Spawn monsters from a group with "monster"
-| Field | Description
-| --- | ---
-| monster | (required) Value: `"MONSTER_GROUP"`. The monster group id, which picks random critters from a list
-| x, y | (required) Spawn coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range.
-| density | (optional) Floating-point multiplier to "chance" (see below).
-| chance | (optional) One-in-N chance to spawn
+| Field | Description |
+| ------- | ------------------------------------------------------------------------------------------------------------ |
+| monster | (required) Value: `"MONSTER_GROUP"`. The monster group id, which picks random critters from a list |
+| x, y | (required) Spawn coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. |
+| density | (optional) Floating-point multiplier to "chance" (see below). |
+| chance | (optional) One-in-N chance to spawn |
Example:
```json
-{ "monster": "GROUP_ZOMBIE", "x": [ 13, 15 ], "y": 15, "chance": 10 }
+{ "monster": "GROUP_ZOMBIE", "x": [13, 15], "y": 15, "chance": 10 }
```
-When using a range for `"x"` or `"y"`, the minimum and maximum values will be used in creating rectangle coordinates to
-be used by `map::place_spawns`. Each monster generated from the monster group will be placed in a different random
-location within the rectangle. The values in the above example will produce a rectangle for `map::place_spawns` from (
-13, 15 ) to ( 15, 15 ) inclusive.
-
-The optional "density" is a floating-point multipier to the "chance" value. If the result is bigger than 100% it
-gurantees one spawn point for every 100% and the rest is evaluated by chance (one added or not). Then the monsters are
-spawned according to their spawn-point cost "cost_multiplier" defined in the monster groups. Additionally all overmap
-densities within a square of raduis 3 (7x7 around player - exact value in mapgen.cpp/MON_RADIUS macro) are added to
-this. The "pack_size" modifier in monstergroups is a random multiplier to the rolled spawn point amount.
-
+When using a range for `"x"` or `"y"`, the minimum and maximum values will be used in creating
+rectangle coordinates to be used by `map::place_spawns`. Each monster generated from the monster
+group will be placed in a different random location within the rectangle. The values in the above
+example will produce a rectangle for `map::place_spawns` from ( 13, 15 ) to ( 15, 15 ) inclusive.
+The optional "density" is a floating-point multipier to the "chance" value. If the result is bigger
+than 100% it gurantees one spawn point for every 100% and the rest is evaluated by chance (one added
+or not). Then the monsters are spawned according to their spawn-point cost "cost_multiplier" defined
+in the monster groups. Additionally all overmap densities within a square of raduis 3 (7x7 around
+player - exact value in mapgen.cpp/MON_RADIUS macro) are added to this. The "pack_size" modifier in
+monstergroups is a random multiplier to the rolled spawn point amount.
### Spawn items from a group with "item"
-| Field | Description
-| --- | ---
-| item | (required) Value: "ITEM_GROUP". The item group id, which picks random stuff from a list.
-| x, y | (required) Spawn coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range.
-| chance | (required) Percentage chance to spawn.
+| Field | Description |
+| ------ | ------------------------------------------------------------------------------------------------------------ |
+| item | (required) Value: "ITEM_GROUP". The item group id, which picks random stuff from a list. |
+| x, y | (required) Spawn coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. |
+| chance | (required) Percentage chance to spawn. |
Example:
```json
-{ "item": "livingroom", "x": 12, "y": [ 5, 15 ], "chance": 50 }
+{ "item": "livingroom", "x": 12, "y": [5, 15], "chance": 50 }
```
-When using a range for `"x"` or `"y"`, the minimum and maximum values will be used in creating rectangle coordinates to
-be used by `map::place_items`. Each item from the item group will be placed in a different random location within the
-rectangle. These values in the above example will produce a rectangle for map::place_items from ( 12, 5 ) to ( 12, 15 )
-inclusive.
-
+When using a range for `"x"` or `"y"`, the minimum and maximum values will be used in creating
+rectangle coordinates to be used by `map::place_items`. Each item from the item group will be placed
+in a different random location within the rectangle. These values in the above example will produce
+a rectangle for map::place_items from ( 12, 5 ) to ( 12, 15 ) inclusive.
## Spawn a single monster with "place_monster"
-**optional** Spawn single monster. Either specific monster or a random monster from a monster group. Is affected by spawn density game setting.
+**optional** Spawn single monster. Either specific monster or a random monster from a monster group.
+Is affected by spawn density game setting.
Value: `[ array of {objects} ]: [ { "monster": ... } ]`
-| Field | Description
-| --- | ---
-| monster | ID of the monster to spawn.
-| group | ID of the monster group from which the spawned monster is selected. `monster` and `group` should not be used together. `group` will act over `monster`.
-| x, y | Spawn coordinates ( specific or area rectangle ). Value: 0-23 or `[ 0-23, 0-23 ]` - random value between `[ a, b ]`.
-| chance | Percentage chance to do spawning. If repeat is used each repeat has separate chance.
-| repeat | The spawning is repeated this many times. Can be a number or a range.
-| pack_size | How many monsters are spawned. Can be single number or range like `[1-4]`. Is affected by the chance and spawn density. Ignored when spawning from a group.
-| one_or_none | Do not allow more than one to spawn due to high spawn density. If repeat is not defined or pack size is defined this defaults to true true, otherwise this defaults to false. Ignored when spawning from a group.
-| friendly | Set true to make the monster friendly. Default false.
-| name | Extra name to display on the monster.
-| target | Set to true to make this into mission target. Only works when the monster is spawned from a mission.
-
-Note that high spawn density game setting can cause extra monsters to spawn when `monster` is used. When `group` is used
-only one monster will spawn.
-
-When using a range for `"x"` or `"y"`, the minimum and maximum values will be used in creating rectangle coordinates to
-be used by `map::place_spawns`. Each monster generated from the monster group will be placed in a different random
-location within the rectangle. Example: `"x": 12, "y": [ 5, 15 ]` - these values will produce a rectangle for
-`map::place_spawns` from ( 12, 5 ) to ( 12, 15 ) inclusive.
+| Field | Description |
+| ----------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| monster | ID of the monster to spawn. |
+| group | ID of the monster group from which the spawned monster is selected. `monster` and `group` should not be used together. `group` will act over `monster`. |
+| x, y | Spawn coordinates ( specific or area rectangle ). Value: 0-23 or `[ 0-23, 0-23 ]` - random value between `[ a, b ]`. |
+| chance | Percentage chance to do spawning. If repeat is used each repeat has separate chance. |
+| repeat | The spawning is repeated this many times. Can be a number or a range. |
+| pack_size | How many monsters are spawned. Can be single number or range like `[1-4]`. Is affected by the chance and spawn density. Ignored when spawning from a group. |
+| one_or_none | Do not allow more than one to spawn due to high spawn density. If repeat is not defined or pack size is defined this defaults to true true, otherwise this defaults to false. Ignored when spawning from a group. |
+| friendly | Set true to make the monster friendly. Default false. |
+| name | Extra name to display on the monster. |
+| target | Set to true to make this into mission target. Only works when the monster is spawned from a mission. |
+
+Note that high spawn density game setting can cause extra monsters to spawn when `monster` is used.
+When `group` is used only one monster will spawn.
+
+When using a range for `"x"` or `"y"`, the minimum and maximum values will be used in creating
+rectangle coordinates to be used by `map::place_spawns`. Each monster generated from the monster
+group will be placed in a different random location within the rectangle. Example:
+`"x": 12, "y": [ 5, 15 ]` - these values will produce a rectangle for `map::place_spawns` from ( 12,
+5 ) to ( 12, 15 ) inclusive.
Example:
+
```json
"place_monster": [
{ "group": "GROUP_REFUGEE_BOSS_ZOMBIE", "name": "Sean McLaughlin", "x": 10, "y": 10, "target": true }
]
```
-This places a single random monster from group "GROUP_REFUGEE_BOSS_ZOMBIE", sets the name to "Sean McLaughlin", spawns
-the monster at coordinate (10, 10) and also sets the monster as the target of this mission.
+This places a single random monster from group "GROUP_REFUGEE_BOSS_ZOMBIE", sets the name to "Sean
+McLaughlin", spawns the monster at coordinate (10, 10) and also sets the monster as the target of
+this mission.
Example:
+
```json
"place_monster": [
{ "monster": "mon_secubot", "x": [ 7, 18 ], "y": [ 7, 18 ], "chance": 30, "repeat": [1, 3] }
]
```
-This places "mon_secubot" at random coordinate (7-18, 7-18). The monster is placed with 30% probablity. The placement is
-repeated by random number of times `[1-3]`.
-
+This places "mon_secubot" at random coordinate (7-18, 7-18). The monster is placed with 30%
+probablity. The placement is repeated by random number of times `[1-3]`.
## Spawn an entire group of monsters with "place_monsters"
-Using `place_monsters` to spawn a group of monsters works in a similar fashion to `place_monster`. The key difference is that `place_monsters` guarantees that each valid entry in the group is spawned. It is strongly advised that you avoid using this flag with larger monster groups, as the total number of spawns is quite difficult to control.
-|Field|Description |
-|--|--|
-| monster | The ID of the monster group that you wish to spawn |
-| x, y | Spawn coordinates ( specific or area rectangle ). Value: 0-23 or `[ 0-23, 0-23 ]` - random value between `[ a, b ]`.
-| chance | Represents a 1 in N chance that the entire group will spawn. This is done once for each repeat. If this dice roll fails, the entire group specified will not spawn. Leave blank to guarantee spawns.
-| repeat | The spawning is repeated this many times. Can be a number or a range. Again, this represents the number of times the group will be spawned.
-| density | This number is multiplied by the spawn density of the world the player is in and then probabilistically rounded to determine how many times to spawn the group. This is done for each time the spawn is repeated. For instance, if the final multiplier from this calculation ends up being `2`, and the repeat value is `6`, then the group will be spawned `2 * 6` or 12 times.
+Using `place_monsters` to spawn a group of monsters works in a similar fashion to `place_monster`.
+The key difference is that `place_monsters` guarantees that each valid entry in the group is
+spawned. It is strongly advised that you avoid using this flag with larger monster groups, as the
+total number of spawns is quite difficult to control.
+| Field | Description |
+| ------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| monster | The ID of the monster group that you wish to spawn |
+| x, y | Spawn coordinates ( specific or area rectangle ). Value: 0-23 or `[ 0-23, 0-23 ]` - random value between `[ a, b ]`. |
+| chance | Represents a 1 in N chance that the entire group will spawn. This is done once for each repeat. If this dice roll fails, the entire group specified will not spawn. Leave blank to guarantee spawns. |
+| repeat | The spawning is repeated this many times. Can be a number or a range. Again, this represents the number of times the group will be spawned. |
+| density | This number is multiplied by the spawn density of the world the player is in and then probabilistically rounded to determine how many times to spawn the group. This is done for each time the spawn is repeated. For instance, if the final multiplier from this calculation ends up being `2`, and the repeat value is `6`, then the group will be spawned `2 * 6` or 12 times. |
## Spawn specific items with a "place_item" array
-**optional** A list of *specific* things to add. WIP: Monsters and vehicles will be here too
+
+**optional** A list of _specific_ things to add. WIP: Monsters and vehicles will be here too
Value: `[ array of {objects} ]: [ { "item", ... }, ... ]`
Example:
+
```json
"place_item": [
{ "item": "weed", "x": 14, "y": 15, "amount": [ 10, 20 ], "repeat": [1, 3], "chance": 20 }
]
```
-| Field | Description
-| --- | ---
-| item | (required) ID of the item to spawn
-| x, y | (required) Spawn coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range.
-| amount | (required) Number of items to spawn. Single integer, or range `[ a, b ]` for a random value in that range.
-| chance | (optional) One-in-N chance to spawn item.
-| repeat | (optional) Value: `[ n1, n2 ]`. Spawn item randomly between `n1` and `n2` times. Only makes sense if the coordinates are random. Example: `[ 1, 3 ]` - repeat 1-3 times.
-
+| Field | Description |
+| ------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
+| item | (required) ID of the item to spawn |
+| x, y | (required) Spawn coordinates. Value from `0-23`, or range `[ 0-23, 0-23 ]` for a random value in that range. |
+| amount | (required) Number of items to spawn. Single integer, or range `[ a, b ]` for a random value in that range. |
+| chance | (optional) One-in-N chance to spawn item. |
+| repeat | (optional) Value: `[ n1, n2 ]`. Spawn item randomly between `n1` and `n2` times. Only makes sense if the coordinates are random. Example: `[ 1, 3 ]` - repeat 1-3 times. |
## Extra map features with specials
+
**optional** Special map features that do more than just placing furniture / terrain.
-Specials can be defined either via a mapping like the terrain / furniture mapping using the "rows" entry above or
-through their exact location by its coordinates.
+Specials can be defined either via a mapping like the terrain / furniture mapping using the "rows"
+entry above or through their exact location by its coordinates.
The mapping is defined with a json object like this:
@@ -624,13 +637,13 @@ The mapping is defined with a json object like this:
}
```
-`""` is one of the types listed below. `` is a json object with content specific to
-the special type. Some types require no data at all or all their data is optional, an empty object is enough for those
-specials. You can define as many mapping as you want.
+`""` is one of the types listed below. `` is a json object with
+content specific to the special type. Some types require no data at all or all their data is
+optional, an empty object is enough for those specials. You can define as many mapping as you want.
-Each mapping can be an array, for things that can appear several times on the tile (e.g. items, fields) each entry of
-the array is applied in order. For traps, furniture and terrain, one entry is randomly chosen (all entries have the same
-chances) and applied.
+Each mapping can be an array, for things that can appear several times on the tile (e.g. items,
+fields) each entry of the array is applied in order. For traps, furniture and terrain, one entry is
+randomly chosen (all entries have the same chances) and applied.
Example (places grass at 2/3 of all '.' square and dirt at 1/3 of them):
@@ -640,8 +653,9 @@ Example (places grass at 2/3 of all '.' square and dirt at 1/3 of them):
}
```
-It is also possible to specify the number of instances (and consequently their chance) directly, which is particularly
-useful for rare occurrences (rather than repeating the common value many times):
+It is also possible to specify the number of instances (and consequently their chance) directly,
+which is particularly useful for rare occurrences (rather than repeating the common value many
+times):
```json
"terrain" : {
@@ -649,7 +663,6 @@ useful for rare occurrences (rather than repeating the common value many times):
}
```
-
Example (places a blood and a bile field on each '.' square):
```json
@@ -659,6 +672,7 @@ Example (places a blood and a bile field on each '.' square):
```
Or define the mappings for one character at once:
+
```json
"mapping" : {
".": {
@@ -669,9 +683,11 @@ Or define the mappings for one character at once:
}
}
```
+
This might be more useful if you want to put many different type of things on one place.
Defining specials through their specific location:
+
```
"place_" : {
{ "x": , "y": , },
@@ -679,10 +695,12 @@ Defining specials through their specific location:
}
```
-`` and `` define where the special is placed (x is horizontal, y vertical). Valid value are in the range 0..23,
-min-max values are also supported: `"x": [ 0, 23 ], "y": [ 0, 23 ]` places the special anyway on the map.
+`` and `` define where the special is placed (x is horizontal, y vertical). Valid value are in
+the range 0..23, min-max values are also supported: `"x": [ 0, 23 ], "y": [ 0, 23 ]` places the
+special anyway on the map.
-Example with mapping (the characters `"O"` and `";"` should appear in the rows array where the specials should appear):
+Example with mapping (the characters `"O"` and `";"` should appear in the rows array where the
+specials should appear):
```json
"gaspumps": {
@@ -693,7 +711,8 @@ Example with mapping (the characters `"O"` and `";"` should appear in the rows a
}
```
-The amount of water to be placed in toilets is optional, an empty entry is therefor completely valid.
+The amount of water to be placed in toilets is optional, an empty entry is therefor completely
+valid.
Example with coordinates:
@@ -707,12 +726,15 @@ Example with coordinates:
```
Terrain, furniture and traps can specified as a single string, not a json object:
+
```json
"traps" : {
".": "tr_beartrap"
}
```
+
Same as
+
```json
"traps" : {
".": { "trap": "tr_beartrap" }
@@ -721,13 +743,12 @@ Same as
### Place smoke, gas, or blood with "fields"
-| Field | Description
-| --- | ---
-| field | (required, string) the field type (e.g. `"fd_blood"`, `"fd_smoke"`)
-| density | (optional, integer) field density. Defaults to 1. Possible values are 1, 2, or 3.
-| intensity | (optional, integer) how concentrated the field is, from 1 to 3 or more. See `data/json/field_type.json`
-| age | (optional, integer) field age. Defaults to 0.
-
+| Field | Description |
+| --------- | ------------------------------------------------------------------------------------------------------- |
+| field | (required, string) the field type (e.g. `"fd_blood"`, `"fd_smoke"`) |
+| density | (optional, integer) field density. Defaults to 1. Possible values are 1, 2, or 3. |
+| intensity | (optional, integer) how concentrated the field is, from 1 to 3 or more. See `data/json/field_type.json` |
+| age | (optional, integer) field age. Defaults to 0. |
### Place NPCs with "npcs"
@@ -737,18 +758,17 @@ Example:
"npcs": { "A": { "class": "NC_REFUGEE", "target": true, "add_trait": "ASTHMA" } }
```
-| Field | Description
-| --- | ---
-| class | (required, string) the npc class id, see `data/json/npcs/npc.json` or define your own npc class.
-| target | (optional, bool) this NPC is a mission target. Only valid for `update_mapgen`.
-| add_trait | (optional, string or string array) this NPC gets these traits, in addition to any from the class definition.
-
+| Field | Description |
+| --------- | ------------------------------------------------------------------------------------------------------------ |
+| class | (required, string) the npc class id, see `data/json/npcs/npc.json` or define your own npc class. |
+| target | (optional, bool) this NPC is a mission target. Only valid for `update_mapgen`. |
+| add_trait | (optional, string or string array) this NPC gets these traits, in addition to any from the class definition. |
### Place signs with "signs"
-Places a sign (furniture `f_sign`) with a message written on it. Either "signage" or "snippet" must be defined. The
-message may include tags like ``, ``, and `` that will insert a randomly generated
-name, or `` that will insert the nearest city name.
+Places a sign (furniture `f_sign`) with a message written on it. Either "signage" or "snippet" must
+be defined. The message may include tags like ``, ``, and ``
+that will insert a randomly generated name, or `` that will insert the nearest city name.
Example:
@@ -756,81 +776,76 @@ Example:
"signs": { "P": { "signage": "Subway map: stop" } }
```
-| Field | Description
-| --- | ---
-| signage | (optional, string) the message that should appear on the sign.
-| snippet | (optional, string) a category of snippets that can appear on the sign.
-
+| Field | Description |
+| ------- | ---------------------------------------------------------------------- |
+| signage | (optional, string) the message that should appear on the sign. |
+| snippet | (optional, string) a category of snippets that can appear on the sign. |
### Place a vending machine and items with "vendingmachines"
-Places a vending machine (furniture) and fills it with items from an item group. The machine can sometimes spawn as broken one.
-
-| Field | Description
-| --- | ---
-| item_group | (optional, string) the item group that is used to create items inside the machine. It defaults to either "vending_food" or "vending_drink" (randomly chosen).
+Places a vending machine (furniture) and fills it with items from an item group. The machine can
+sometimes spawn as broken one.
+| Field | Description |
+| ---------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| item_group | (optional, string) the item group that is used to create items inside the machine. It defaults to either "vending_food" or "vending_drink" (randomly chosen). |
### Place a toilet with some amount of water with "toilets"
Places a toilet (furniture) and adds water to it.
-| Field | Description
-| --- | ---
-| amount | (optional, integer or min/max array) the amount of water to be placed in the toilet.
-
+| Field | Description |
+| ------ | ------------------------------------------------------------------------------------ |
+| amount | (optional, integer or min/max array) the amount of water to be placed in the toilet. |
### Place a gas or diesel pump with some fuel with "gaspumps"
Places a gas pump with gasoline (or sometimes diesel) in it.
-| Field | Description
-| --- | ---
-| amount | (optional, integer or min/max array) the amount of fuel to be placed in the pump.
-| fuel | (optional, string: "gasoline" or "diesel") the type of fuel to be placed in the pump.
-
+| Field | Description |
+| ------ | ------------------------------------------------------------------------------------- |
+| amount | (optional, integer or min/max array) the amount of fuel to be placed in the pump. |
+| fuel | (optional, string: "gasoline" or "diesel") the type of fuel to be placed in the pump. |
### Place items from an item group with "items"
-| Field | Description
-| --- | ---
-| item | (required, string or itemgroup object) the item group to use.
-| chance | (optional, integer or min/max array) x in 100 chance that a loop will continue to spawn items from the group (which itself may spawn multiple items or not depending on its type, see `ITEM_SPAWN.md`), unless the chance is 100, in which case it will trigger the item group spawn exactly 1 time (see `map::place_items`).
-| repeat | (optional, integer or min/max array) the number of times to repeat this placement, default is 1.
-
+| Field | Description |
+| ------ | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| item | (required, string or itemgroup object) the item group to use. |
+| chance | (optional, integer or min/max array) x in 100 chance that a loop will continue to spawn items from the group (which itself may spawn multiple items or not depending on its type, see `ITEM_SPAWN.md`), unless the chance is 100, in which case it will trigger the item group spawn exactly 1 time (see `map::place_items`). |
+| repeat | (optional, integer or min/max array) the number of times to repeat this placement, default is 1. |
### Place monsters from a monster group with "monsters"
The actual monsters are spawned when the map is loaded. Fields:
-| Field | Description
-| --- | ---
-| monster | (required, string) a monster group id, when the map is loaded, a random monsters from that group are spawned.
-| density | (optional, float) if defined, it overrides the default monster density at the location (monster density is bigger towards the city centers) (see `map::place_spawns`).
-| chance | (optional, integer or min/max array) one in x chance of spawn point being created (see `map::place_spawns`).
-
+| Field | Description |
+| ------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| monster | (required, string) a monster group id, when the map is loaded, a random monsters from that group are spawned. |
+| density | (optional, float) if defined, it overrides the default monster density at the location (monster density is bigger towards the city centers) (see `map::place_spawns`). |
+| chance | (optional, integer or min/max array) one in x chance of spawn point being created (see `map::place_spawns`). |
### Place a vehicle by type or group with "vehicles"
-| Field | Description
-| --- | ---
-| vehicle | (required, string) type of the vehicle or id of a vehicle group.
-| chance | (optional, integer or min/max array) x in 100 chance of the vehicle spawning at all. The default is 1 (which means 1% probability that the vehicle spawns, you probably want something larger).
-| rotation | (optional, integer) the direction the vehicle faces.
-| fuel | (optional, integer) the fuel status. Default is -1 which makes the tanks 1-7% full. Positive values are interpreted as percentage of the vehicles tanks to fill (e.g. 100 means completely full).
-| status | (optional, integer) default is -1 (light damage), a value of 0 means perfect condition, 1 means heavily damaged.
-
+| Field | Description |
+| -------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| vehicle | (required, string) type of the vehicle or id of a vehicle group. |
+| chance | (optional, integer or min/max array) x in 100 chance of the vehicle spawning at all. The default is 1 (which means 1% probability that the vehicle spawns, you probably want something larger). |
+| rotation | (optional, integer) the direction the vehicle faces. |
+| fuel | (optional, integer) the fuel status. Default is -1 which makes the tanks 1-7% full. Positive values are interpreted as percentage of the vehicles tanks to fill (e.g. 100 means completely full). |
+| status | (optional, integer) default is -1 (light damage), a value of 0 means perfect condition, 1 means heavily damaged. |
### Place a specific item with "item"
-| Field | Description
-| --- | ---
-| item | (required, string) the item type id of the new item.
-| chance | (optional, integer or min/max array) one in x chance that the item will spawn. Default is 1, meaning it will always spawn.
-| amount | (optional, integer or min/max array) the number of items to spawn, default is 1.
-| repeat | (optional, integer or min/max array) the number of times to repeat this placement, default is 1.
+| Field | Description |
+| ------ | -------------------------------------------------------------------------------------------------------------------------- |
+| item | (required, string) the item type id of the new item. |
+| chance | (optional, integer or min/max array) one in x chance that the item will spawn. Default is 1, meaning it will always spawn. |
+| amount | (optional, integer or min/max array) the number of items to spawn, default is 1. |
+| repeat | (optional, integer or min/max array) the number of times to repeat this placement, default is 1. |
-To use this type with explicit coordinates use the name "place_item" (this if for backwards compatibility) like this:
+To use this type with explicit coordinates use the name "place_item" (this if for backwards
+compatibility) like this:
```json
"item": {
@@ -843,48 +858,45 @@ To use this type with explicit coordinates use the name "place_item" (this if fo
### Place a specific monster with "monster"
-| Field | Description
-| --- | ---
-| monster | (required, string) type id of the monster (e.g. `mon_zombie`).
-| friendly | (optional, bool) whether the monster is friendly, default is false.
-| name | (optional, string) a name for that monster, optional, default is to create an unnamed monster.
-| target | (optional, bool) this monster is a mission target. Only valid for `update_mapgen`.
-
+| Field | Description |
+| -------- | ---------------------------------------------------------------------------------------------- |
+| monster | (required, string) type id of the monster (e.g. `mon_zombie`). |
+| friendly | (optional, bool) whether the monster is friendly, default is false. |
+| name | (optional, string) a name for that monster, optional, default is to create an unnamed monster. |
+| target | (optional, bool) this monster is a mission target. Only valid for `update_mapgen`. |
### Place a trap with "traps"
-| Field | Description
-| --- | ---
-| trap | (required, string) type id of the trap (e.g. `tr_beartrap`).
-
+| Field | Description |
+| ----- | ------------------------------------------------------------ |
+| trap | (required, string) type id of the trap (e.g. `tr_beartrap`). |
### Place furniture with "furniture"
-| Field | Description
-| --- | ---
-| furn | (required, string) type id of the furniture (e.g. `f_chair`).
-
+| Field | Description |
+| ----- | ------------------------------------------------------------- |
+| furn | (required, string) type id of the furniture (e.g. `f_chair`). |
### Place terrain with "terrain"
If the terrain has the value "roof" set and is in an enclosed space, it's indoors.
-| Field | Description
-| --- | ---
-| ter | (required, string) type id of the terrain (e.g. `t_floor`).
-
+| Field | Description |
+| ----- | ----------------------------------------------------------- |
+| ter | (required, string) type id of the terrain (e.g. `t_floor`). |
### Place rubble and smash existing terrain with "rubble"
-Creates rubble and bashes existing terrain (this step is applied last, after other things like furniture/terrain have
-been set). Creating rubble invokes the bashing function that can destroy terrain and cause structures to collapse.
+Creates rubble and bashes existing terrain (this step is applied last, after other things like
+furniture/terrain have been set). Creating rubble invokes the bashing function that can destroy
+terrain and cause structures to collapse.
-| Field | Description
-| --- | ---
-| rubble_type | (optional, furniture id, default: `f_rubble`) the type of the created rubble.
-| items | (optional, bool, default: false) place items that result from bashing the structure.
-| floor_type | (optional, terrain id, default: `t_dirt`) only used if there is a non-bashable wall at the location or with overwrite = true.
-| overwrite | (optional, bool, default: false) if true it just writes on top of what currently exists.
+| Field | Description |
+| ----------- | ----------------------------------------------------------------------------------------------------------------------------- |
+| rubble_type | (optional, furniture id, default: `f_rubble`) the type of the created rubble. |
+| items | (optional, bool, default: false) place items that result from bashing the structure. |
+| floor_type | (optional, terrain id, default: `t_dirt`) only used if there is a non-bashable wall at the location or with overwrite = true. |
+| overwrite | (optional, bool, default: false) if true it just writes on top of what currently exists. |
To use this type with explicit coordinates use the name "place_rubble" (no plural) like this:
@@ -896,17 +908,17 @@ To use this type with explicit coordinates use the name "place_rubble" (no plura
### Place spilled liquids with "place_liquids"
-Creates a liquid item at the specified location. Liquids can't currently be picked up (except for gasoline in tanks or
-pumps), but can be used to add flavor to mapgen.
+Creates a liquid item at the specified location. Liquids can't currently be picked up (except for
+gasoline in tanks or pumps), but can be used to add flavor to mapgen.
-| Field | Description
-| --- | ---
-| liquid | (required, item id) the item (a liquid)
-| amount | (optional, integer/min-max array) amount of liquid to place (a value of 0 defaults to the item's default charges)
-| chance | (optional, integer/min-max array) one in x chance of spawning a liquid, default value is 1 (100%)
+| Field | Description |
+| ------ | ----------------------------------------------------------------------------------------------------------------- |
+| liquid | (required, item id) the item (a liquid) |
+| amount | (optional, integer/min-max array) amount of liquid to place (a value of 0 defaults to the item's default charges) |
+| chance | (optional, integer/min-max array) one in x chance of spawning a liquid, default value is 1 (100%) |
-Example for dropping a default amount of gasoline (200 units) on the ground (either by using a character in the rows
-array or explicit coordinates):
+Example for dropping a default amount of gasoline (200 units) on the ground (either by using a
+character in the rows array or explicit coordinates):
```json
"liquids": {
@@ -919,38 +931,37 @@ array or explicit coordinates):
### Place a specific item or an item from a group with "loot"
-Places item(s) from an item group, or an individual item. An important distinction between this and `place_item` and
-`item`/`items` is that `loot` can spawn a single item from a distribution group (without looping). It can also spawn a
-matching magazine and ammo for guns.
+Places item(s) from an item group, or an individual item. An important distinction between this and
+`place_item` and `item`/`items` is that `loot` can spawn a single item from a distribution group
+(without looping). It can also spawn a matching magazine and ammo for guns.
**Note**: Either `group` or `item` must be specified, but not both.
-| Field | Description
-| --- | ---
-| group | (string) the item group to use (see `ITEM_SPAWN.md` for notes on collection vs distribution groups)
-| item | (string) the type id of the item to spawn
-| chance | (optional, integer) x in 100 chance of item(s) spawning. Defaults to 100.
-| ammo | (optional, integer) x in 100 chance of item(s) spawning with the default amount of ammo. Defaults to 0.
-| magazine | (optional, integer) x in 100 chance of item(s) spawning with the default magazine. Defaults to 0.
-
+| Field | Description |
+| -------- | ------------------------------------------------------------------------------------------------------- |
+| group | (string) the item group to use (see `ITEM_SPAWN.md` for notes on collection vs distribution groups) |
+| item | (string) the type id of the item to spawn |
+| chance | (optional, integer) x in 100 chance of item(s) spawning. Defaults to 100. |
+| ammo | (optional, integer) x in 100 chance of item(s) spawning with the default amount of ammo. Defaults to 0. |
+| magazine | (optional, integer) x in 100 chance of item(s) spawning with the default magazine. Defaults to 0. |
### Plant seeds in a planter with "sealed_item"
Places an item or item group inside furniture that has special handling for items.
-This is intended for furniture such as `f_plant_harvest` with the `PLANT` flag, because placing items on such furniture
-via the other means will not work (since they have the `NOITEM` FLAG).
+This is intended for furniture such as `f_plant_harvest` with the `PLANT` flag, because placing
+items on such furniture via the other means will not work (since they have the `NOITEM` FLAG).
-On such furniture, there is supposed to be a single (hidden) seed item which dictates the species of plant. Using
-`sealed_item`, you can create such plants by specifying the furniture and a seed item.
+On such furniture, there is supposed to be a single (hidden) seed item which dictates the species of
+plant. Using `sealed_item`, you can create such plants by specifying the furniture and a seed item.
**Note**: Exactly one of "item" or "items" must be given (but not both).
-| Field | Description
-| --- | ---
-| furniture | (string) the id of the chosen furniture.
-| item | spawn an item as the "item" special
-| items | spawn an item group as the "items" special.
+| Field | Description |
+| --------- | ------------------------------------------- |
+| furniture | (string) the id of the chosen furniture. |
+| item | spawn an item as the "item" special |
+| items | spawn an item group as the "items" special. |
Example:
@@ -963,28 +974,26 @@ Example:
},
```
-
### Place messages with "graffiti"
-Places a graffiti message at the location. Either "text" or "snippet" must be defined. The message may include tags like
-``, ``, and `` that will insert a randomly generated name, or `` that will
-insert the nearest city name.
-
-| Field | Description
-| --- | ---
-| text | (optional, string) the message that will be placed.
-| snippet | (optional, string) a category of snippets that the message will be pulled from.
+Places a graffiti message at the location. Either "text" or "snippet" must be defined. The message
+may include tags like ``, ``, and `` that will insert a randomly
+generated name, or `` that will insert the nearest city name.
+| Field | Description |
+| ------- | ------------------------------------------------------------------------------- |
+| text | (optional, string) the message that will be placed. |
+| snippet | (optional, string) a category of snippets that the message will be pulled from. |
### Place a zone for an NPC faction with "zones"
NPCs in the faction will use the zone to influence the AI.
-| Field | Description
-| --- | ---
-| type | (required, string) Values: `"NPC_RETREAT"`, `"NPC_NO_INVESTIGATE"`, or `"NPC_INVESTIGATE_ONLY"`.
-| faction | (required, string) the faction id of the NPC faction that will use the zone.
-| name | (optional, string) the name of the zone.
+| Field | Description |
+| ------- | ------------------------------------------------------------------------------------------------ |
+| type | (required, string) Values: `"NPC_RETREAT"`, `"NPC_NO_INVESTIGATE"`, or `"NPC_INVESTIGATE_ONLY"`. |
+| faction | (required, string) the faction id of the NPC faction that will use the zone. |
+| name | (optional, string) the name of the zone. |
The `type` field values affect NPC behavior. NPCs will:
@@ -992,89 +1001,88 @@ The `type` field values affect NPC behavior. NPCs will:
- Not move to the see the source of unseen sounds coming from `NPC_NO_INVESTIGATE` zones.
- Not move to the see the source of unseen sounds coming from outside `NPC_INVESTIGATE_ONLY` zones.
-
### Translate terrain type with "translate_ter"
-Translates one type of terrain into another type of terrain. There is no reason to do this with normal mapgen, but it
-is useful for setting a baseline with `update_mapgen`.
-
-| Field | Description
-| --- | ---
-| from | (required, string) the terrain id of the terrain to be transformed
-| to | (required, string) the terrain id that the from terrain will transformed into
+Translates one type of terrain into another type of terrain. There is no reason to do this with
+normal mapgen, but it is useful for setting a baseline with `update_mapgen`.
+| Field | Description |
+| ----- | ----------------------------------------------------------------------------- |
+| from | (required, string) the terrain id of the terrain to be transformed |
+| to | (required, string) the terrain id that the from terrain will transformed into |
### Apply mapgen transformation with "ter_furn_transforms"
-Run a `ter_furn_transform` at the specified location. This is mostly useful for applying transformations as part of
-an `update_mapgen`, as normal mapgen can just specify the terrain directly.
+Run a `ter_furn_transform` at the specified location. This is mostly useful for applying
+transformations as part of an `update_mapgen`, as normal mapgen can just specify the terrain
+directly.
- "transform": (required, string) the id of the `ter_furn_transform` to run.
-
## Rotate the map with "rotation"
-Rotates the generated map after all the other mapgen stuff has been done. The value can be a single integer or a range
-(out of which a value will be randomly chosen). Example: `"rotation": [ 0, 3 ]`
+Rotates the generated map after all the other mapgen stuff has been done. The value can be a single
+integer or a range (out of which a value will be randomly chosen). Example: `"rotation": [ 0, 3 ]`
Values are 90° steps.
-
## Pre-load a base mapgen with "predecessor_mapgen"
-Specifying an overmap terrain id here will run the entire mapgen for that overmap terrain type first, before applying
-the rest of the mapgen defined here. The primary use case for this is when our mapgen for a location takes place in a
-natural feature like a forest, swamp, or lake shore. Many existing JSON mapgen attempt to emulate the mapgen of the type
-they're being placed on (e.g. a cabin in the forest has placed the trees, grass and clutter of a forest to try to make
-the cabin fit in) which leads to them being out of sync when the generation of that type changes. By specifying the
-`predecessor_mapgen`, you can instead focus on the things that are added to the existing location type.
+Specifying an overmap terrain id here will run the entire mapgen for that overmap terrain type
+first, before applying the rest of the mapgen defined here. The primary use case for this is when
+our mapgen for a location takes place in a natural feature like a forest, swamp, or lake shore. Many
+existing JSON mapgen attempt to emulate the mapgen of the type they're being placed on (e.g. a cabin
+in the forest has placed the trees, grass and clutter of a forest to try to make the cabin fit in)
+which leads to them being out of sync when the generation of that type changes. By specifying the
+`predecessor_mapgen`, you can instead focus on the things that are added to the existing location
+type.
Example: `"predecessor_mapgen": "forest"`
-
# Using update_mapgen
-**update_mapgen** is a variant of normal JSON mapgen. Instead of creating a new overmap tile, it
-updates an existing overmap tile with a specific set of changes. Currently, it only works within
-the NPC mission interface, but it will be expanded to be a general purpose tool for modifying
-existing maps.
+**update_mapgen** is a variant of normal JSON mapgen. Instead of creating a new overmap tile, it
+updates an existing overmap tile with a specific set of changes. Currently, it only works within the
+NPC mission interface, but it will be expanded to be a general purpose tool for modifying existing
+maps.
-update_mapgen generally uses the same fields as JSON mapgen, with a few exceptions. update_mapgen has a few new fields
-to support missions, as well as ways to specify which overmap tile will be updated.
+update_mapgen generally uses the same fields as JSON mapgen, with a few exceptions. update_mapgen
+has a few new fields to support missions, as well as ways to specify which overmap tile will be
+updated.
## Overmap tile specification
-update_mapgen updates an existing overmap tile. These fields provide a way to specify which tile to update.
+update_mapgen updates an existing overmap tile. These fields provide a way to specify which tile to
+update.
### "assign_mission_target"
-assign_mission_target assigns an overmap tile as the target of a mission. Any update_mapgen in the same scope will
-update that overmap tile. The closet overmap terrain with the required terrain ID will be used, and if there is no
-matching terrain, an overmap special of om_special type will be created and then the om_terrain within that special will
-be used.
-
-| Field | Description
-| --- | ---
-| om_terrain | (required, string) the overmap terrain ID of the mission target
-| om_special | (required, string) the overmap special ID of the mission target
+assign_mission_target assigns an overmap tile as the target of a mission. Any update_mapgen in the
+same scope will update that overmap tile. The closet overmap terrain with the required terrain ID
+will be used, and if there is no matching terrain, an overmap special of om_special type will be
+created and then the om_terrain within that special will be used.
+| Field | Description |
+| ---------- | --------------------------------------------------------------- |
+| om_terrain | (required, string) the overmap terrain ID of the mission target |
+| om_special | (required, string) the overmap special ID of the mission target |
### "om_terrain"
-The closest overmap tile of type om_terrain in the closest overmap special of type om_special will be used. The overmap
-tile will be updated but will not be set as the mission target.
+The closest overmap tile of type om_terrain in the closest overmap special of type om_special will
+be used. The overmap tile will be updated but will not be set as the mission target.
-| Field | Description
-| --- | ---
-| om_terrain | (required, string) the overmap terrain ID of the mission target
-| om_special | (required, string) the overmap special ID of the mission target
+| Field | Description |
+| ---------- | --------------------------------------------------------------- |
+| om_terrain | (required, string) the overmap terrain ID of the mission target |
+| om_special | (required, string) the overmap special ID of the mission target |
# Mission specials
+
update_mapgen adds new optional keywords to a few mapgen JSON items.
### "target"
-place_npc, place_monster, and place_computer can take an optional target boolean. If they have `"target": true` and are
-invoked by update_mapgen with a valid mission, then the NPC, monster, or computer will be marked as the target of the
-mission.
-
+place_npc, place_monster, and place_computer can take an optional target boolean. If they have
+`"target": true` and are invoked by update_mapgen with a valid mission, then the NPC, monster, or
+computer will be marked as the target of the mission.
diff --git a/doc/MARTIALART_JSON.md b/doc/MARTIALART_JSON.md
index 6d9e6ef9e642..3fa4105a1260 100644
--- a/doc/MARTIALART_JSON.md
+++ b/doc/MARTIALART_JSON.md
@@ -103,30 +103,36 @@ The bonuses arrays contain any number of bonus entries like this:
}
```
-"stat": affected statistic, any of: "hit", "dodge", "block", "speed", "movecost", "damage", "armor", "arpen",
-"type": damage type for the affected statistic ("bash", "cut", "heat", etc.), only needed if the affected statistic is "damage", "armor", or "arpen".
-"scale": the value of the bonus itself.
-"scaling-stat": scaling stat, any of: "str", "dex", "int", "per". Optional. If the scaling stat is specified, the value of the bonus is multiplied by the corresponding user stat.
+"stat": affected statistic, any of: "hit", "dodge", "block", "speed", "movecost", "damage", "armor",
+"arpen", "type": damage type for the affected statistic ("bash", "cut", "heat", etc.), only needed
+if the affected statistic is "damage", "armor", or "arpen". "scale": the value of the bonus itself.
+"scaling-stat": scaling stat, any of: "str", "dex", "int", "per". Optional. If the scaling stat is
+specified, the value of the bonus is multiplied by the corresponding user stat.
Bonuses must be written in the correct order.
-Tokens of `useless` type will not cause an error, but will not have any effect.
-For example, `speed` in a technique will have no effect (`movecost` should be used for techniques).
+Tokens of `useless` type will not cause an error, but will not have any effect. For example, `speed`
+in a technique will have no effect (`movecost` should be used for techniques).
Currently extra elemental damage is not applied, but extra elemental armor is (after regular armor).
-Examples:
-Incoming bashing damage is decreased by 30% of strength value. Only useful on buffs:
-* `flat_bonuses : [ { "stat": "armor", "type": "bash", "scaling-stat": "str", "scale": 0.3 } ]`
+Examples: Incoming bashing damage is decreased by 30% of strength value. Only useful on buffs:
+
+- `flat_bonuses : [ { "stat": "armor", "type": "bash", "scaling-stat": "str", "scale": 0.3 } ]`
All cutting damage dealt is multiplied by `(10% of dexterity)*(damage)`:
-* `mult_bonuses : [ { "stat": "damage", "type": "cut", "scaling-stat": "dex", "scale": 0.1 } ]`
+
+- `mult_bonuses : [ { "stat": "damage", "type": "cut", "scaling-stat": "dex", "scale": 0.1 } ]`
Move cost is decreased by 100% of strength value
-* `flat_bonuses : [ { "stat": "movecost", "scaling-stat": "str", "scale": -1.0 } ]`
+
+- `flat_bonuses : [ { "stat": "movecost", "scaling-stat": "str", "scale": -1.0 } ]`
### Place relevant items in the world and chargen
-Starting trait selection of your martial art goes in mutations.json. Place your art in the right category (self-defense, Shaolin animal form, melee style, etc)
+Starting trait selection of your martial art goes in mutations.json. Place your art in the right
+category (self-defense, Shaolin animal form, melee style, etc)
-Use json/itemgroups/ to place your martial art book and any martial weapons you've made for the art into spawns in various locations in the world. If you don't place your weapons in there, only recipes to craft them will be an option.
+Use json/itemgroups/ to place your martial art book and any martial weapons you've made for the art
+into spawns in various locations in the world. If you don't place your weapons in there, only
+recipes to craft them will be an option.
diff --git a/doc/MISSIONS_JSON.md b/doc/MISSIONS_JSON.md
index f35569888800..4bbfc8a7f6a1 100644
--- a/doc/MISSIONS_JSON.md
+++ b/doc/MISSIONS_JSON.md
@@ -1,162 +1,181 @@
# Creating missions
-NPCs can assign missions to the player. There is a fairly regular structure for this:
+NPCs can assign missions to the player. There is a fairly regular structure for this:
```JSON
- {
- "id": "MISSION_GET_BLACK_BOX_TRANSCRIPT",
- "type": "mission_definition",
- "name": "Retrieve Black Box Transcript",
- "description": "Decrypt the contents of the black box using a terminal from a nearby lab.",
- "goal": "MGOAL_FIND_ITEM",
- "difficulty": 2,
- "value": 150000,
- "item": "black_box_transcript",
- "start": {
- "effect": { "u_buy_item": "black_box" },
- "assign_mission_target": { "om_terrain": "lab", "reveal_radius": 3 }
- },
- "origins": [ "ORIGIN_SECONDARY" ],
- "followup": "MISSION_EXPLORE_SARCOPHAGUS",
- "dialogue": {
- "describe": "With the black box in hand, we need to find a lab.",
- "offer": "Thanks to your searching we've got the black box but now we need to have a look'n-side her. Now, most buildings don't have power anymore but there are a few that might be of use. Have you ever seen one of those science labs that have popped up in the middle of nowhere? Them suckers have a glowing terminal out front so I know they have power somewhere inside'em. If you can get inside and find a computer lab that still works you ought to be able to find out what's in the black box.",
- "accepted": "Fuck ya, America!",
- "rejected": "Do you have any better ideas?",
- "advice": "When I was play'n with the terminal for the one I ran into it kept asking for an ID card. Finding one would be the first order of business.",
- "inquire": "How 'bout that black box?",
- "success": "America, fuck ya! I was in the guard a few years back so I'm confident I can make heads-or-tails of these transmissions.",
- "success_lie": "What?! I out'ta whip you're ass.",
- "failure": "Damn, I maybe we can find an egg-head to crack the terminal."
- }
+{
+ "id": "MISSION_GET_BLACK_BOX_TRANSCRIPT",
+ "type": "mission_definition",
+ "name": "Retrieve Black Box Transcript",
+ "description": "Decrypt the contents of the black box using a terminal from a nearby lab.",
+ "goal": "MGOAL_FIND_ITEM",
+ "difficulty": 2,
+ "value": 150000,
+ "item": "black_box_transcript",
+ "start": {
+ "effect": { "u_buy_item": "black_box" },
+ "assign_mission_target": { "om_terrain": "lab", "reveal_radius": 3 }
+ },
+ "origins": ["ORIGIN_SECONDARY"],
+ "followup": "MISSION_EXPLORE_SARCOPHAGUS",
+ "dialogue": {
+ "describe": "With the black box in hand, we need to find a lab.",
+ "offer": "Thanks to your searching we've got the black box but now we need to have a look'n-side her. Now, most buildings don't have power anymore but there are a few that might be of use. Have you ever seen one of those science labs that have popped up in the middle of nowhere? Them suckers have a glowing terminal out front so I know they have power somewhere inside'em. If you can get inside and find a computer lab that still works you ought to be able to find out what's in the black box.",
+ "accepted": "Fuck ya, America!",
+ "rejected": "Do you have any better ideas?",
+ "advice": "When I was play'n with the terminal for the one I ran into it kept asking for an ID card. Finding one would be the first order of business.",
+ "inquire": "How 'bout that black box?",
+ "success": "America, fuck ya! I was in the guard a few years back so I'm confident I can make heads-or-tails of these transmissions.",
+ "success_lie": "What?! I out'ta whip you're ass.",
+ "failure": "Damn, I maybe we can find an egg-head to crack the terminal."
}
+}
```
### type
+
Must always be there and must always be "mission_definition".
### id
-The mission id is required, but for new missions, it can be arbitrary. Convention is to start
-it with "MISSION" and to use a fairly descriptive name.
+
+The mission id is required, but for new missions, it can be arbitrary. Convention is to start it
+with "MISSION" and to use a fairly descriptive name.
### name
+
The name is also required, and is displayed to the user in the 'm'issions menu.
### description
+
Not required, but it's strongly recommended that you summarize all relevant info for the mission.
-You may refer to mission end effects of the "u_buy_item" type, as long as they do not come at a
-cost to the player. See the example below:
+You may refer to mission end effects of the "u_buy_item" type, as long as they do not come at a cost
+to the player. See the example below:
+
```JSON
- "id": "MISSION_EXAMPLE_TOKENS",
- "type": "mission_definition",
- "name": "Murder Money",
- "description": "Whack the target in exchange for c-notes and cigarettes.",
- "goal": "MGOAL_ASSASSINATE",
- "end": {
- "effect": [
- { "u_buy_item": "FMCNote", "count": 999 },
- { "u_buy_item": "cig", "count": 666 } ]
- }
+"id": "MISSION_EXAMPLE_TOKENS",
+"type": "mission_definition",
+"name": "Murder Money",
+"description": "Whack the target in exchange for c-notes and cigarettes.",
+"goal": "MGOAL_ASSASSINATE",
+"end": {
+ "effect": [
+ { "u_buy_item": "FMCNote", "count": 999 },
+ { "u_buy_item": "cig", "count": 666 } ]
+}
```
-This system may be expanded in the future to allow referring to other mission parameters and effects.
+
+This system may be expanded in the future to allow referring to other mission parameters and
+effects.
### goal
+
Must be included, and must be one of these strings:
-goal string | Goal conditions
---- | ---
-`MGOAL_GO_TO` | Reach a specific overmap tile
-`MGOAL_GO_TO_TYPE` | Reach any instance of a specified overmap tile type
-`MGOAL_COMPUTER_TOGGLE` | Activating the correct terminal will complete the mission
-`MGOAL_FIND_ITEM` | Find 1 or more items of a given type
-`MGOAL_FIND_ITEM_GROUP` | Find 1 or more items of a given item_group.
-`MGOAL_FIND_ANY_ITEM` | Find 1 or more items of a given type, tagged for this mission
-`MGOAL_FIND_MONSTER` | Find and retrieve a friendly monster
-`MGOAL_FIND_NPC` | Find a specific NPC
-`MGOAL_TALK_TO_NPC` | Talk to a specific NPC
-`MGOAL_RECRUIT_NPC` | Recruit a specific NPC
-`MGOAL_RECRUIT_NPC_CLASS` | Recruit an NPC of a specific class
-`MGOAL_ASSASSINATE` | Kill a specific NPC
-`MGOAL_KILL_MONSTER` | Kill a specific hostile monster
-`MGOAL_KILL_MONSTER_TYPE` | Kill some number of a specific monster type
-`MGOAL_KILL_MONSTER_SPEC` | Kill some number of monsters from a specific species
-`MGOAL_CONDITION` | Satisfy the dynamically created condition and talk to the mission giver
+| goal string | Goal conditions |
+| ------------------------- | ----------------------------------------------------------------------- |
+| `MGOAL_GO_TO` | Reach a specific overmap tile |
+| `MGOAL_GO_TO_TYPE` | Reach any instance of a specified overmap tile type |
+| `MGOAL_COMPUTER_TOGGLE` | Activating the correct terminal will complete the mission |
+| `MGOAL_FIND_ITEM` | Find 1 or more items of a given type |
+| `MGOAL_FIND_ITEM_GROUP` | Find 1 or more items of a given item_group. |
+| `MGOAL_FIND_ANY_ITEM` | Find 1 or more items of a given type, tagged for this mission |
+| `MGOAL_FIND_MONSTER` | Find and retrieve a friendly monster |
+| `MGOAL_FIND_NPC` | Find a specific NPC |
+| `MGOAL_TALK_TO_NPC` | Talk to a specific NPC |
+| `MGOAL_RECRUIT_NPC` | Recruit a specific NPC |
+| `MGOAL_RECRUIT_NPC_CLASS` | Recruit an NPC of a specific class |
+| `MGOAL_ASSASSINATE` | Kill a specific NPC |
+| `MGOAL_KILL_MONSTER` | Kill a specific hostile monster |
+| `MGOAL_KILL_MONSTER_TYPE` | Kill some number of a specific monster type |
+| `MGOAL_KILL_MONSTER_SPEC` | Kill some number of monsters from a specific species |
+| `MGOAL_CONDITION` | Satisfy the dynamically created condition and talk to the mission giver |
### monster_species
+
For "MGOAL_KILL_MONSTER_SPEC", sets the target monster species.
### monster_type
+
For "MGOAL_KILL_MONSTER_TYPE", sets the target monster type.
### monster_kill_goal
-For "MGOAL_KILL_MONSTER_SPEC" and "MGOAL_KILL_MONSTER_TYPE", sets the number of monsters above
-the player's current kill count that must be killed to complete the mission.
+
+For "MGOAL_KILL_MONSTER_SPEC" and "MGOAL_KILL_MONSTER_TYPE", sets the number of monsters above the
+player's current kill count that must be killed to complete the mission.
### goal_condition
-For "MGOAL_CONDITION", defines the condition that must be satisified for the mission to be considered complete.
-Conditions are explained in more detail in [NPCs.md](./NPCs.md), and are used here in exactly the same way.
+
+For "MGOAL_CONDITION", defines the condition that must be satisified for the mission to be
+considered complete. Conditions are explained in more detail in [NPCs.md](./NPCs.md), and are used
+here in exactly the same way.
### dialogue
-This is a dictionary of strings. The NPC says these exact strings in response to the player
-inquiring about the mission or reporting its completion. All these strings are required, even if they may not be used in the mission.
-
-string ID | Usage
---- | ---
-`describe` | The NPC's overall description of the mission
-`offer` | The specifics of the mission given when the player selects that mission for consideration
-`accepted` | The NPC's response if the player accepts the mission
-`rejected` | The NPC's response if the player refuses the mission
-`advice` | If the player asks for advice on how to complete the mission, they hear this
-`inquire` | This is used if the NPC asks the player how the mission is going
-`success` | The NPC's response to a report that the mission was successful
-`success_lie` | The NPC's response if they catch the player lying about a mission's success
-`failure` | The NPC's response if the player reports a failed mission
+
+This is a dictionary of strings. The NPC says these exact strings in response to the player
+inquiring about the mission or reporting its completion. All these strings are required, even if
+they may not be used in the mission.
+
+| string ID | Usage |
+| ------------- | ----------------------------------------------------------------------------------------- |
+| `describe` | The NPC's overall description of the mission |
+| `offer` | The specifics of the mission given when the player selects that mission for consideration |
+| `accepted` | The NPC's response if the player accepts the mission |
+| `rejected` | The NPC's response if the player refuses the mission |
+| `advice` | If the player asks for advice on how to complete the mission, they hear this |
+| `inquire` | This is used if the NPC asks the player how the mission is going |
+| `success` | The NPC's response to a report that the mission was successful |
+| `success_lie` | The NPC's response if they catch the player lying about a mission's success |
+| `failure` | The NPC's response if the player reports a failed mission |
### start
-Optional field. If it is present and a string, it must be name of a function in mission_start::
-that takes a mission * and performs the start code for the mission. This allows missions other
-than the standard mission types to be run. A hardcoded function is currently necessary to set
-up missions with "MGOAL_COMPUTER_TOGGLE".
+
+Optional field. If it is present and a string, it must be name of a function in mission_start:: that
+takes a mission * and performs the start code for the mission. This allows missions other than the
+standard mission types to be run. A hardcoded function is currently necessary to set up missions
+with "MGOAL_COMPUTER_TOGGLE".
Alternately, if present, it can be an object as described below.
### start / end / fail effects
+
If any of these optional fields are present they can be objects with the following fields contained:
#### effect
-This is an effects array, exactly as defined in [NPCs.md](./NPCs.md), and can use any of the values from
-effects. In all cases, the NPC involved is the quest giver.
+
+This is an effects array, exactly as defined in [NPCs.md](./NPCs.md), and can use any of the values
+from effects. In all cases, the NPC involved is the quest giver.
#### reveal_om_ter
-This can be a string or a list of strings, all of which must be overmap terrain ids. A randomly
-selected overmap terrain tile with that id - or one of the ids from list, randomly selected -
-will be revealed, and there is a 1 in 3 chance that the road route to the map tile will also
-be revealed.
+
+This can be a string or a list of strings, all of which must be overmap terrain ids. A randomly
+selected overmap terrain tile with that id - or one of the ids from list, randomly selected - will
+be revealed, and there is a 1 in 3 chance that the road route to the map tile will also be revealed.
#### assign_mission_target
-The `assign_mission_target` object specifies the criteria for finding (or creating if
-necessary) a particular overmap terrain and designating it as the mission target. Its parameters
-allow control over how it is picked and how some effects (such as revealing the surrounding area)
-are applied afterwards. The `om_terrain` is the only required field.
-
- Identifier | Description
---- | ---
-`om_terrain` | ID of overmap terrain which will be selected as the target. Mandatory.
-`om_terrain_match_type`| Matching rule to use with `om_terrain`. Defaults to TYPE. Details below.
-`om_special` | ID of overmap special containing the overmap terrain.
-`om_terrain_replace` | ID of overmap terrain to be found and replaced if `om_terrain` cannot be found.
-`reveal_radius` | Radius in overmap terrain coordinates to reveal.
-`must_see` | If true, the `om_terrain` must have been seen already.
-`cant_see` | If true, the `om_terrain` must not have been seen already.
-`random` | If true, a random matching `om_terrain` is used. If false, the closest is used.
-`search_range` | Range in overmap terrain coordinates to look for a matching `om_terrain`.
-`min_distance` | Range in overmap terrain coordinates. Instances of `om_terrain` in this range will be ignored.
-`origin_npc` | Start the search at the NPC's, rather than the player's, current position.
-`z` | If specified, will be used rather than the player or NPC's z when searching.
-`offset_x`,
`offset_y`,
`offset_z` | After finding or creating `om_terrain`, offset the mission target terrain by the offsets in overmap terrain coordinates.
+The `assign_mission_target` object specifies the criteria for finding (or creating if necessary) a
+particular overmap terrain and designating it as the mission target. Its parameters allow control
+over how it is picked and how some effects (such as revealing the surrounding area) are applied
+afterwards. The `om_terrain` is the only required field.
+
+| Identifier | Description |
+| ------------------------------------------ | ------------------------------------------------------------------------------------------------------------------------ |
+| `om_terrain` | ID of overmap terrain which will be selected as the target. Mandatory. |
+| `om_terrain_match_type` | Matching rule to use with `om_terrain`. Defaults to TYPE. Details below. |
+| `om_special` | ID of overmap special containing the overmap terrain. |
+| `om_terrain_replace` | ID of overmap terrain to be found and replaced if `om_terrain` cannot be found. |
+| `reveal_radius` | Radius in overmap terrain coordinates to reveal. |
+| `must_see` | If true, the `om_terrain` must have been seen already. |
+| `cant_see` | If true, the `om_terrain` must not have been seen already. |
+| `random` | If true, a random matching `om_terrain` is used. If false, the closest is used. |
+| `search_range` | Range in overmap terrain coordinates to look for a matching `om_terrain`. |
+| `min_distance` | Range in overmap terrain coordinates. Instances of `om_terrain` in this range will be ignored. |
+| `origin_npc` | Start the search at the NPC's, rather than the player's, current position. |
+| `z` | If specified, will be used rather than the player or NPC's z when searching. |
+| `offset_x`,
`offset_y`,
`offset_z` | After finding or creating `om_terrain`, offset the mission target terrain by the offsets in overmap terrain coordinates. |
**example**
+
```JSON
{
"assign_mission_target": {
@@ -171,65 +190,71 @@ are applied afterwards. The `om_terrain` is the only required field.
}
```
-If the `om_terrain` is part of an overmap special, it's essential to specify the `om_special`
-value as well--otherwise, the game will not know how to spawn the entire special.
+If the `om_terrain` is part of an overmap special, it's essential to specify the `om_special` value
+as well--otherwise, the game will not know how to spawn the entire special.
`om_terrain_match_type` defaults to TYPE if unspecified, and has the following possible values:
-* `EXACT` - The provided string must completely match the overmap terrain id,
- including linear direction suffixes for linear terrain types or rotation
- suffixes for rotated terrain types.
+- `EXACT` - The provided string must completely match the overmap terrain id, including linear
+ direction suffixes for linear terrain types or rotation suffixes for rotated terrain types.
-* `TYPE` - The provided string must completely match the base type id of the
- overmap terrain id, which means that suffixes for rotation and linear terrain
- types are ignored.
+- `TYPE` - The provided string must completely match the base type id of the overmap terrain id,
+ which means that suffixes for rotation and linear terrain types are ignored.
-* `PREFIX` - The provided string must be a complete prefix (with additional
- parts delimited by an underscore) of the overmap terrain id. For example,
- "forest" will match "forest" or "forest_thick" but not "forestcabin".
+- `PREFIX` - The provided string must be a complete prefix (with additional parts delimited by an
+ underscore) of the overmap terrain id. For example, "forest" will match "forest" or "forest_thick"
+ but not "forestcabin".
-* `CONTAINS` - The provided string must be contained within the overmap terrain
- id, but may occur at the beginning, end, or middle and does not have any rules
- about underscore delimiting.
+- `CONTAINS` - The provided string must be contained within the overmap terrain id, but may occur at
+ the beginning, end, or middle and does not have any rules about underscore delimiting.
-If an `om_special` must be placed, it will follow the same placement rules as defined in its
-overmap special definition, respecting allowed terrains, distance from cities, road connections,
-and so on. Consequently, the more restrictive the rules, the less likely this placement will
-succeed (as it is competing for space with already-spawned specials).
+If an `om_special` must be placed, it will follow the same placement rules as defined in its overmap
+special definition, respecting allowed terrains, distance from cities, road connections, and so on.
+Consequently, the more restrictive the rules, the less likely this placement will succeed (as it is
+competing for space with already-spawned specials).
-`om_terrain_replace` is only relevent if the `om_terrain` is not part of an overmap special.
-This value is used if the `om_terrain` cannot be found, and will be used as an alternative target
-which will then be replaced with the `om_terrain` value.
+`om_terrain_replace` is only relevent if the `om_terrain` is not part of an overmap special. This
+value is used if the `om_terrain` cannot be found, and will be used as an alternative target which
+will then be replaced with the `om_terrain` value.
-If `must_see` is set to true, the game will not create the terrain if it can't be found. It tries
-to avoid having new terrains magically appear in areas where the player has already been, and this
-is a consequence.
+If `must_see` is set to true, the game will not create the terrain if it can't be found. It tries to
+avoid having new terrains magically appear in areas where the player has already been, and this is a
+consequence.
`reveal_radius`, `min_distance`, and `search_range` are all in overmap terrain coordinates (an
-overmap is currently 180 x 180 OMT units). The search is centered on the player unless
-`origin_npc` is set, in which case it centered on the NPC. Currently, there is rarely a
-difference between the two, but it is possible to assign missions over a radio. The search gives
-preference to existing overmaps (and will only spawn new overmaps if the terrain can't be found
-on existing overmaps), and only executes on the player's current z-level. The `z` attribute can
-be used to override this and search on a different z-level than the player--this is an absolute
-value rather than relative.
+overmap is currently 180 x 180 OMT units). The search is centered on the player unless `origin_npc`
+is set, in which case it centered on the NPC. Currently, there is rarely a difference between the
+two, but it is possible to assign missions over a radio. The search gives preference to existing
+overmaps (and will only spawn new overmaps if the terrain can't be found on existing overmaps), and
+only executes on the player's current z-level. The `z` attribute can be used to override this and
+search on a different z-level than the player--this is an absolute value rather than relative.
`offset_x`, `offset_y`, and `offset_z` change the final location of the mission target by their
-values. This can change the mission target's overmap terrain type away from `om_terrain`.
+values. This can change the mission target's overmap terrain type away from `om_terrain`.
#### update_mapgen
-The `update_mapgen` object or array provides a way to modify existing overmap tiles (including the ones created by "assign_mission_target") to add mission specific monsters, NPCs, computers, or items.
+
+The `update_mapgen` object or array provides a way to modify existing overmap tiles (including the
+ones created by "assign_mission_target") to add mission specific monsters, NPCs, computers, or
+items.
As an array, `update_mapgen` consists of two or more `update_mapgen` objects.
-As an object, `update_mapgen` contains any valid JSON mapgen objects. The objects are placed on the mission target terrain from "assign_mission_target" or optionally the closest overmap terrain specified by the `om_terrain` and `om_special` fields. If "mapgen_update_id" is specified, the "mapge_update" object with the matching "mapgen_update_id" will be executed.
+As an object, `update_mapgen` contains any valid JSON mapgen objects. The objects are placed on the
+mission target terrain from "assign_mission_target" or optionally the closest overmap terrain
+specified by the `om_terrain` and `om_special` fields. If "mapgen_update_id" is specified, the
+"mapge_update" object with the matching "mapgen_update_id" will be executed.
See doc/MAPGEN.md for more details on JSON mapgen and `update_mapgen`.
-An NPC, monster, or computer placed using `update_mapgen` will be the target of a mission if it has the `target` boolean set to `true` in its `place` object in `update_mapgen`.
+An NPC, monster, or computer placed using `update_mapgen` will be the target of a mission if it has
+the `target` boolean set to `true` in its `place` object in `update_mapgen`.
## Adding new missions to NPC dialogue
-In order to assign missions to NPCs, the first step is to find that NPC's definition. For unique NPCs this is usually at the top of the npc's JSON file and looks something like this:
+
+In order to assign missions to NPCs, the first step is to find that NPC's definition. For unique
+NPCs this is usually at the top of the npc's JSON file and looks something like this:
+
```JSON
{
"type": "npc",
@@ -245,23 +270,26 @@ In order to assign missions to NPCs, the first step is to find that NPC's defini
"faction": "lobby_beggars"
},
```
+
Add a new line that defines the NPC's starting mission, eg:
+
```
"mission_offered": "MISSION_BEGGAR_2_BOX_SMALL"
```
-Any NPC that has missions needs to have a dialogue option that leads to TALK_MISSION_LIST, to get the player
-started on their first mission for the NPC, and either:
+Any NPC that has missions needs to have a dialogue option that leads to TALK_MISSION_LIST, to get
+the player started on their first mission for the NPC, and either:
-* Add one of their talk_topic IDs to the list of generic mission reponse IDs in the first
-talk_topic of data/json/npcs/TALK_COMMON_MISSION.json, or
-* Have a similar talk_topic with responses that lead to TALK_MISSION_INQUIRE and
-TALK_MISSION_LIST_ASSIGNED.
+- Add one of their talk_topic IDs to the list of generic mission reponse IDs in the first talk_topic
+ of data/json/npcs/TALK_COMMON_MISSION.json, or
+- Have a similar talk_topic with responses that lead to TALK_MISSION_INQUIRE and
+ TALK_MISSION_LIST_ASSIGNED.
Either of these options will allow the player to do normal mission management dialogue with the NPC.
-This is an example of how a custom mission inquiry might appear. This will only appear in the NPC's dialogue
-options if the player has already been assigned a mission.
+This is an example of how a custom mission inquiry might appear. This will only appear in the NPC's
+dialogue options if the player has already been assigned a mission.
+
```JSON
{
"type": "talk_topic",
diff --git a/doc/MODDING.md b/doc/MODDING.md
index e1c7c86c2ed6..8fc2f5ea7137 100644
--- a/doc/MODDING.md
+++ b/doc/MODDING.md
@@ -1,59 +1,97 @@
# Modding guide
-Certain features of the game can be modified without rebuilding the game from source code. This includes professions, monsters, npcs, and more. Just modify the pertinent files and run the game to see your changes.
+Certain features of the game can be modified without rebuilding the game from source code. This
+includes professions, monsters, npcs, and more. Just modify the pertinent files and run the game to
+see your changes.
-The majority of modding is done by editing JSON files. An in-depth review of all json files and their appropriate fields is available in [JSON_INFO.md](JSON_INFO.md).
+The majority of modding is done by editing JSON files. An in-depth review of all json files and
+their appropriate fields is available in [JSON_INFO.md](JSON_INFO.md).
## The basics
### Creating a barebones mod
-A mod is created by creating a folder within Cataclysm's `data/mods` directory. The mod's properties are set by the `modinfo.json` file that is present within that folder. In order for Cataclysm to recognize the folder as a mod, it **must** have a `modinfo.json` file present within it.
+
+A mod is created by creating a folder within Cataclysm's `data/mods` directory. The mod's properties
+are set by the `modinfo.json` file that is present within that folder. In order for Cataclysm to
+recognize the folder as a mod, it **must** have a `modinfo.json` file present within it.
+
+
### Modinfo.json
-The modinfo.json file is a file that contains metadata for your mod. Every mod must have a `modinfo.json` file in order for Cataclysm to find it.
-A barebones `modinfo.json` file looks like this:
-````json
+
+The modinfo.json file is a file that contains metadata for your mod. Every mod must have a
+`modinfo.json` file in order for Cataclysm to find it. A barebones `modinfo.json` file looks like
+this:
+
+```json
[
{
"type": "MOD_INFO",
"id": "Mod_ID",
"name": "Mod's Display Name",
- "authors": [ "Your name here", "Your friend's name if you want" ],
+ "authors": ["Your name here", "Your friend's name if you want"],
"description": "Your description here",
"category": "content",
- "dependencies": [ "bn" ]
+ "dependencies": ["bn"]
}
]
-````
-The `category` attribute denotes where the mod will appear in the mod selection menu. These are the available categories to choose from, with some examples chosen from mods that existed when this document was written. Pick whichever one applies best to your mod when writing your modinfo file.
- - `content` - A mod that adds a lot of stuff. Typically reserved for very large mods or complete game overhauls (eg: Core game files, Aftershock)
- - `items` - A mod that adds new items and recipes to the game (eg: More survival tools)
- - `creatures` - A mod that adds new creatures or NPCs to the game (eg: Modular turrets)
- - `misc_additions` - Miscellaneous content additions to the game (eg: Alternative map key, Crazy cataclysm)
- - `buildings` - New overmap locations and structures (eg: Fuji's more buildings). If you're blacklisting buildings from spawning, this is also a usable category (eg: No rail stations).
- - `vehicles` - New cars or vehicle parts (eg: Tanks and other vehicles)
- - `rebalance` - A mod designed to rebalance the game in some way (eg: Safe autodocs).
- - `magical` - A mod that adds something magic-related to the game (eg: Necromancy)
- - `item_exclude` - A mod that stops items from spawning in the world (eg: No survivor armor, No drugs)
- - `monster_exclude` - A mod that stops certain monster varieties from spawning in the world (eg: No fungal monsters, No ants)
- - `graphical` - A mod that adjusts game graphics in some way (eg: Graphical overmap)
-
-The `dependencies` attribute is used to tell Cataclysm that your mod is dependent on something present in another mod. If you have no dependencies outside of the core game, then just including `dda` in the list is good enough. If your mod depends on another one to work properly, adding that mod's `id` attribute to the array causes Cataclysm to force that mod to load before yours.
+```
+
+The `category` attribute denotes where the mod will appear in the mod selection menu. These are the
+available categories to choose from, with some examples chosen from mods that existed when this
+document was written. Pick whichever one applies best to your mod when writing your modinfo file.
+
+- `content` - A mod that adds a lot of stuff. Typically reserved for very large mods or complete
+ game overhauls (eg: Core game files, Aftershock)
+- `items` - A mod that adds new items and recipes to the game (eg: More survival tools)
+- `creatures` - A mod that adds new creatures or NPCs to the game (eg: Modular turrets)
+- `misc_additions` - Miscellaneous content additions to the game (eg: Alternative map key, Crazy
+ cataclysm)
+- `buildings` - New overmap locations and structures (eg: Fuji's more buildings). If you're
+ blacklisting buildings from spawning, this is also a usable category (eg: No rail stations).
+- `vehicles` - New cars or vehicle parts (eg: Tanks and other vehicles)
+- `rebalance` - A mod designed to rebalance the game in some way (eg: Safe autodocs).
+- `magical` - A mod that adds something magic-related to the game (eg: Necromancy)
+- `item_exclude` - A mod that stops items from spawning in the world (eg: No survivor armor, No
+ drugs)
+- `monster_exclude` - A mod that stops certain monster varieties from spawning in the world (eg: No
+ fungal monsters, No ants)
+- `graphical` - A mod that adjusts game graphics in some way (eg: Graphical overmap)
+
+The `dependencies` attribute is used to tell Cataclysm that your mod is dependent on something
+present in another mod. If you have no dependencies outside of the core game, then just including
+`dda` in the list is good enough. If your mod depends on another one to work properly, adding that
+mod's `id` attribute to the array causes Cataclysm to force that mod to load before yours.
For more details on `MOD_INFO` object, see [JSON_INFO.md](JSON_INFO.md#mod_info).
## Actually adding things to your mod
+
Now that you have a basic mod, you can get around to actually putting some stuff into it!
### File structure
-It's a good idea to put different categories of additions into different json files. Any json files that are present in the mod folder or its subfolders will be detected and read by Cataclysm, but otherwise, there are no restrictions on what you can put where.
+
+It's a good idea to put different categories of additions into different json files. Any json files
+that are present in the mod folder or its subfolders will be detected and read by Cataclysm, but
+otherwise, there are no restrictions on what you can put where.
### JSON_INFO.md
-It's worth reading [JSON_INFO.md](JSON_INFO.md) to get a comprehensive list of everything you can do with these mods. The rest of this document will have a few examples to copy and paste, but it is by no means comprehensive. The base game's data is also defined in the same way as any mod you write, so taking a look through the game's json files (in `data/json`) can also teach you a lot. If the game finds any issues in your JSON syntax when you try to load a game world, it will spit out an error message, and you won't be able to load that game until the issue is fixed.
+
+It's worth reading [JSON_INFO.md](JSON_INFO.md) to get a comprehensive list of everything you can do
+with these mods. The rest of this document will have a few examples to copy and paste, but it is by
+no means comprehensive. The base game's data is also defined in the same way as any mod you write,
+so taking a look through the game's json files (in `data/json`) can also teach you a lot. If the
+game finds any issues in your JSON syntax when you try to load a game world, it will spit out an
+error message, and you won't be able to load that game until the issue is fixed.
### Adding a scenario
-Scenarios are what the game uses to determine your general situation when you create a character. They determine when and where your character may spawn in the world, and what professions can be used. They are also used to determine whether those professions can have mutations starting out. Below you will find the JSON definition for the game's built-in `Large Building` scenario.
-````json
+
+Scenarios are what the game uses to determine your general situation when you create a character.
+They determine when and where your character may spawn in the world, and what professions can be
+used. They are also used to determine whether those professions can have mutations starting out.
+Below you will find the JSON definition for the game's built-in `Large Building` scenario.
+
+```json
[
{
"type": "scenario",
@@ -79,14 +117,18 @@ Scenarios are what the game uses to determine your general situation when you cr
"hospital_9"
],
"start_name": "In Large Building",
- "flags": [ "SUR_START", "CITY_START", "LONE_START" ]
+ "flags": ["SUR_START", "CITY_START", "LONE_START"]
}
]
-````
+```
### Adding a profession.
-Professions are what the game calls the character classes you can choose from when the game starts. Professions can start with traits, skills, items, and even pets! Below is the definition for the Police Officer profession:
-````json
+
+Professions are what the game calls the character classes you can choose from when the game starts.
+Professions can start with traits, skills, items, and even pets! Below is the definition for the
+Police Officer profession:
+
+```json
[
{
"type": "profession",
@@ -94,29 +136,43 @@ Professions are what the game calls the character classes you can choose from wh
"name": "Police Officer",
"description": "Just a small-town deputy when you got the call, you were still ready to come to the rescue. Except that soon it was you who needed rescuing - you were lucky to escape with your life. Who's going to respect your authority when the government this badge represents might not even exist anymore?",
"points": 2,
- "skills": [ { "level": 3, "name": "gun" }, { "level": 3, "name": "pistol" } ],
- "traits": [ "PROF_POLICE" ],
+ "skills": [{ "level": 3, "name": "gun" }, { "level": 3, "name": "pistol" }],
+ "traits": ["PROF_POLICE"],
"items": {
"both": {
- "items": [ "pants_army", "socks", "badge_deputy", "sheriffshirt", "police_belt", "smart_phone", "boots", "whistle", "wristwatch" ],
+ "items": [
+ "pants_army",
+ "socks",
+ "badge_deputy",
+ "sheriffshirt",
+ "police_belt",
+ "smart_phone",
+ "boots",
+ "whistle",
+ "wristwatch"
+ ],
"entries": [
{ "group": "charged_two_way_radio" },
- { "item": "ear_plugs", "custom-flags": [ "no_auto_equip" ] },
+ { "item": "ear_plugs", "custom-flags": ["no_auto_equip"] },
{ "item": "usp_45", "ammo-item": "45_acp", "charges": 12, "container-item": "holster" },
{ "item": "legpouch_large", "contents-group": "army_mags_usp45" }
]
},
- "male": [ "boxer_shorts" ],
- "female": [ "bra", "boy_shorts" ]
+ "male": ["boxer_shorts"],
+ "female": ["bra", "boy_shorts"]
}
}
]
-````
+```
### Adding an item
-Items are where you really want to read the [JSON_INFO.md](JSON_INFO.md) file, just because there's so much that you can do with them, and every category of item is a little bit different.
+
+Items are where you really want to read the [JSON_INFO.md](JSON_INFO.md) file, just because there's
+so much that you can do with them, and every category of item is a little bit different.
+
-````json
+
+```json
[
{
"id": "family_photo",
@@ -128,21 +184,27 @@ Items are where you really want to read the [JSON_INFO.md](JSON_INFO.md) file, j
"weight": "1 g",
"volume": 0,
"price": 800,
- "material": [ "paper" ],
+ "material": ["paper"],
"symbol": "*",
"color": "light_gray"
}
]
-````
+```
### Preventing monsters from spawning
-This kind of mod is relatively simple, but very useful. If you don't want to deal with certain types of monsters in your world, this is how you do it. There are two ways to go about this, and both will be detailed below. You can blacklist entire monster groups, or you can blacklist certain monsters. In order to do either of those things, you need that monster's ID. These can be found in the relevant data files. For the core game, these are in the `data/json/monsters` directory.
-The example below is from the `No Ants` mod, and will stop any kind of ant from spawning in-game.
-````json
+
+This kind of mod is relatively simple, but very useful. If you don't want to deal with certain types
+of monsters in your world, this is how you do it. There are two ways to go about this, and both will
+be detailed below. You can blacklist entire monster groups, or you can blacklist certain monsters.
+In order to do either of those things, you need that monster's ID. These can be found in the
+relevant data files. For the core game, these are in the `data/json/monsters` directory. The example
+below is from the `No Ants` mod, and will stop any kind of ant from spawning in-game.
+
+```json
[
{
"type": "MONSTER_BLACKLIST",
- "categories": [ "GROUP_ANT", "GROUP_ANT_ACID" ]
+ "categories": ["GROUP_ANT", "GROUP_ANT_ACID"]
},
{
"type": "MONSTER_BLACKLIST",
@@ -158,74 +220,106 @@ The example below is from the `No Ants` mod, and will stop any kind of ant from
]
}
]
-````
+```
+
### Preventing locations from spawning
+
-Preventing certain types of locations from spawning in-game is a little trickier depending on the type of thing you want to target. An overmap building can either be a standard building, or an overmap special. If you want to block things with a specific flag from spawning, you can blacklist those in a very similar manner to monsters. The example below is also from the `No Ants` mod, and stops all anthills from spawning.
-````json
+
+Preventing certain types of locations from spawning in-game is a little trickier depending on the
+type of thing you want to target. An overmap building can either be a standard building, or an
+overmap special. If you want to block things with a specific flag from spawning, you can blacklist
+those in a very similar manner to monsters. The example below is also from the `No Ants` mod, and
+stops all anthills from spawning.
+
+```json
[
{
"type": "region_overlay",
- "regions": [ "all" ],
- "overmap_feature_flag_settings": { "blacklist": [ "ANT" ] }
+ "regions": ["all"],
+ "overmap_feature_flag_settings": { "blacklist": ["ANT"] }
}
]
-````
+```
-If the location you want to blacklist is an overmap special, you'll likely have to copy it from its definition, and manually set it's `occurrences` attribute to read `[ 0, 0 ]`.
+If the location you want to blacklist is an overmap special, you'll likely have to copy it from its
+definition, and manually set it's `occurrences` attribute to read `[ 0, 0 ]`.
-Finally, if you're trying to blacklist something that spawns inside of cities, you can do that with a region overlay. The below example is used in the `No rail stations` mod, and stops railroad stations from spawning inside of cities. It doesn't stop the railroad station overmap special from spawning, though.
-````json
+Finally, if you're trying to blacklist something that spawns inside of cities, you can do that with
+a region overlay. The below example is used in the `No rail stations` mod, and stops railroad
+stations from spawning inside of cities. It doesn't stop the railroad station overmap special from
+spawning, though.
+
+```json
[
{
"type": "region_overlay",
- "regions": [ "all" ],
+ "regions": ["all"],
"city": { "houses": { "railroad_city": 0 } }
}
]
-````
+```
### Disabling certain scenarios
-The `SCENARIO_BLACKLIST` can be either a blacklist or a whitelist.
-When it is a whitelist, it blacklists all scenarios but the ones specified.
-No more than one blacklist can be specified at one time - this is in all json loaded for a particular game (all mods + base game), not just your specific mod.
-The format is as follows:
+
+The `SCENARIO_BLACKLIST` can be either a blacklist or a whitelist. When it is a whitelist, it
+blacklists all scenarios but the ones specified. No more than one blacklist can be specified at one
+time - this is in all json loaded for a particular game (all mods + base game), not just your
+specific mod. The format is as follows:
+
```json
[
{
"type": "SCENARIO_BLACKLIST",
"subtype": "whitelist",
- "scenarios": [ "largebuilding" ]
+ "scenarios": ["largebuilding"]
}
]
```
-Valid values for `subtype` are `whitelist` and `blacklist`.
-`scenarios` is an array of the scenario ids that you want to blacklist or whitelist.
+
+Valid values for `subtype` are `whitelist` and `blacklist`. `scenarios` is an array of the scenario
+ids that you want to blacklist or whitelist.
## Important note on json files
-The following characters: `[ { , } ] : "` are *very* important when adding or modifying JSON files. This means a single missing `,` or `[` or `}` can be the difference between a working file and a hanging game at startup.
-If you want to include these characters in whatever you are doing (for instance, if you want to put a quote inside an item's description), you can do so by putting a backslash before the relevant character. This is known as "escaping" that character. For instance, I can make an item's description contain a quote if I want by doing this:
-````json
+The following characters: `[ { , } ] : "` are _very_ important when adding or modifying JSON files.
+This means a single missing `,` or `[` or `}` can be the difference between a working file and a
+hanging game at startup. If you want to include these characters in whatever you are doing (for
+instance, if you want to put a quote inside an item's description), you can do so by putting a
+backslash before the relevant character. This is known as "escaping" that character. For instance, I
+can make an item's description contain a quote if I want by doing this:
+
+```json
...
"description": "This is a shirt that says \"I wanna kill ALL the zombies\" on the front.",
...
-````
+```
In game, that appears like this:
`This is a shirt that says "I wanna kill ALL the zombies" on the front.`
-Many editors have features that let you track `{ [` and `] }` to see if they're balanced (ie, have a matching opposite); These editors will also respect escaped characters properly. [Notepad++](https://notepad-plus-plus.org/) is a popular, free editor on Windows that contains this feature. On Linux, there are a plethora of options, and you probably already have a preferred one 🙂
+Many editors have features that let you track `{ [` and `] }` to see if they're balanced (ie, have a
+matching opposite); These editors will also respect escaped characters properly.
+[Notepad++](https://notepad-plus-plus.org/) is a popular, free editor on Windows that contains this
+feature. On Linux, there are a plethora of options, and you probably already have a preferred one 🙂
-You may also want to use some online or offline JSON validator, e.g. https://dev.narc.ro/cataclysm/format.html , or `json_formatter` bundled with every release of the game (just drag-and-drop the .json file on it).
+You may also want to use some online or offline JSON validator, e.g.
+https://dev.narc.ro/cataclysm/format.html , or `json_formatter` bundled with every release of the
+game (just drag-and-drop the .json file on it).
## Addendum
+
+
### No Zombie Revival
-This mod is very simple, but it's worth a special section because it does things a little differently than other mods, and documentation on it is tricky to find.
-The entire mod can fit into fifteen lines of JSON, and it's presented below. Just copy/paste that into a modinfo.json file, and put it in a new folder in your mods directory.
-````json
+This mod is very simple, but it's worth a special section because it does things a little
+differently than other mods, and documentation on it is tricky to find.
+
+The entire mod can fit into fifteen lines of JSON, and it's presented below. Just copy/paste that
+into a modinfo.json file, and put it in a new folder in your mods directory.
+
+```json
[
{
"type": "MOD_INFO",
@@ -233,7 +327,7 @@ The entire mod can fit into fifteen lines of JSON, and it's presented below. Jus
"name": "Prevent Zombie Revivication",
"description": "Disables zombie revival.",
"category": "rebalance",
- "dependencies": [ "bn" ]
+ "dependencies": ["bn"]
},
{
"type": "monster_adjustment",
@@ -241,4 +335,4 @@ The entire mod can fit into fifteen lines of JSON, and it's presented below. Jus
"flag": { "name": "REVIVES", "value": false }
}
]
-````
+```
diff --git a/doc/MONSTERS.md b/doc/MONSTERS.md
index 55472d67f10d..3d3abcb5e960 100644
--- a/doc/MONSTERS.md
+++ b/doc/MONSTERS.md
@@ -1,6 +1,7 @@
# Monster types
Monster types are specified as JSON object with "type" member set to "MONSTER":
+
```JSON
{
"type": "MONSTER",
@@ -9,13 +10,16 @@ Monster types are specified as JSON object with "type" member set to "MONSTER":
}
```
-The id member should be the unique id of the type. It can be any string, by convention it has the prefix "mon_". This id can be referred to in various places, like monster groups or in mapgen to spawn specific monsters.
+The id member should be the unique id of the type. It can be any string, by convention it has the
+prefix "mon_". This id can be referred to in various places, like monster groups or in mapgen to
+spawn specific monsters.
For quantity strings (ie. volume, weight) use the largest unit you can keep full precision with.
Monster types support the following properties (mandatory, except if noted otherwise):
## "name"
+
(string or object)
```JSON
@@ -36,254 +40,330 @@ Name displayed in-game, and optionally the plural name and a translation context
If the plural name is not specified, it defaults to singular name + "s".
-Ctxt is used to help translators in case of homonyms (two different things with the same name). For example, pike the fish and pike the weapon.
+Ctxt is used to help translators in case of homonyms (two different things with the same name). For
+example, pike the fish and pike the weapon.
## "description"
+
(string)
In-game description for the monster.
## "categories"
+
(array of strings, optional)
-Monster categories. Can be NULL, CLASSIC (only mobs found in classic zombie movies) or WILDLIFE (natural animals). If they are not CLASSIC or WILDLIFE, they will not spawn in classic mode. One can add or remove entries in mods via "add:flags" and "remove:flags".
+Monster categories. Can be NULL, CLASSIC (only mobs found in classic zombie movies) or WILDLIFE
+(natural animals). If they are not CLASSIC or WILDLIFE, they will not spawn in classic mode. One can
+add or remove entries in mods via "add:flags" and "remove:flags".
## "species"
+
(array of strings, optional)
-A list of species ids. One can add or remove entries in mods via "add:species" and "remove:species", see Modding below. Properties (currently only triggers) from species are added to the properties of each monster that belong to the species.
+A list of species ids. One can add or remove entries in mods via "add:species" and "remove:species",
+see Modding below. Properties (currently only triggers) from species are added to the properties of
+each monster that belong to the species.
-In mainline game it can be HUMAN, ROBOT, ZOMBIE, MAMMAL, BIRD, FISH, REPTILE, WORM, MOLLUSK, AMPHIBIAN, INSECT, SPIDER, FUNGUS, PLANT, NETHER, MUTANT, BLOB, HORROR, ABERRATION, HALLUCINATION and UNKNOWN.
+In mainline game it can be HUMAN, ROBOT, ZOMBIE, MAMMAL, BIRD, FISH, REPTILE, WORM, MOLLUSK,
+AMPHIBIAN, INSECT, SPIDER, FUNGUS, PLANT, NETHER, MUTANT, BLOB, HORROR, ABERRATION, HALLUCINATION
+and UNKNOWN.
## "volume"
+
(string)
```JSON
"volume": "40 L"
```
-The numeric part of the string must be an integer. Accepts L, and ml as units. Note that l and mL are not currently accepted.
+
+The numeric part of the string must be an integer. Accepts L, and ml as units. Note that l and mL
+are not currently accepted.
## "weight"
+
(string)
```JSON
"weight": "3 kg"
```
-The numeric part of the string must be an integer. Use the largest unit you can keep full precision with. For example: 3 kg, not 3000 g. Accepts g and kg as units.
+
+The numeric part of the string must be an integer. Use the largest unit you can keep full precision
+with. For example: 3 kg, not 3000 g. Accepts g and kg as units.
## "scent_tracked"
+
(array of strings, optional)
List of scenttype_id tracked by this monster. scent_types are defined in scent_types.json
## "scent_ignored"
+
(array of strings, optional)
List of scenttype_id ignored by this monster. scent_types are defined in scent_types.json
## "symbol", "color"
+
(string)
-Symbol and color representing monster in-game. The symbol must be a UTF-8 string, that is exactly one console cell width (may be several Unicode characters). See [COLOR.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/COLOR.md) for details.
+Symbol and color representing monster in-game. The symbol must be a UTF-8 string, that is exactly
+one console cell width (may be several Unicode characters). See
+[COLOR.md](https://github.com/CleverRaven/Cataclysm-DDA/blob/master/doc/COLOR.md) for details.
## "size"
+
(string, optional)
Size flag, see JSON_FLAGS.md.
## "material"
+
(array of strings, optional)
-The materials the monster is primarily composed of. Must contain valid material ids. An empty array (which is the default) is also allowed, the monster is made of no specific material. One can add or remove entries in mods via "add:material" and "remove:material".
+The materials the monster is primarily composed of. Must contain valid material ids. An empty array
+(which is the default) is also allowed, the monster is made of no specific material. One can add or
+remove entries in mods via "add:material" and "remove:material".
## "phase"
+
(string, optional)
-It describes monster's body state of matter. However, it doesn't seem to have any gameplay purpose, right now.
-It can be SOLID, LIQUID, GAS, PLASMA or NULL.
+It describes monster's body state of matter. However, it doesn't seem to have any gameplay purpose,
+right now. It can be SOLID, LIQUID, GAS, PLASMA or NULL.
## "default_faction"
+
(string)
-The id of the faction the monster belongs to, this affects what other monsters it will fight. See Monster factions.
+The id of the faction the monster belongs to, this affects what other monsters it will fight. See
+Monster factions.
## "bodytype"
+
(string)
-The id of the monster's bodytype, which is a general description of the layout of the monster's body.
+The id of the monster's bodytype, which is a general description of the layout of the monster's
+body.
Value should be one of:
-| value | description
-| --- | ---
-| angel | a winged human
-| bear | a four legged animal that can stand on its hind legs
-| bird | a two legged animal with two wings
-| blob | a blob of material
-| crab | a multilegged animal with two large arms
-| quadruped | a four legged animal
-| elephant | a very large quadruped animal with a large head and torso with equal sized limbs
-| fish | an aquatic animal with a streamlined body and fins
-| flying insect | a six legged animal with a head and two body segments and wings
-| frog | a four legged animal with a neck and with very large rear legs and small forelegs
-| gator | a four legged animal with a very long body and short legs
-| human | a bipedal animal with two arms
-| insect | a six legged animal with a head and two body segments
-| kangaroo | a pentapedal animal that utilizes a large tail for stability with very large rear legs and smaller forearms
-| lizard | a smaller form of 'gator'
-| migo | whatever form migos have
-| pig | a four legged animal with the head in the same line as the body
-| spider | an eight legged animal with a small head on a large abdomen
-| snake | an animal with a long body and no limbs
+| value | description |
+| ------------- | ----------------------------------------------------------------------------------------------------------- |
+| angel | a winged human |
+| bear | a four legged animal that can stand on its hind legs |
+| bird | a two legged animal with two wings |
+| blob | a blob of material |
+| crab | a multilegged animal with two large arms |
+| quadruped | a four legged animal |
+| elephant | a very large quadruped animal with a large head and torso with equal sized limbs |
+| fish | an aquatic animal with a streamlined body and fins |
+| flying insect | a six legged animal with a head and two body segments and wings |
+| frog | a four legged animal with a neck and with very large rear legs and small forelegs |
+| gator | a four legged animal with a very long body and short legs |
+| human | a bipedal animal with two arms |
+| insect | a six legged animal with a head and two body segments |
+| kangaroo | a pentapedal animal that utilizes a large tail for stability with very large rear legs and smaller forearms |
+| lizard | a smaller form of 'gator' |
+| migo | whatever form migos have |
+| pig | a four legged animal with the head in the same line as the body |
+| spider | an eight legged animal with a small head on a large abdomen |
+| snake | an animal with a long body and no limbs |
## "attack_cost"
+
(integer, optional)
Number of moves per regular attack.
## "diff"
+
(integer, optional)
-Monster baseline difficulty. Impacts the shade used to label the monster, and if it is above 30 a kill will be recorded in the memorial log. Monster difficult is calculated based on expected melee damage, dodge, armor, hit points, speed, morale, aggression, and vision ranges. The calculation does not handle ranged special attacks or unique special attacks very well, and baseline difficulty can be used to account for that. Suggested values:
+Monster baseline difficulty. Impacts the shade used to label the monster, and if it is above 30 a
+kill will be recorded in the memorial log. Monster difficult is calculated based on expected melee
+damage, dodge, armor, hit points, speed, morale, aggression, and vision ranges. The calculation does
+not handle ranged special attacks or unique special attacks very well, and baseline difficulty can
+be used to account for that. Suggested values:
-| value | description
-| --- | ---
-| 2 | a limited defensive ability such as a skitterbot's taser, or a weak special like a shrieker zombie's special ability to alert nearby monsters, or a minor bonus to attack like poison or venom.
-| 5 | a limited ranged attack weaker than spitter zombie's spit, or a powerful defensive ability like a shocker zombie's zapback or an acid zombie's acid spray.
-| 10 | a powerful ranged attack, like a spitters zombie's spit or an turret's 9mm SMG.
-| 15 | a powerful ranged attack with additional hazards, like a corrosize zombie's spit
-| 20 | a very powerful ranged attack, like a laser turret or military turret's 5.56mm rifle, or a powerful special ability, like a zombie necromancer's ability to raise other zombies.
-| 30 | a ranged attack that is deadly even for armored characters, like an anti-material turret's .50 BMG rifle.
+| value | description |
+| ----- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| 2 | a limited defensive ability such as a skitterbot's taser, or a weak special like a shrieker zombie's special ability to alert nearby monsters, or a minor bonus to attack like poison or venom. |
+| 5 | a limited ranged attack weaker than spitter zombie's spit, or a powerful defensive ability like a shocker zombie's zapback or an acid zombie's acid spray. |
+| 10 | a powerful ranged attack, like a spitters zombie's spit or an turret's 9mm SMG. |
+| 15 | a powerful ranged attack with additional hazards, like a corrosize zombie's spit |
+| 20 | a very powerful ranged attack, like a laser turret or military turret's 5.56mm rifle, or a powerful special ability, like a zombie necromancer's ability to raise other zombies. |
+| 30 | a ranged attack that is deadly even for armored characters, like an anti-material turret's .50 BMG rifle. |
-Most monsters should have difficulty 0 - even dangerous monsters like a zombie hulk or razorclaw alpha. Difficulty should only be used for exceptional, ranged, special attacks.
+Most monsters should have difficulty 0 - even dangerous monsters like a zombie hulk or razorclaw
+alpha. Difficulty should only be used for exceptional, ranged, special attacks.
## "aggression"
+
(integer, optional)
-Defines how aggressive the monster is. Ranges from -99 (totally passive) to 100 (guaranteed hostility on detection)
+Defines how aggressive the monster is. Ranges from -99 (totally passive) to 100 (guaranteed
+hostility on detection)
## "morale"
+
(integer, optional)
-Monster morale. Defines how low monster HP can get before it retreats. This number is treated as % of their max HP.
+Monster morale. Defines how low monster HP can get before it retreats. This number is treated as %
+of their max HP.
## "speed"
+
(integer)
-Monster speed. 100 is the normal speed for a human being - higher values are faster and lower values are slower.
+Monster speed. 100 is the normal speed for a human being - higher values are faster and lower values
+are slower.
## "mountable_weight_ratio"
+
(float, optional)
-Used as the acceptable rider vs. mount weight percentage ratio. Defaults to "0.3", which means the mount is capable of carrying riders weighing <= 30% of the mount's weight.
+Used as the acceptable rider vs. mount weight percentage ratio. Defaults to "0.3", which means the
+mount is capable of carrying riders weighing <= 30% of the mount's weight.
## "melee_skill"
+
(integer, optional)
-Monster melee skill, ranges from 0 - 10, with 4 being an average mob. See GAME_BALANCE.txt for more examples
+Monster melee skill, ranges from 0 - 10, with 4 being an average mob. See GAME_BALANCE.txt for more
+examples
## "dodge"
+
(integer, optional)
Monster dodge skill. See GAME_BALANCE.txt for an explanation of dodge mechanics.
## "melee_damage"
+
(array of objects, optional)
-List of damage instances added to die roll on monster melee attack.
- - `damage_type` valid entries are : "true", "biological", "bash", "cut", "acid", "stab", "heat", "cold" and "electric".
- - `amount` amount of damage.
- - `armor_penetration` how much of the armor the damage instance ignores.
- - `armor_multiplier` is a multiplier on `armor_penetration`.
- - `damage_multiplier` is a multiplier on `amount`.
+List of damage instances added to die roll on monster melee attack. - `damage_type` valid entries
+are : "true", "biological", "bash", "cut", "acid", "stab", "heat", "cold" and "electric". - `amount`
+amount of damage. - `armor_penetration` how much of the armor the damage instance ignores. -
+`armor_multiplier` is a multiplier on `armor_penetration`. - `damage_multiplier` is a multiplier on
+`amount`.
+
```JSON
- "melee_damage": [ { "damage_type": "electric", "amount": 4.0, "armor_penetration": 1, "armor_multiplier": 1.2, "damage_multiplier": 1.4 } ],
+"melee_damage": [ { "damage_type": "electric", "amount": 4.0, "armor_penetration": 1, "armor_multiplier": 1.2, "damage_multiplier": 1.4 } ],
```
## "melee_dice", "melee_dice_sides"
+
(integer, optional)
-Number of dices and their sides that are rolled on monster melee attack. This defines the amount of bash damage.
+Number of dices and their sides that are rolled on monster melee attack. This defines the amount of
+bash damage.
## "grab_strength"
+
(integer, optional)
-Intensity of the grab effect applied by this monster. Defaults to 1, is only useful for monster with a GRAB special attack and the GRABS flag. A monster with grab_strength = n applies a grab as if it was n zombies. A player with max(Str,Dex)<=n has no chance of breaking that grab.
+Intensity of the grab effect applied by this monster. Defaults to 1, is only useful for monster with
+a GRAB special attack and the GRABS flag. A monster with grab_strength = n applies a grab as if it
+was n zombies. A player with max(Str,Dex)<=n has no chance of breaking that grab.
## "melee_cut"
+
(integer, optional)
Amount of cutting damage added to die roll on monster melee attack.
## "armor_bash", "armor_cut", "armor_stab", "armor_acid", "armor_fire"
+
(integer, optional)
Monster protection from bashing, cutting, stabbing, acid and fire damage.
## "vision_day", "vision_night"
+
(integer, optional)
Vision range in full daylight and in total darkness.
## "luminance"
+
(integer, optional)
Amount of light passively output by monster. Ranges from 0 to 10.
## "hp"
+
(integer)
Monster hit points.
## "death_drops"
+
(string or item group, optional)
-An item group that is used to spawn items when the monster dies. This can be an inlined item group, see ITEM_SPAWN.md. The default subtype is "distribution".
+An item group that is used to spawn items when the monster dies. This can be an inlined item group,
+see ITEM_SPAWN.md. The default subtype is "distribution".
## "death_function"
+
(array of strings, optional)
-How the monster behaves on death. See JSON_FLAGS.md for a list of possible functions. One can add or remove entries in mods via "add:death_function" and "remove:death_function".
+How the monster behaves on death. See JSON_FLAGS.md for a list of possible functions. One can add or
+remove entries in mods via "add:death_function" and "remove:death_function".
## "emit_field"
-(array of objects of emit_id and time_duration, optional)
-"emit_fields": [ { "emit_id": "emit_gum_web", "delay": "30 m" } ],
-What field the monster emits and how often it does so. Time duration can use strings: "1 h", "60 m", "3600 s" etc...
+(array of objects of emit_id and time_duration, optional) "emit_fields": [ { "emit_id":
+"emit_gum_web", "delay": "30 m" } ],
+
+What field the monster emits and how often it does so. Time duration can use strings: "1 h", "60 m",
+"3600 s" etc...
## "regenerates"
+
(integer, optional)
Number of hitpoints regenerated per turn.
## "regeneration_modifiers"
+
( array of objects consisting of effect, base_mod (float) and scaling_mod (float), optional )
"regeneration_modifiers": [ { "effect": "on_fire", "base_mod": -0.3, "scaling_mod": -0.15 } ],
-What effects (if any) affect the monster's regeneration positively/negatively.
-The mods stack additively (Intensity 2 on_fire produces a multiplier of 0.45) but are applied multiplicatively.
+What effects (if any) affect the monster's regeneration positively/negatively. The mods stack
+additively (Intensity 2 on_fire produces a multiplier of 0.45) but are applied multiplicatively.
## "regenerates_in_dark"
+
(boolean, optional)
Monster regenerates very quickly in poorly lit tiles.
## "regen_morale"
+
(boolean, optional)
Will stop fleeing if at max hp, and regen anger and morale.
## "special_attacks"
+
(array of special attack definitions, optional)
-Monster's special attacks. This should be an array, each element of it should be an object (new style) or an array (old style).
+Monster's special attacks. This should be an array, each element of it should be an object (new
+style) or an array (old style).
+
+The old style array should contain 2 elements: the id of the attack (see JSON_FLAGS.md for a list)
+and the cooldown for that attack. Example (grab attack every 10 turns):
-The old style array should contain 2 elements: the id of the attack (see JSON_FLAGS.md for a list) and the cooldown for that attack. Example (grab attack every 10 turns):
```JSON
"special_attacks": [ [ "GRAB", 10 ] ]
```
-The new style object should contain at least a "type" member (string) and "cooldown" member (integer). It may contain additional members as required by the specific type. Possible types are listed below. Example:
+The new style object should contain at least a "type" member (string) and "cooldown" member
+(integer). It may contain additional members as required by the specific type. Possible types are
+listed below. Example:
+
```JSON
"special_attacks": [
{ "type": "leap", "cooldown": 10, "max_range": 4 }
@@ -291,6 +371,7 @@ The new style object should contain at least a "type" member (string) and "coold
```
"special_attacks" may contain any mixture of old and new style entries:
+
```JSON
"special_attacks": [
[ "GRAB", 10 ],
@@ -298,41 +379,56 @@ The new style object should contain at least a "type" member (string) and "coold
]
```
-One can add entries with "add:death_function", which takes the same content as the "special_attacks" member and remove entries with "remove:death_function", which requires an array of attack types. Example:
+One can add entries with "add:death_function", which takes the same content as the "special_attacks"
+member and remove entries with "remove:death_function", which requires an array of attack types.
+Example:
+
```JSON
"remove:special_attacks": [ "GRAB" ],
"add:special_attacks": [ [ "SHRIEK", 20 ] ]
```
## "flags"
+
(array of strings, optional)
-Monster flags. See JSON_FLAGS.md for a full list. One can add or remove entries in mods via "add:flags" and "remove:flags".
+Monster flags. See JSON_FLAGS.md for a full list. One can add or remove entries in mods via
+"add:flags" and "remove:flags".
## "fear_triggers", "anger_triggers", "placate_triggers"
+
(array of strings, optional)
What makes the monster afraid / angry / what calms it. See JSON_FLAGS.md for a full list
## "revert_to_itype"
+
(string, optional)
-If not empty and a valid item id, the monster can be converted into this item by the player, when it's friendly. This is usually used for turrets and similar to revert the turret monster back into the turret item, which can be picked up and placed elsewhere.
+If not empty and a valid item id, the monster can be converted into this item by the player, when
+it's friendly. This is usually used for turrets and similar to revert the turret monster back into
+the turret item, which can be picked up and placed elsewhere.
## "starting_ammo"
+
(object, optional)
-An object containing ammo that newly spawned monsters start with. This is useful for a monster that has a special attack that consumes ammo. Example:
+An object containing ammo that newly spawned monsters start with. This is useful for a monster that
+has a special attack that consumes ammo. Example:
+
```JSON
"starting_ammo": { "9mm": 100, "40mm_frag": 100 }
```
## "upgrades"
+
(boolean or object, optional)
-Controls how this monster is upgraded over time. It can either be the single value `false` (which is the default and disables upgrading) or an object describing the upgrades.
+Controls how this monster is upgraded over time. It can either be the single value `false` (which is
+the default and disables upgrading) or an object describing the upgrades.
Example:
+
```JSON
"upgrades": {
"into_group": "GROUP_ZOMBIE_UPGRADE",
@@ -343,57 +439,70 @@ Example:
The upgrades object may have the following members:
### "half_life"
-(int)
-Time in which half of the monsters upgrade according to an approximated exponential progression. It is scaled with the evolution scaling factor which defaults to 4 days.
+
+(int) Time in which half of the monsters upgrade according to an approximated exponential
+progression. It is scaled with the evolution scaling factor which defaults to 4 days.
### "into_group"
-(string, optional)
-The upgraded monster's type is taken from the specified group. The cost in these groups is for an upgrade in the spawn process (related to the rare "replace_monster_group" and "new_monster_group_id" attributes of spawning groups).
+
+(string, optional) The upgraded monster's type is taken from the specified group. The cost in these
+groups is for an upgrade in the spawn process (related to the rare "replace_monster_group" and
+"new_monster_group_id" attributes of spawning groups).
### "into"
-(string, optional)
-The upgraded monster's type.
+
+(string, optional) The upgraded monster's type.
### "age_grow"
-(int, optional)
-Number of days needed for monster to change into another monster.
+
+(int, optional) Number of days needed for monster to change into another monster.
## "reproduction"
-(dictionary, optional)
-The monster's reproduction cycle, if any. Supports:
+
+(dictionary, optional) The monster's reproduction cycle, if any. Supports:
### "baby_monster"
-(string, optional)
-the id of the monster spawned on reproduction for monsters who give live births. You must declare either this or `baby_egg` for reproduction to work.
+
+(string, optional) the id of the monster spawned on reproduction for monsters who give live births.
+You must declare either this or `baby_egg` for reproduction to work.
### "baby_egg"
-(string, optional)
-The id of the egg type to spawn for egg-laying monsters. You must declare either this or "baby_monster" for reproduction to work.
+
+(string, optional) The id of the egg type to spawn for egg-laying monsters. You must declare either
+this or "baby_monster" for reproduction to work.
### "baby_count"
-(int)
-Number of new creatures or eggs to spawn on reproduction.
+
+(int) Number of new creatures or eggs to spawn on reproduction.
### "baby_timer"
-(int)
-Number of days between reproduction events.
+
+(int) Number of days between reproduction events.
## "baby_flags"
-(Array, optional)
-Designate seasons during which this monster is capable of reproduction. ie: `[ "SPRING", "SUMMER" ]`
+
+(Array, optional) Designate seasons during which this monster is capable of reproduction. ie:
+`[ "SPRING", "SUMMER" ]`
## "special_when_hit"
+
(array, optional)
-A special defense attack, triggered when the monster is attacked. It should contain an array with the id of the defense (see Monster defense attacks in JSON_FLAGS.md) and the chance for that defense to be actually triggered. Example:
+A special defense attack, triggered when the monster is attacked. It should contain an array with
+the id of the defense (see Monster defense attacks in JSON_FLAGS.md) and the chance for that defense
+to be actually triggered. Example:
+
```JSON
"special_when_hit": [ "ZAPBACK", 100 ]
```
## "attack_effs"
+
(array of objects, optional)
-A set of effects that may get applied to the attacked creature when the monster successfully attacks. Example:
+A set of effects that may get applied to the attacked creature when the monster successfully
+attacks. Example:
+
```JSON
"attack_effs": [
{
@@ -407,192 +516,251 @@ A set of effects that may get applied to the attacked creature when the monster
Each element of the array should be an object containing the following members:
### "id"
+
(string)
The id of the effect that is to be applied.
### "duration"
+
(integer, optional)
How long (in turns) the effect should last.
### "affect_hit_bp"
+
(boolean, optional)
Whether the effect should be applied to the hit body part instead of the one set below.
### "bp"
-(string, optional)
-The body part that where the effect is applied. The default is to apply the effect to the whole body. Note that some effects may require a specific body part (e.g. "hot") and others may require the whole body (e.g. "meth").
+(string, optional)
+The body part that where the effect is applied. The default is to apply the effect to the whole
+body. Note that some effects may require a specific body part (e.g. "hot") and others may require
+the whole body (e.g. "meth").
### "chance"
+
(integer, optional)
The chance of the effect getting applied.
-
-
# Modding
-Monster types can be overridden or modified in mods. To do so, one has to add an "edit-mode" member, which can contain either:
-- "create" (the default if the member does not exist), an error will be shown if a type with the given id already exists.
-- "override", an existing type (if any) with the given id will be removed and the new data will be loaded as a completely new type.
-- "modify", an existing type will be modified. If there is no type with the given id, an error will be shown.
+Monster types can be overridden or modified in mods. To do so, one has to add an "edit-mode" member,
+which can contain either:
-Mandatory properties (all that are not marked as optional) are only required if edit mode is "create" or "override".
+- "create" (the default if the member does not exist), an error will be shown if a type with the
+ given id already exists.
+- "override", an existing type (if any) with the given id will be removed and the new data will be
+ loaded as a completely new type.
+- "modify", an existing type will be modified. If there is no type with the given id, an error will
+ be shown.
+
+Mandatory properties (all that are not marked as optional) are only required if edit mode is
+"create" or "override".
Example (rename the zombie monster, leaves all other properties untouched):
+
```JSON
{
- "type": "MONSTER",
- "edit-mode": "modify",
- "id": "mon_zombie",
- "name": "clown"
+ "type": "MONSTER",
+ "edit-mode": "modify",
+ "id": "mon_zombie",
+ "name": "clown"
}
```
-The default edit mode ("create") is suitable for new types, if their id conflicts with the types from other mods or from the core data, an error will be shown. The edit mode "override" is suitable for re-defining a type from scratch, it ensures that all mandatory members are listed and leaves no traces of the previous definitions. Edit mode "modify" is for small changes, like adding a flag or removing a special attack.
-Modifying a type overrides the properties with the new values, this example sets the special attacks to contain *only* the "SHRIEK" attack:
+The default edit mode ("create") is suitable for new types, if their id conflicts with the types
+from other mods or from the core data, an error will be shown. The edit mode "override" is suitable
+for re-defining a type from scratch, it ensures that all mandatory members are listed and leaves no
+traces of the previous definitions. Edit mode "modify" is for small changes, like adding a flag or
+removing a special attack.
+
+Modifying a type overrides the properties with the new values, this example sets the special attacks
+to contain _only_ the "SHRIEK" attack:
+
```JSON
{
- "type": "MONSTER",
- "edit-mode": "modify",
- "id": "mon_zombie",
- "special_attacks": [ [ "SHRIEK", 20 ] ]
+ "type": "MONSTER",
+ "edit-mode": "modify",
+ "id": "mon_zombie",
+ "special_attacks": [["SHRIEK", 20]]
}
```
-Some properties allow adding and removing entries, as documented above, usually via members with the "add:"/"remove:" prefix.
-
+Some properties allow adding and removing entries, as documented above, usually via members with the
+"add:"/"remove:" prefix.
# Monster special attack types
+
The listed attack types can be as monster special attacks (see "special_attacks").
## "leap"
+
Makes the monster leap a few tiles. It supports the following additional properties:
### "max_range"
+
(Required) Maximal range of attack.
### "min_range"
+
(Required) Minimal range needed for attack.
### "allow_no_target"
+
This prevents monster from using the ability on empty space.
### "move_cost"
+
Turns needed to complete special attack. 100 move_cost with 100 speed is equal to 1 second/turn.
#### "min_consider_range", "max_consider_range"
+
Minimal range and maximal range to consider for using specific attack.
## "bite"
+
Makes monster use teeth to bite opponent. Some monsters can give infection by doing so.
### "damage_max_instance"
+
Max damage it can deal on one bite.
### "min_mul", "max_mul"
+
How hard is to get free of bite without killing attacker.
### "move_cost"
+
Turns needed to complete special attack. 100 move_cost with 100 speed is equal to 1 second/turn.
### "accuracy"
+
(Integer) How accurate it is. Not many monsters use it though.
### "no_infection_chance"
+
Chance to not give infection.
## "gun"
+
Fires a gun at a target. If friendly, will avoid harming the player.
### "gun_type"
+
(Required) Valid item id of a gun that will be used to perform the attack.
### "ammo_type"
-(Required) Valid item id of the ammo the gun will be loaded with.
-Monster should also have a "starting_ammo" field with this ammo.
-For example
+
+(Required) Valid item id of the ammo the gun will be loaded with. Monster should also have a
+"starting_ammo" field with this ammo. For example
+
```
"ammo_type" : "50bmg",
"starting_ammo" : {"50bmg":100}
```
### "max_ammo"
+
Cap on ammo. If ammo goes above this value for any reason, a debug message will be printed.
### "fake_str"
+
Strength stat of the fake NPC that will execute the attack. 8 if not specified.
### "fake_dex"
+
Dexterity stat of the fake NPC that will execute the attack. 8 if not specified.
### "fake_int"
+
Intelligence stat of the fake NPC that will execute the attack. 8 if not specified.
### "fake_per"
+
Perception stat of the fake NPC that will execute the attack. 8 if not specified.
### "fake_skills"
+
Array of 2 element arrays of skill id and skill level pairs.
### "move_cost"
+
Move cost of executing the attack
// If true, gives "grace period" to player
+
### "require_targeting_player"
-If true, the monster will need to "target" the player,
-wasting `targeting_cost` moves, putting the attack on cooldown and making warning sounds,
-unless it attacked something that needs to be targeted recently.
+
+If true, the monster will need to "target" the player, wasting `targeting_cost` moves, putting the
+attack on cooldown and making warning sounds, unless it attacked something that needs to be targeted
+recently.
### "require_targeting_npc"
+
As above, but with npcs.
### "require_targeting_monster"
+
As above, but with monsters.
### "targeting_timeout"
-Targeting status will be applied for this many turns.
-Note that targeting applies to turret, not targets.
+
+Targeting status will be applied for this many turns. Note that targeting applies to turret, not
+targets.
### "targeting_timeout_extend"
+
Successfully attacking will extend the targeting for this many turns. Can be negative.
### "targeting_cost"
-Move cost of targeting the player. Only applied if attacking the player and didn't target player within last 5 turns.
+
+Move cost of targeting the player. Only applied if attacking the player and didn't target player
+within last 5 turns.
### "no_crits"
-If true then attacking create will be unable to score ranged critical hit on target,
-also good hits no longer has high change to go to the head, instead there is high change to score hit to the body and limbs.
-If false then default behavior used - critical hits are allowed and good hit has high chance to reach head.
+
+If true then attacking create will be unable to score ranged critical hit on target, also good hits
+no longer has high change to go to the head, instead there is high change to score hit to the body
+and limbs. If false then default behavior used - critical hits are allowed and good hit has high
+chance to reach head.
### "laser_lock"
-If true and attacking a creature that isn't laser-locked but needs to be targeted,
-the monster will act as if it had no targeting status (and waste time targeting),
-the target will become laser-locked, and if the target is the player, it will cause a warning.
+
+If true and attacking a creature that isn't laser-locked but needs to be targeted, the monster will
+act as if it had no targeting status (and waste time targeting), the target will become
+laser-locked, and if the target is the player, it will cause a warning.
Laser-locking affects the target, but isn't tied to specific attacker.
### "range"
+
Maximum range at which targets will be acquired.
### "range_no_burst"
+
Maximum range at which targets will be attacked with a burst (if applicable).
### "burst_limit"
+
Limit on burst size.
### "description"
+
Description of the attack being executed if seen by the player.
### "targeting_sound"
+
Description of the sound made when targeting.
### "targeting_volume"
+
Volume of the sound made when targeting.
### "no_ammo_sound"
+
Description of the sound made when out of ammo.
diff --git a/doc/NPCs.md b/doc/NPCs.md
index 24c9d5a9b5c0..6e125f16bd7d 100644
--- a/doc/NPCs.md
+++ b/doc/NPCs.md
@@ -1,7 +1,10 @@
TODO: document the "npc" structure, used to load NPC template
# Writing dialogues
-Dialogues work like state machines. They start with a certain topic (the NPC says something), the player character can respond (choosing one of several responses), and that response sets the new talk topic. This goes on until the dialogue is finished, or the NPC turns hostile.
+
+Dialogues work like state machines. They start with a certain topic (the NPC says something), the
+player character can respond (choosing one of several responses), and that response sets the new
+talk topic. This goes on until the dialogue is finished, or the NPC turns hostile.
Note that it is perfectly fine to have a response that switches the topic back to itself.
@@ -9,15 +12,17 @@ NPC missions are controlled by a separate but related JSON structure and are doc
[the missions docs](MISSIONS_JSON.md).
Two topics are special:
+
- `TALK_DONE` ends the dialogue immediately.
- `TALK_NONE` goes to the previously talked about topic.
-
### Validating Dialogues
+
Keeping track of talk topics and making sure that all the topics referenced in responses are
-defined, and all defined topics are referenced in a response or an NPC's chat, is very tricky.
-There is a python script in `tools/dialogue_validator.py` that will map all topics to responses
-and vice versa. Invoke it with
+defined, and all defined topics are referenced in a response or an NPC's chat, is very tricky. There
+is a python script in `tools/dialogue_validator.py` that will map all topics to responses and vice
+versa. Invoke it with
+
```sh
python tools/dialogue_validator.py data/json/npcs/* data/json/npcs/Backgrounds/* data/json/npcs/refugee_center/*
```
@@ -27,39 +32,51 @@ If you are writing a mod with dialogue, you can add the paths to the mod's dialo
## Talk topics
Each topic consists of:
+
1. a topic id (e.g. `TALK_ARSONIST`)
2. a dynamic line, spoken by the NPC.
3. an optional list of effects that occur when the NPC speaks the dynamic line
4. a list of responses that can be spoken by the player character.
-5. a list of repeated responses that can be spoken by the player character, automatically generated if the player or NPC has items in a list of items.
+5. a list of repeated responses that can be spoken by the player character, automatically generated
+ if the player or NPC has items in a list of items.
-One can specify new topics in json. It is currently not possible to define the starting topic, so you have to add a response to some of the default topics (e.g. `TALK_STRANGER_FRIENDLY` or `TALK_STRANGER_NEUTRAL`) or to topics that can be reached somehow.
+One can specify new topics in json. It is currently not possible to define the starting topic, so
+you have to add a response to some of the default topics (e.g. `TALK_STRANGER_FRIENDLY` or
+`TALK_STRANGER_NEUTRAL`) or to topics that can be reached somehow.
Format:
+
```json
{
- "type": "talk_topic",
- "id": "TALK_ARSONIST",
- "dynamic_line": "What now?",
- "responses": [
- {
- "text": "I don't know either",
- "topic": "TALK_DONE"
- }
- ],
- "replace_built_in_responses": true
+ "type": "talk_topic",
+ "id": "TALK_ARSONIST",
+ "dynamic_line": "What now?",
+ "responses": [
+ {
+ "text": "I don't know either",
+ "topic": "TALK_DONE"
+ }
+ ],
+ "replace_built_in_responses": true
}
```
### type
+
Must always be there and must always be `"talk_topic"`.
### id
-The topic id can be one of the built-in topics or a new id. However, if several talk topics *in json* have the same id, the last topic definition will override the previous ones.
-The topic id can also be an array of strings. This is loaded as if several topics with the exact same content have been given in json, each associated with an id from the `id`, array. Note that loading from json will append responses and, if defined in json, override the `dynamic_line` and the `replace_built_in_responses` setting. This allows adding responses to several topics at once.
+The topic id can be one of the built-in topics or a new id. However, if several talk topics _in
+json_ have the same id, the last topic definition will override the previous ones.
+
+The topic id can also be an array of strings. This is loaded as if several topics with the exact
+same content have been given in json, each associated with an id from the `id`, array. Note that
+loading from json will append responses and, if defined in json, override the `dynamic_line` and the
+`replace_built_in_responses` setting. This allows adding responses to several topics at once.
This example adds the "I'm going now!" response to all the listed topics.
+
```C++
{
"type": "talk_topic",
@@ -75,23 +92,40 @@ This example adds the "I'm going now!" response to all the listed topics.
```
### dynamic_line
-The `dynamic_line` is the line spoken by the NPC. It is optional. If it is not defined and the topic has the same id as a built-in topic, the `dynamic_line` from that built-in topic will be used. Otherwise the NPC will say nothing. See the chapter about `dynamic_line` below for more details.
+
+The `dynamic_line` is the line spoken by the NPC. It is optional. If it is not defined and the topic
+has the same id as a built-in topic, the `dynamic_line` from that built-in topic will be used.
+Otherwise the NPC will say nothing. See the chapter about `dynamic_line` below for more details.
### speaker_effect
-The `speaker_effect` is an object or array of effects that will occur after the NPC speaks the `dynamic_line`, no matter which response the player chooses. See the chapter about Speaker Effects below for more details.
+
+The `speaker_effect` is an object or array of effects that will occur after the NPC speaks the
+`dynamic_line`, no matter which response the player chooses. See the chapter about Speaker Effects
+below for more details.
### response
-The `responses` entry is an array with possible responses. It must not be empty. Each entry must be a response object. See the chapter about Responses below for more details.
+
+The `responses` entry is an array with possible responses. It must not be empty. Each entry must be
+a response object. See the chapter about Responses below for more details.
### replace_built_in_responses
-`replace_built_in_responses` is an optional boolean that defines whether to dismiss the built-in responses for that topic (default is `false`). If there are no built-in responses, this won't do anything. If `true`, the built-in responses are ignored and only those from this definition in the current json are used. If `false`, the responses from the current json are used along with the built-in responses (if any).
+
+`replace_built_in_responses` is an optional boolean that defines whether to dismiss the built-in
+responses for that topic (default is `false`). If there are no built-in responses, this won't do
+anything. If `true`, the built-in responses are ignored and only those from this definition in the
+current json are used. If `false`, the responses from the current json are used along with the
+built-in responses (if any).
---
## dynamic_line
-A dynamic line can either be a simple string, or an complex object, or an array with `dynamic_line` entries. If it's an array, an entry will be chosen randomly every time the NPC needs it. Each entry has the same probability.
+
+A dynamic line can either be a simple string, or an complex object, or an array with `dynamic_line`
+entries. If it's an array, an entry will be chosen randomly every time the NPC needs it. Each entry
+has the same probability.
Example:
+
```json
"dynamic_line": [
"generic text",
@@ -102,65 +136,83 @@ Example:
]
```
-A complex `dynamic_line` usually contains several `dynamic_line` entry and some condition that determines which is used. If dynamic lines are not nested, they are processed in the order of the entries below. The possible types of lines follow.
+A complex `dynamic_line` usually contains several `dynamic_line` entry and some condition that
+determines which is used. If dynamic lines are not nested, they are processed in the order of the
+entries below. The possible types of lines follow.
-In all cases, `npc_` refers to the NPC, and `u_` refers to the player. Optional lines do not have to be defined, but the NPC should always have something to say. Entries are always parsed as `dynamic_line` and can be nested.
+In all cases, `npc_` refers to the NPC, and `u_` refers to the player. Optional lines do not have to
+be defined, but the NPC should always have something to say. Entries are always parsed as
+`dynamic_line` and can be nested.
#### Several lines joined together
-The dynamic line is a list of dynamic lines, all of which are displayed. The dynamic lines in the list are processed normally.
+
+The dynamic line is a list of dynamic lines, all of which are displayed. The dynamic lines in the
+list are processed normally.
+
```json
{
- "and": [
- {
- "npc_male": true,
- "yes": "I'm a man.",
- "no": "I'm a woman."
- },
- " ",
- {
- "u_female": true,
- "no": "You're a man.",
- "yes": "You're a woman."
- }
- ]
+ "and": [
+ {
+ "npc_male": true,
+ "yes": "I'm a man.",
+ "no": "I'm a woman."
+ },
+ " ",
+ {
+ "u_female": true,
+ "no": "You're a man.",
+ "yes": "You're a woman."
+ }
+ ]
}
```
#### A line to be translated with gender context
-The line is to be given a gender context for the NPC, player, or both, to aid
-translation in languages where that matters. For example:
+
+The line is to be given a gender context for the NPC, player, or both, to aid translation in
+languages where that matters. For example:
+
```json
{
- "gendered_line": "Thank you.",
- "relevant_genders": [ "npc" ]
+ "gendered_line": "Thank you.",
+ "relevant_genders": ["npc"]
}
```
+
("Thank you" is different for male and female speakers in e.g. Portuguese).
-Valid choices for entries in the `"relevant_genders"` list are `"npc"` and
-`"u"`.
+Valid choices for entries in the `"relevant_genders"` list are `"npc"` and `"u"`.
#### A randomly selected hint
+
The dynamic line will be randomly chosen from the hints snippets.
```json
{
- "give_hint": true
+ "give_hint": true
}
```
#### Based on a previously generated reason
-The dynamic line will be chosen from a reason generated by an earlier effect. The reason will be cleared. Use of it should be gated on the `"has_reason"` condition.
+
+The dynamic line will be chosen from a reason generated by an earlier effect. The reason will be
+cleared. Use of it should be gated on the `"has_reason"` condition.
```json
{
- "has_reason": { "use_reason": true },
- "no": "What is it, boss?"
+ "has_reason": { "use_reason": true },
+ "no": "What is it, boss?"
}
```
#### Based on any Dialogue condition
-The dynamic line will be chosen based on whether a single dialogue condition is true or false. Dialogue conditions cannot be chained via `"and"`, `"or"`, or `"not"`. If the condition is true, the `"yes"` response will be chosen and otherwise the `"no"` response will be chosen. Both the `'"yes"` and `"no"` reponses are optional. Simple string conditions may be followed by `"true"` to make them fields in the dynamic line dictionary, or they can be followed by the response that will be chosen if the condition is true and the `"yes"` response can be omitted.
+
+The dynamic line will be chosen based on whether a single dialogue condition is true or false.
+Dialogue conditions cannot be chained via `"and"`, `"or"`, or `"not"`. If the condition is true, the
+`"yes"` response will be chosen and otherwise the `"no"` response will be chosen. Both the `'"yes"`
+and `"no"` reponses are optional. Simple string conditions may be followed by `"true"` to make them
+fields in the dynamic line dictionary, or they can be followed by the response that will be chosen
+if the condition is true and the `"yes"` response can be omitted.
```json
{
@@ -193,12 +245,18 @@ The dynamic line will be chosen based on whether a single dialogue condition is
}
}
```
+
---
## Speaker Effects
-The `speaker_effect` entry contains dialogue effects that occur after the NPC speaks the `dynamic_line` but before the player responds and regardless of the player response. Each effect can have an optional condition, and will only be applied if the condition is true. Each `speaker_effect` can also have an optional `sentinel`, which guarantees the effect will only run once.
+
+The `speaker_effect` entry contains dialogue effects that occur after the NPC speaks the
+`dynamic_line` but before the player responds and regardless of the player response. Each effect can
+have an optional condition, and will only be applied if the condition is true. Each `speaker_effect`
+can also have an optional `sentinel`, which guarantees the effect will only run once.
Format:
+
```json
"speaker_effect": {
"sentinel": "...",
@@ -206,7 +264,9 @@ Format:
"effect": "..."
}
```
+
or:
+
```json
"speaker_effect": [
{
@@ -222,70 +282,95 @@ or:
]
```
-The `sentinel` can be any string, but sentinels are unique to each `TALK_TOPIC`. If there are multiple `speaker_effect`s within the `TALK_TOPIC`, they should have different sentinels. Sentinels are not required, but since the `speaker_effect` will run every time the dialogue returns to the `TALK_TOPIC`, they are highly encouraged to avoid inadvertently repeating the same effects.
+The `sentinel` can be any string, but sentinels are unique to each `TALK_TOPIC`. If there are
+multiple `speaker_effect`s within the `TALK_TOPIC`, they should have different sentinels. Sentinels
+are not required, but since the `speaker_effect` will run every time the dialogue returns to the
+`TALK_TOPIC`, they are highly encouraged to avoid inadvertently repeating the same effects.
-The `effect` can be any legal effect, as described below. The effect can be a simple string, object, or an array of strings and objects, as normal for objects.
+The `effect` can be any legal effect, as described below. The effect can be a simple string, object,
+or an array of strings and objects, as normal for objects.
-The optional `condition` can be any legal condition, as described below. If a `condition` is present, the `effect` will only occur if the `condition` is true.
+The optional `condition` can be any legal condition, as described below. If a `condition` is
+present, the `effect` will only occur if the `condition` is true.
-Speaker effects are useful for setting status variables to indicate that player has talked to the NPC without complicating the responses with multiple effect variables. They can also be used, with a sentinel, to run a `mapgen_update` effect the first time the player hears some dialogue from the NPC.
+Speaker effects are useful for setting status variables to indicate that player has talked to the
+NPC without complicating the responses with multiple effect variables. They can also be used, with a
+sentinel, to run a `mapgen_update` effect the first time the player hears some dialogue from the
+NPC.
---
+
## Responses
-A response contains at least a text, which is display to the user and "spoken" by the player character (its content has no meaning for the game) and a topic to which the dialogue will switch to. It can also have a trial object which can be used to either lie, persuade or intimidate the NPC, see below for details. There can be different results, used either when the trial succeeds and when it fails.
+
+A response contains at least a text, which is display to the user and "spoken" by the player
+character (its content has no meaning for the game) and a topic to which the dialogue will switch
+to. It can also have a trial object which can be used to either lie, persuade or intimidate the NPC,
+see below for details. There can be different results, used either when the trial succeeds and when
+it fails.
Format:
+
```json
{
- "text": "I, the player, say to you...",
- "condition": "...something...",
- "trial": {
- "type": "PERSUADE",
- "difficulty": 10
- },
- "success": {
- "topic": "TALK_DONE",
- "effect": "...",
- "opinion": {
- "trust": 0,
- "fear": 0,
- "value": 0,
- "anger": 0,
- "owed": 0,
- "favors": 0
- }
- },
- "failure": {
- "topic": "TALK_DONE"
+ "text": "I, the player, say to you...",
+ "condition": "...something...",
+ "trial": {
+ "type": "PERSUADE",
+ "difficulty": 10
+ },
+ "success": {
+ "topic": "TALK_DONE",
+ "effect": "...",
+ "opinion": {
+ "trust": 0,
+ "fear": 0,
+ "value": 0,
+ "anger": 0,
+ "owed": 0,
+ "favors": 0
}
+ },
+ "failure": {
+ "topic": "TALK_DONE"
+ }
}
```
Alternatively a short format:
+
```json
{
- "text": "I, the player, say to you...",
- "effect": "...",
- "topic": "TALK_WHATEVER"
+ "text": "I, the player, say to you...",
+ "effect": "...",
+ "topic": "TALK_WHATEVER"
}
```
+
The short format is equivalent to (an unconditional switching of the topic, `effect` is optional):
+
```json
{
- "text": "I, the player, say to you...",
- "trial": {
- "type": "NONE"
- },
- "success": {
- "effect": "...",
- "topic": "TALK_WHATEVER"
- }
+ "text": "I, the player, say to you...",
+ "trial": {
+ "type": "NONE"
+ },
+ "success": {
+ "effect": "...",
+ "topic": "TALK_WHATEVER"
+ }
}
```
-The optional boolean keys "switch" and "default" are false by default. Only the first response with `"switch": true`, `"default": false`, and a valid condition will be displayed, and no other responses with `"switch": true` will be displayed. If no responses with `"switch": true` and `"default": false` are displayed, then any and all responses with `"switch": true` and `"default": true` will be displayed. In either case, all responses that have `"switch": false` (whether or not they have `"default": true` is set) will be displayed as long their conditions are satisfied.
+The optional boolean keys "switch" and "default" are false by default. Only the first response with
+`"switch": true`, `"default": false`, and a valid condition will be displayed, and no other
+responses with `"switch": true` will be displayed. If no responses with `"switch": true` and
+`"default": false` are displayed, then any and all responses with `"switch": true` and
+`"default": true` will be displayed. In either case, all responses that have `"switch": false`
+(whether or not they have `"default": true` is set) will be displayed as long their conditions are
+satisfied.
#### switch and default Example
+
```json
"responses": [
{ "text": "You know what, never mind.", "topic": "TALK_NONE" },
@@ -300,51 +385,77 @@ The optional boolean keys "switch" and "default" are false by default. Only the
{ "text": "Gotta go!", "topic": "TALK_DONE" }
]
```
+
The player will always have the option to return to a previous topic or end the conversation, and
-will otherwise have the option to give a $500, $50, or $5 bribe if they have the funds. If they
+will otherwise have the option to give a $500, $50, or $5 bribe if they have the funds. If they
don't have at least $50, they will also have the option to provide some other bribe.
### truefalsetext
-The player will have one response text if a condition is true, and another if it is false, but the same trial for either line. `condition`, `true`, and `false` are all mandatory.
+
+The player will have one response text if a condition is true, and another if it is false, but the
+same trial for either line. `condition`, `true`, and `false` are all mandatory.
```json
{
- "truefalsetext": {
- "condition": { "u_has_item": "FMCNote" },
- "true": "I may have the money, I'm not giving you any.",
- "false": "I don't have that money."
- },
- "topic": "TALK_WONT_PAY"
+ "truefalsetext": {
+ "condition": { "u_has_item": "FMCNote" },
+ "true": "I may have the money, I'm not giving you any.",
+ "false": "I don't have that money."
+ },
+ "topic": "TALK_WONT_PAY"
}
```
### text
+
Will be shown to the user, no further meaning.
### trial
-Optional, if not defined, `"NONE"` is used. Otherwise one of `"NONE"`, `"LIE"`, `"PERSUADE"`, `"INTIMIDATE"`, or `"CONDITION"`. If `"NONE"` is used, the `failure` object is not read, otherwise it's mandatory.
-The `difficulty` is only required if type is not `"NONE"` or `"CONDITION"` and specifies the success chance in percent (it is however modified by various things like mutations). Higher difficulties are easier to pass.
+Optional, if not defined, `"NONE"` is used. Otherwise one of `"NONE"`, `"LIE"`, `"PERSUADE"`,
+`"INTIMIDATE"`, or `"CONDITION"`. If `"NONE"` is used, the `failure` object is not read, otherwise
+it's mandatory.
-An optional `mod` array takes any of the following modifiers and increases the difficulty by the NPC's opinion of your character or personality trait for that modifier multiplied by the value: `"ANGER"`, `"FEAR"`, `"TRUST"`, `"VALUE"`, `"AGRESSION"`, `"ALTRUISM"`, `"BRAVERY"`, `"COLLECTOR"`. The special `"POS_FEAR"` modifier treats NPC's fear of your character below 0 as though it were 0. The special `"TOTAL"` modifier sums all previous modifiers and then multiplies the result by its value and is used when setting the owed value.
+The `difficulty` is only required if type is not `"NONE"` or `"CONDITION"` and specifies the success
+chance in percent (it is however modified by various things like mutations). Higher difficulties are
+easier to pass.
-`"CONDITION"` trials take a mandatory `condition` instead of `difficulty`. The `success` object is chosen if the `condition` is true and the `failure` is chosen otherwise.
+An optional `mod` array takes any of the following modifiers and increases the difficulty by the
+NPC's opinion of your character or personality trait for that modifier multiplied by the value:
+`"ANGER"`, `"FEAR"`, `"TRUST"`, `"VALUE"`, `"AGRESSION"`, `"ALTRUISM"`, `"BRAVERY"`, `"COLLECTOR"`.
+The special `"POS_FEAR"` modifier treats NPC's fear of your character below 0 as though it were 0.
+The special `"TOTAL"` modifier sums all previous modifiers and then multiplies the result by its
+value and is used when setting the owed value.
+
+`"CONDITION"` trials take a mandatory `condition` instead of `difficulty`. The `success` object is
+chosen if the `condition` is true and the `failure` is chosen otherwise.
### success and failure
-Both objects have the same structure. `topic` defines which topic the dialogue will switch to. `opinion` is optional, if given it defines how the opinion of the NPC will change. The given values are *added* to the opinion of the NPC, they are all optional and default to 0. `effect` is a function that is executed after choosing the response, see below.
+
+Both objects have the same structure. `topic` defines which topic the dialogue will switch to.
+`opinion` is optional, if given it defines how the opinion of the NPC will change. The given values
+are _added_ to the opinion of the NPC, they are all optional and default to 0. `effect` is a
+function that is executed after choosing the response, see below.
The opinion of the NPC affects several aspects of the interaction with NPCs:
+
- Higher trust makes it easier to lie and persuade, and it usually a good thing.
-- Higher fear makes it easier to intimidate, but the NPC may flee from you (and will not talk to you).
-- Higher value makes it easier to persuade them and to give them orders, it's a kind of a friendship indicator.
-- High anger value (about 20 points more than fear, but this also depends on the NPCs personality) makes the NPC hostile and is usually a bad thing.
-The combination of fear and trust decide together with the personality of the NPC the initial talk topic (`"TALK_MUG"`, `"TALK_STRANGER_AGGRESSIVE"`, `"TALK_STRANGER_SCARED"`, `"TALK_STRANGER_WARY"`, `"TALK_STRANGER_FRIENDLY"`, or `"TALK_STRANGER_NEUTRAL"`).
+- Higher fear makes it easier to intimidate, but the NPC may flee from you (and will not talk to
+ you).
+- Higher value makes it easier to persuade them and to give them orders, it's a kind of a friendship
+ indicator.
+- High anger value (about 20 points more than fear, but this also depends on the NPCs personality)
+ makes the NPC hostile and is usually a bad thing. The combination of fear and trust decide
+ together with the personality of the NPC the initial talk topic (`"TALK_MUG"`,
+ `"TALK_STRANGER_AGGRESSIVE"`, `"TALK_STRANGER_SCARED"`, `"TALK_STRANGER_WARY"`,
+ `"TALK_STRANGER_FRIENDLY"`, or `"TALK_STRANGER_NEUTRAL"`).
For the actual usage of that data, search the source code for `"op_of_u"`.
The `failure` object is used if the trial fails, the `success` object is used otherwise.
### Sample trials
+
```json
"trial": { "type": "PERSUADE", "difficulty": 0, "mod": [ [ "TRUST", 3 ], [ "VALUE", 3 ], [ "ANGER", -3 ] ] }
"trial": { "type": "INTIMIDATE", "difficulty": 20, "mod": [ [ "FEAR", 8 ], [ "VALUE", 2 ], [ "TRUST", 2 ], [ "BRAVERY", -2 ] ] }
@@ -365,145 +476,172 @@ The `failure` object is used if the trial fails, the `success` object is used ot
```
### condition
-This is an optional condition which can be used to prevent the response under certain circumstances. If not defined, it defaults to always `true`. If the condition is not met, the response is not included in the list of possible responses. For possible content see Dialogue Conditions below.
+
+This is an optional condition which can be used to prevent the response under certain circumstances.
+If not defined, it defaults to always `true`. If the condition is not met, the response is not
+included in the list of possible responses. For possible content see Dialogue Conditions below.
---
## Repeat Responses
-Repeat responses are responses that should be added to the response list multiple times, once for each instance of an item.
+
+Repeat responses are responses that should be added to the response list multiple times, once for
+each instance of an item.
A repeat response has the following format:
+
```json
{
"for_item": [
- "jerky", "meat_smoked", "fish_smoked", "cooking_oil", "cooking_oil2", "cornmeal", "flour",
- "fruit_wine", "beer", "sugar"
+ "jerky",
+ "meat_smoked",
+ "fish_smoked",
+ "cooking_oil",
+ "cooking_oil2",
+ "cornmeal",
+ "flour",
+ "fruit_wine",
+ "beer",
+ "sugar"
],
"response": { "text": "Delivering .", "topic": "TALK_DELIVER_ASK" }
}
```
-`"response"` is mandatory and must be a standard dialogue response, as described above. `"switch"` is allowed in repeat responses and works normally.
+`"response"` is mandatory and must be a standard dialogue response, as described above. `"switch"`
+is allowed in repeat responses and works normally.
-One of `"for_item"` or `"for_category"`, and each can either be a single string or list of items or item categories. The `response` is generated for each item in the list in the player or NPC's inventory.
+One of `"for_item"` or `"for_category"`, and each can either be a single string or list of items or
+item categories. The `response` is generated for each item in the list in the player or NPC's
+inventory.
-`"is_npc"` is an optional bool value, and if it is present, the NPC's inventory list is checked. By default, the player's inventory list is checked.
+`"is_npc"` is an optional bool value, and if it is present, the NPC's inventory list is checked. By
+default, the player's inventory list is checked.
-`"include_containers"` is an optional bool value, and if it is present, items containing an item will generate separate responses from the item itself.
+`"include_containers"` is an optional bool value, and if it is present, items containing an item
+will generate separate responses from the item itself.
---
## Dialogue Effects
-The `effect` field of `speaker_effect` or a `response` can be any of the following effects. Multiple effects should be arranged in a list and are processed in the order listed.
+
+The `effect` field of `speaker_effect` or a `response` can be any of the following effects. Multiple
+effects should be arranged in a list and are processed in the order listed.
#### Missions
-Effect | Description
----|---
-`assign_mission` | Assigns a previously selected mission to your character.
-`mission_success` | Resolves the current mission successfully.
-`mission_failure` | Resolves the current mission as a failure.
-`clear_mission` | Clears the mission from the your character's assigned missions.
-`mission_reward` | Gives the player the mission's reward.
+| Effect | Description |
+| ----------------- | --------------------------------------------------------------- |
+| `assign_mission` | Assigns a previously selected mission to your character. |
+| `mission_success` | Resolves the current mission successfully. |
+| `mission_failure` | Resolves the current mission as a failure. |
+| `clear_mission` | Clears the mission from the your character's assigned missions. |
+| `mission_reward` | Gives the player the mission's reward. |
#### Stats / Morale
-Effect | Description
----|---
-`give_aid` | Removes all bites, infection, and bleeding from your character's body and heals 10-25 HP of injury on each of your character's body parts. Takes 30 minutes. NPC receives 30 minutes of `currently_busy` effect on start.
-`give_aid_all` | Performs `give_aid` on your character and each of your character's NPC allies in crafting range. Takes 1 hour. NPC receives 1 hour of `currently_busy` effect on start.
-`buy_haircut` | Gives your character a haircut morale boost for 12 hours.
-`buy_shave` | Gives your character a shave morale boost for 6 hours.
-`morale_chat` | Gives your character a pleasant conversation morale boost for 6 hours.
-`player_weapon_away` | Makes your character put away (unwield) their weapon.
-`player_weapon_drop` | Makes your character drop their weapon.
+| Effect | Description |
+| -------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `give_aid` | Removes all bites, infection, and bleeding from your character's body and heals 10-25 HP of injury on each of your character's body parts. Takes 30 minutes. NPC receives 30 minutes of `currently_busy` effect on start. |
+| `give_aid_all` | Performs `give_aid` on your character and each of your character's NPC allies in crafting range. Takes 1 hour. NPC receives 1 hour of `currently_busy` effect on start. |
+| `buy_haircut` | Gives your character a haircut morale boost for 12 hours. |
+| `buy_shave` | Gives your character a shave morale boost for 6 hours. |
+| `morale_chat` | Gives your character a pleasant conversation morale boost for 6 hours. |
+| `player_weapon_away` | Makes your character put away (unwield) their weapon. |
+| `player_weapon_drop` | Makes your character drop their weapon. |
#### Character effects / Mutations
-Effect | Description
----|---
-`u_add_effect: effect_string`, (*one of* `duration: duration_string`, `duration: duration_int`)
`npc_add_effect: effect_string`, (*one of* `duration: duration_string`, `duration: duration_int`) | Your character or the NPC will gain the effect for `duration_string` or `duration_int` turns. If `duration_string` is `"PERMANENT"`, the effect will be added permanently.
-`u_add_trait: trait_string`
`npc_add_trait: trait_string` | Your character or the NPC will gain the trait.
-`u_lose_effect: effect_string`
`npc_lose_effect: effect_string` | Your character or the NPC will lose the effect if they have it.
-`u_lose_trait: trait_string`
`npc_lose_trait: trait_string` | Your character or the NPC will lose the trait.
-`u_add_var, npc_add_var`: `var_name, type: type_str`, `context: context_str`, `value: value_str` | Your character or the NPC will store `value_str` as a variable that can be later retrieved by `u_has_var` or `npc_has_var`. `npc_add_var` can be used to store arbitrary local variables, and `u_add_var` can be used to store arbitrary "global" variables, and should be used in preference to setting effects.
-`u_lose_var`, `npc_lose_var`: `var_name`, `type: type_str`, `context: context_str` | Your character or the NPC will clear any stored variable that has the same `var_name`, `type_str`, and `context_str`.
-`u_adjust_var, npc_adjust_var`: `var_name, type: type_str`, `context: context_str`, `adjustment: adjustment_num` | Your character or the NPC will adjust the stored variable by `adjustment_num`.
-`barber_hair` | Opens a menu allowing the player to choose a new hair style.
-`barber_beard` | Opens a menu allowing the player to choose a new beard style.
-`u_learn_recipe: recipe_string` | Your character will learn and memorize the recipe `recipe_string`.
-`npc_first_topic: topic_string` | NPC permanently changes first dialogue topic to `topic_string`.
+| Effect | Description |
+| ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `u_add_effect: effect_string`, (_one of_ `duration: duration_string`, `duration: duration_int`)
`npc_add_effect: effect_string`, (_one of_ `duration: duration_string`, `duration: duration_int`) | Your character or the NPC will gain the effect for `duration_string` or `duration_int` turns. If `duration_string` is `"PERMANENT"`, the effect will be added permanently. |
+| `u_add_trait: trait_string`
`npc_add_trait: trait_string` | Your character or the NPC will gain the trait. |
+| `u_lose_effect: effect_string`
`npc_lose_effect: effect_string` | Your character or the NPC will lose the effect if they have it. |
+| `u_lose_trait: trait_string`
`npc_lose_trait: trait_string` | Your character or the NPC will lose the trait. |
+| `u_add_var, npc_add_var`: `var_name, type: type_str`, `context: context_str`, `value: value_str` | Your character or the NPC will store `value_str` as a variable that can be later retrieved by `u_has_var` or `npc_has_var`. `npc_add_var` can be used to store arbitrary local variables, and `u_add_var` can be used to store arbitrary "global" variables, and should be used in preference to setting effects. |
+| `u_lose_var`, `npc_lose_var`: `var_name`, `type: type_str`, `context: context_str` | Your character or the NPC will clear any stored variable that has the same `var_name`, `type_str`, and `context_str`. |
+| `u_adjust_var, npc_adjust_var`: `var_name, type: type_str`, `context: context_str`, `adjustment: adjustment_num` | Your character or the NPC will adjust the stored variable by `adjustment_num`. |
+| `barber_hair` | Opens a menu allowing the player to choose a new hair style. |
+| `barber_beard` | Opens a menu allowing the player to choose a new beard style. |
+| `u_learn_recipe: recipe_string` | Your character will learn and memorize the recipe `recipe_string`. |
+| `npc_first_topic: topic_string` | NPC permanently changes first dialogue topic to `topic_string`. |
#### Trade / Items
-Effect | Description
----|---
-`start_trade` | Opens the trade screen and allows trading with the NPC.
-`buy_10_logs` | Places 10 logs in the ranch garage, and makes the NPC unavailable for 1 day.
-`buy_100_logs` | Places 100 logs in the ranch garage, and makes the NPC unavailable for 7 days.
-`give_equipment` | Allows your character to select items from the NPC's inventory and transfer them to your inventory.
-`npc_gets_item` | Allows your character to select an item from your character's inventory and transfer it to the NPC's inventory. The NPC will not accept it if they do not have space or weight to carry it, and will set a reason that can be referenced in a future dynamic line with `"use_reason"`.
-`npc_gets_item_to_use` | Allow your character to select an item from your character's inventory and transfer it to the NPC's inventory. The NPC will attempt to wield it and will not accept it if it is too heavy or is an inferior weapon to what they are currently using, and will set a reason that can be referenced in a future dynamic line with `"use_reason"`.
-`u_buy_item: item_string`, (*optional* `cost: cost_num`, *optional* `count: count_num`, *optional* `container: container_string`) | The NPC will give your character the item or `count_num` copies of the item, contained in container, and will subtract `cost_num` from `op_of_u.owed` if specified. If the `op_o_u.owed` is less than `cost_num`, the trade window will open and the player will have to trade to make up the difference; the NPC will not give the player the item unless `cost_num` is satisfied.
If cost isn't present, the NPC gives your character the item at no charge.
-`u_sell_item: item_string`, (*optional* `cost: cost_num`, *optional* `count: count_num`) | Your character will give the NPC the item or `count_num` copies of the item, and will add `cost_num` to the NPC's `op_of_u.owed` if specified.
If cost isn't present, the your character gives the NPC the item at no charge.
This effect will fail if you do not have at least `count_num` copies of the item, so it should be checked with `u_has_items`.
-`u_bulk_trade_accept`
`npc_bulk_trade_accept` | Only valid after a `repeat_response`. The player trades all instances of the item from the `repeat_response` with the NPC. For `u_bulk_trade_accept`, the player loses the items from their inventory and gains the same value of the NPC's faction currecy; for `npc_bulk_trade_accept`, the player gains the items from the NPC's inventory and loses the same value of the NPC's faction currency. If there is remaining value, or the NPC doesn't have a faction currency, the remainder goes into the NPC's `op_of_u.owed`.
-`u_bulk_donate`
`npc_bulk_donate` | Only valid after a `repeat_response`. The player or NPC transfers all instances of the item from the `repeat_response`. For `u_bulk_donate`, the player loses the items from their inventory and the NPC gains them; for `npc_bulk_donate`, the player gains the items from the NPC's inventory and the NPC loses them.
-`u_spend_ecash: amount` | Remove `amount` from your character's pre-cataclysm bank account. Negative values means your character gains e-cash. NPCs should **not** deal in e-cash, only personal debts and items (including faction currency).
-`add_debt: mod_list` | Increases the NPC's debt to the player by the values in the `mod_list`.
The following would increase the NPC's debt to the player by 1500x the NPC's altruism and 1000x the NPC's opinion of the player's value: `{ "effect": { "add_debt": [ [ "ALTRUISM", 3 ], [ "VALUE", 2 ], [ "TOTAL", 500 ] ] } }`
-`u_consume_item`, `npc_consume_item: item_string`, (*optional* `count: count_num`) | You or the NPC will delete the item or `count_num` copies of the item from their inventory.
This effect will fail if the you or NPC does not have at least `count_num` copies of the item, so it should be checked with `u_has_items` or `npc_has_items`.
-`u_remove_item_with`, `npc_remove_item_with: item_string` | You or the NPC will delete any instances of item in inventory.
This is an unconditional remove and will not fail if you or the NPC does not have the item.
-`u_buy_monster: monster_type_string`, (*optional* `cost: cost_num`, *optional* `count: count_num`, *optional* `name: name_string`, *optional* `pacified: pacified_bool`) | The NPC will give your character `count_num` (default 1) instances of the monster as pets and will subtract `cost_num` from `op_of_u.owed` if specified. If the `op_o_u.owed` is less than `cost_num`, the trade window will open and the player will have to trade to make up the difference; the NPC will not give the player the item unless `cost_num` is satisfied.
If cost isn't present, the NPC gives your character the item at no charge.
If `name_string` is specified the monster(s) will have the specified name. If `pacified_bool` is set to true, the monster will have the pacified effect applied.
-
+| Effect | Description |
+| ------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `start_trade` | Opens the trade screen and allows trading with the NPC. |
+| `buy_10_logs` | Places 10 logs in the ranch garage, and makes the NPC unavailable for 1 day. |
+| `buy_100_logs` | Places 100 logs in the ranch garage, and makes the NPC unavailable for 7 days. |
+| `give_equipment` | Allows your character to select items from the NPC's inventory and transfer them to your inventory. |
+| `npc_gets_item` | Allows your character to select an item from your character's inventory and transfer it to the NPC's inventory. The NPC will not accept it if they do not have space or weight to carry it, and will set a reason that can be referenced in a future dynamic line with `"use_reason"`. |
+| `npc_gets_item_to_use` | Allow your character to select an item from your character's inventory and transfer it to the NPC's inventory. The NPC will attempt to wield it and will not accept it if it is too heavy or is an inferior weapon to what they are currently using, and will set a reason that can be referenced in a future dynamic line with `"use_reason"`. |
+| `u_buy_item: item_string`, (_optional_ `cost: cost_num`, _optional_ `count: count_num`, _optional_ `container: container_string`) | The NPC will give your character the item or `count_num` copies of the item, contained in container, and will subtract `cost_num` from `op_of_u.owed` if specified. If the `op_o_u.owed` is less than `cost_num`, the trade window will open and the player will have to trade to make up the difference; the NPC will not give the player the item unless `cost_num` is satisfied.
If cost isn't present, the NPC gives your character the item at no charge. |
+| `u_sell_item: item_string`, (_optional_ `cost: cost_num`, _optional_ `count: count_num`) | Your character will give the NPC the item or `count_num` copies of the item, and will add `cost_num` to the NPC's `op_of_u.owed` if specified.
If cost isn't present, the your character gives the NPC the item at no charge.
This effect will fail if you do not have at least `count_num` copies of the item, so it should be checked with `u_has_items`. |
+| `u_bulk_trade_accept`
`npc_bulk_trade_accept` | Only valid after a `repeat_response`. The player trades all instances of the item from the `repeat_response` with the NPC. For `u_bulk_trade_accept`, the player loses the items from their inventory and gains the same value of the NPC's faction currecy; for `npc_bulk_trade_accept`, the player gains the items from the NPC's inventory and loses the same value of the NPC's faction currency. If there is remaining value, or the NPC doesn't have a faction currency, the remainder goes into the NPC's `op_of_u.owed`. |
+| `u_bulk_donate`
`npc_bulk_donate` | Only valid after a `repeat_response`. The player or NPC transfers all instances of the item from the `repeat_response`. For `u_bulk_donate`, the player loses the items from their inventory and the NPC gains them; for `npc_bulk_donate`, the player gains the items from the NPC's inventory and the NPC loses them. |
+| `u_spend_ecash: amount` | Remove `amount` from your character's pre-cataclysm bank account. Negative values means your character gains e-cash. NPCs should **not** deal in e-cash, only personal debts and items (including faction currency). |
+| `add_debt: mod_list` | Increases the NPC's debt to the player by the values in the `mod_list`.
The following would increase the NPC's debt to the player by 1500x the NPC's altruism and 1000x the NPC's opinion of the player's value: `{ "effect": { "add_debt": [ [ "ALTRUISM", 3 ], [ "VALUE", 2 ], [ "TOTAL", 500 ] ] } }` |
+| `u_consume_item`, `npc_consume_item: item_string`, (_optional_ `count: count_num`) | You or the NPC will delete the item or `count_num` copies of the item from their inventory.
This effect will fail if the you or NPC does not have at least `count_num` copies of the item, so it should be checked with `u_has_items` or `npc_has_items`. |
+| `u_remove_item_with`, `npc_remove_item_with: item_string` | You or the NPC will delete any instances of item in inventory.
This is an unconditional remove and will not fail if you or the NPC does not have the item. |
+| `u_buy_monster: monster_type_string`, (_optional_ `cost: cost_num`, _optional_ `count: count_num`, _optional_ `name: name_string`, _optional_ `pacified: pacified_bool`) | The NPC will give your character `count_num` (default 1) instances of the monster as pets and will subtract `cost_num` from `op_of_u.owed` if specified. If the `op_o_u.owed` is less than `cost_num`, the trade window will open and the player will have to trade to make up the difference; the NPC will not give the player the item unless `cost_num` is satisfied.
If cost isn't present, the NPC gives your character the item at no charge.
If `name_string` is specified the monster(s) will have the specified name. If `pacified_bool` is set to true, the monster will have the pacified effect applied. |
#### Behavior / AI
-Effect | Description
----|---
-`assign_guard` | Makes the NPC into a guard. If allied and at a camp, they will be assigned to that camp.
-`stop_guard` | Releases the NPC from their guard duty (also see `assign_guard`). Friendly NPCs will return to following.
-`start_camp` | The NPC will start a faction camp with the player.
-`recover_camp` | Makes the NPC the overseer of an existing camp that doesn't have an overseer.
-`remove_overseer` | Makes the NPC stop being an overseer, abandoning the faction camp.
-`wake_up` | Wakes up sleeping, but not sedated, NPCs.
-`reveal_stats` | Reveals the NPC's stats, based on the player's skill at assessing them.
-`end_conversation` | Ends the conversation and makes the NPC ignore you from now on.
-`insult_combat` | Ends the conversation and makes the NPC hostile, adds a message that character starts a fight with the NPC.
-`hostile` | Makes the NPC hostile and ends the conversation.
-`flee` | Makes the NPC flee from your character.
-`follow` | Makes the NPC follow your character, joining the "Your Followers" faction.
-`leave` | Makes the NPC leave the "Your Followers" faction and stop following your character.
-`follow_only` | Makes the NPC follow your character without changing factions.
-`stop_following` | Makes the NPC stop following your character without changing factions.
-`npc_thankful` | Makes the NPC positively inclined toward your character.
-`drop_weapon` | Makes the NPC drop their weapon.
-`stranger_neutral` | Changes the NPC's attitude to neutral.
-`start_mugging` | The NPC will approach your character and steal from your character, attacking if your character resists.
-`lead_to_safety` | The NPC will gain the LEAD attitude and give your character the mission of reaching safety.
-`start_training` | The NPC will train your character in a skill or martial art.
-`companion_mission: role_string` | The NPC will offer you a list of missions for your allied NPCs, depending on the NPC's role.
-`basecamp_mission` | The NPC will offer you a list of missions for your allied NPCs, depending on the local basecamp.
-`bionic_install` | The NPC installs a bionic from your character's inventory onto your character, using very high skill, and charging you according to the operation's difficulty.
-`bionic_remove` | The NPC removes a bionic from your character, using very high skill, and charging you according to the operation's difficulty.
-`npc_class_change: class_string` | Change the NPC's faction to `class_string`.
-`npc_faction_change: faction_string` | Change the NPC's faction membership to `faction_string`.
-`u_faction_rep: rep_num` | Increases your reputation with the NPC's current faction, or decreases it if `rep_num` is negative.
-`toggle_npc_rule: rule_string` | Toggles the value of a boolean NPC follower AI rule such as `"use_silent"` or `"allow_bash"`
-`set_npc_rule: rule_string` | Sets the value of a boolean NPC follower AI rule such as `"use_silent"` or `"allow_bash"`
-`clear_npc_rule: rule_string` | Clears the value of a boolean NPC follower AI rule such as `"use_silent"` or `"allow_bash"`
-`set_npc_engagement_rule: rule_string` | Sets the NPC follower AI rule for engagement distance to the value of `rule_string`.
-`set_npc_aim_rule: rule_string` | Sets the NPC follower AI rule for aiming speed to the value of `rule_string`.
-`npc_die` | The NPC will die at the end of the conversation.
+| Effect | Description |
+| -------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `assign_guard` | Makes the NPC into a guard. If allied and at a camp, they will be assigned to that camp. |
+| `stop_guard` | Releases the NPC from their guard duty (also see `assign_guard`). Friendly NPCs will return to following. |
+| `start_camp` | The NPC will start a faction camp with the player. |
+| `recover_camp` | Makes the NPC the overseer of an existing camp that doesn't have an overseer. |
+| `remove_overseer` | Makes the NPC stop being an overseer, abandoning the faction camp. |
+| `wake_up` | Wakes up sleeping, but not sedated, NPCs. |
+| `reveal_stats` | Reveals the NPC's stats, based on the player's skill at assessing them. |
+| `end_conversation` | Ends the conversation and makes the NPC ignore you from now on. |
+| `insult_combat` | Ends the conversation and makes the NPC hostile, adds a message that character starts a fight with the NPC. |
+| `hostile` | Makes the NPC hostile and ends the conversation. |
+| `flee` | Makes the NPC flee from your character. |
+| `follow` | Makes the NPC follow your character, joining the "Your Followers" faction. |
+| `leave` | Makes the NPC leave the "Your Followers" faction and stop following your character. |
+| `follow_only` | Makes the NPC follow your character without changing factions. |
+| `stop_following` | Makes the NPC stop following your character without changing factions. |
+| `npc_thankful` | Makes the NPC positively inclined toward your character. |
+| `drop_weapon` | Makes the NPC drop their weapon. |
+| `stranger_neutral` | Changes the NPC's attitude to neutral. |
+| `start_mugging` | The NPC will approach your character and steal from your character, attacking if your character resists. |
+| `lead_to_safety` | The NPC will gain the LEAD attitude and give your character the mission of reaching safety. |
+| `start_training` | The NPC will train your character in a skill or martial art. |
+| `companion_mission: role_string` | The NPC will offer you a list of missions for your allied NPCs, depending on the NPC's role. |
+| `basecamp_mission` | The NPC will offer you a list of missions for your allied NPCs, depending on the local basecamp. |
+| `bionic_install` | The NPC installs a bionic from your character's inventory onto your character, using very high skill, and charging you according to the operation's difficulty. |
+| `bionic_remove` | The NPC removes a bionic from your character, using very high skill, and charging you according to the operation's difficulty. |
+| `npc_class_change: class_string` | Change the NPC's faction to `class_string`. |
+| `npc_faction_change: faction_string` | Change the NPC's faction membership to `faction_string`. |
+| `u_faction_rep: rep_num` | Increases your reputation with the NPC's current faction, or decreases it if `rep_num` is negative. |
+| `toggle_npc_rule: rule_string` | Toggles the value of a boolean NPC follower AI rule such as `"use_silent"` or `"allow_bash"` |
+| `set_npc_rule: rule_string` | Sets the value of a boolean NPC follower AI rule such as `"use_silent"` or `"allow_bash"` |
+| `clear_npc_rule: rule_string` | Clears the value of a boolean NPC follower AI rule such as `"use_silent"` or `"allow_bash"` |
+| `set_npc_engagement_rule: rule_string` | Sets the NPC follower AI rule for engagement distance to the value of `rule_string`. |
+| `set_npc_aim_rule: rule_string` | Sets the NPC follower AI rule for aiming speed to the value of `rule_string`. |
+| `npc_die` | The NPC will die at the end of the conversation. |
#### Map Updates
-`mapgen_update: mapgen_update_id_string`
`mapgen_update:` *list of `mapgen_update_id_string`s*, (optional `assign_mission_target` parameters) | With no other parameters, updates the overmap tile at the player's current location with the changes described in `mapgen_update_id` (or for each `mapgen_update_id` in the list). The `assign_mission_target` parameters can be used to change the location of the overmap tile that gets updated. See [the missions docs](MISSIONS_JSON.md) for `assign_mission_target` parameters and [the mapgen docs](MAPGEN.md) for `mapgen_update`.
+
+`mapgen_update: mapgen_update_id_string`
`mapgen_update:` _list of `mapgen_update_id_string`s_,
+(optional `assign_mission_target` parameters) | With no other parameters, updates the overmap tile
+at the player's current location with the changes described in `mapgen_update_id` (or for each
+`mapgen_update_id` in the list). The `assign_mission_target` parameters can be used to change the
+location of the overmap tile that gets updated. See [the missions docs](MISSIONS_JSON.md) for
+`assign_mission_target` parameters and [the mapgen docs](MAPGEN.md) for `mapgen_update`.
#### Deprecated
-Effect | Description
----|---
-`deny_follow`
`deny_lead`
`deny_train`
`deny_personal_info` | Sets the appropriate effect on the NPC for a few hours.
These are *deprecated* in favor of the more flexible `npc_add_effect` described above.
+| Effect | Description |
+| -------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `deny_follow`
`deny_lead`
`deny_train`
`deny_personal_info` | Sets the appropriate effect on the NPC for a few hours.
These are _deprecated_ in favor of the more flexible `npc_add_effect` described above. |
#### Sample effects
+
```json
{ "topic": "TALK_EVAC_GUARD3_HOSTILE", "effect": [ { "u_faction_rep": -15 }, { "npc_change_faction": "hells_raiders" } ] }
{ "text": "Let's trade then.", "effect": "start_trade", "topic": "TALK_EVAC_MERCHANT" },
@@ -512,122 +650,138 @@ Effect | Description
```
---
+
### opinion changes
+
As a special effect, an NPC's opinion of your character can change. Use the following:
#### opinion: { }
-trust, value, fear, and anger are optional keywords inside the opinion object. Each keyword must be followed by a numeric value. The NPC's opinion is modified by the value.
+
+trust, value, fear, and anger are optional keywords inside the opinion object. Each keyword must be
+followed by a numeric value. The NPC's opinion is modified by the value.
#### Sample opinions
+
```json
{ "effect": "follow", "opinion": { "trust": 1, "value": 1 }, "topic": "TALK_DONE" }
{ "topic": "TALK_DENY_FOLLOW", "effect": "deny_follow", "opinion": { "fear": -1, "value": -1, "anger": 1 } }
```
#### mission_opinion: { }
-trust, value, fear, and anger are optional keywords inside the `mission_opinion` object. Each keyword must be followed by a numeric value. The NPC's opinion is modified by the value.
+
+trust, value, fear, and anger are optional keywords inside the `mission_opinion` object. Each
+keyword must be followed by a numeric value. The NPC's opinion is modified by the value.
---
## Dialogue conditions
-Conditions can be a simple string with no other values, a key and an int, a key and a string, a key and an array, or a key and an object. Arrays and objects can nest with each other and can contain any other condition.
+
+Conditions can be a simple string with no other values, a key and an int, a key and a string, a key
+and an array, or a key and an object. Arrays and objects can nest with each other and can contain
+any other condition.
The following keys and simple strings are available:
#### Boolean logic
-Condition | Type | Description
---- | --- | ---
-`"and"` | array | `true` if every condition in the array is true. Can be used to create complex condition tests, like `"[INTELLIGENCE 10+][PERCEPTION 12+] Your jacket is torn. Did you leave that scrap of fabric behind?"`
-`"or"` | array | `true` if any condition in the array is true. Can be used to create complex condition tests, like `"[STRENGTH 9+] or [DEXTERITY 9+] I'm sure I can handle one zombie."`
-`"not"` | object | `true` if the condition in the object or string is false. Can be used to create complex conditions test by negating other conditions, for text such as
`"[INTELLIGENCE 7-] Hitting the reactor with a hammer should shut it off safely, right?"`
+| Condition | Type | Description |
+| --------- | ------ | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `"and"` | array | `true` if every condition in the array is true. Can be used to create complex condition tests, like `"[INTELLIGENCE 10+][PERCEPTION 12+] Your jacket is torn. Did you leave that scrap of fabric behind?"` |
+| `"or"` | array | `true` if any condition in the array is true. Can be used to create complex condition tests, like `"[STRENGTH 9+] or [DEXTERITY 9+] I'm sure I can handle one zombie."` |
+| `"not"` | object | `true` if the condition in the object or string is false. Can be used to create complex conditions test by negating other conditions, for text such as
`"[INTELLIGENCE 7-] Hitting the reactor with a hammer should shut it off safely, right?"` |
#### Player or NPC conditions
-These conditions can be tested for the player using the `"u_"` form, and for the NPC using the `"npc_"` form.
-
-Condition | Type | Description
---- | --- | ---
-`"u_male"`
`"npc_male"` | simple string | `true` if the player character or NPC is male.
-`"u_female"`
`"npc_female"` | simple string | `true` if the player character or NPC is female.
-`"u_at_om_location"`
`"npc_at_om_location"` | string | `true` if the player character or NPC is standing on an overmap tile with `u_at_om_location`'s id. The special string `"FACTION_CAMP_ANY"` changes it to return true if the player or NPC is standing on a faction camp overmap tile. The special string `"FACTION_CAMP_START"` changes it to return true if the overmap tile that the player or NPC is standing on can be turned into a faction camp overmap tile.
-`"u_has_trait"`
`"npc_has_trait"` | string | `true` if the player character or NPC has a specific trait. Simpler versions of `u_has_any_trait` and `npc_has_any_trait` that only checks for one trait.
-`"u_has_trait_flag"`
`"npc_has_trait_flag"` | string | `true` if the player character or NPC has any traits with the specific trait flag. More robust versions of `u_has_any_trait` and `npc_has_any_trait`. The special trait flag `"MUTATION_THRESHOLD"` checks to see if the player or NPC has crossed a mutation threshold.
-`"u_has_any_trait"`
`"npc_has_any_trait"` | array | `true` if the player character or NPC has any trait or mutation in the array. Used to check multiple specific traits.
-`"u_has_var"`, `"npc_has_var"` | string | `"type": type_str`, `"context": context_str`, and `"value": value_str` are required fields in the same dictionary as `"u_has_var"` or `"npc_has_var"`.
`true` is the player character or NPC has a variable set by `"u_add_var"` or `"npc_add_var"` with the string, `type_str`, `context_str`, and `value_str`.
-`"u_compare_var"`, `"npc_compare_var"` | dictionary | `"type": type_str`, `"context": context_str`, `"op": op_str`, `"value": value_num` are required fields, referencing a var as in `"u_add_var"` or `"npc_add_var"`.
`true` if the player character or NPC has a stored variable that is true for the provided operator `op_str` (one of `==`, `!=`, `<`, `>`, `<=`, `>=`) and value.
-`"u_has_strength"`
`"npc_has_strength"` | int | `true` if the player character's or NPC's strength is at least the value of `u_has_strength` or `npc_has_strength`.
-`"u_has_dexterity"`
`"npc_has_dexterity"` | int | `true` if the player character's or NPC's dexterity is at least the value of `u_has_dexterity` or `npc_has_dexterity`.
-`"u_has_intelligence"`
`"npc_has_intelligence"` | int | `true` if the player character's or NPC's intelligence is at least the value of `u_has_intelligence` or `npc_has_intelligence`.
-`"u_has_perception"`
`"npc_has_perception"` | int | `true` if the player character's or NPC's perception is at least the value of `u_has_perception` or `npc_has_perception`.
-`"u_has_item"`
`"npc_has_item"` | string | `true` if the player character or NPC has something with `u_has_item`'s or `npc_has_item`'s `item_id` in their inventory.
-`"u_has_items"`
`"npc_has_item"` | dictionary | `u_has_items` or `npc_has_items` must be a dictionary with an `item` string and a `count` int.
`true` if the player character or NPC has at least `count` charges or counts of `item` in their inventory.
-`"u_has_item_category"`
`"npc_has_item_category"` | string | `"count": item_count` is an optional field that must be in the same dictionary and defaults to 1 if not specified. `true` if the player or NPC has `item_count` items with the same category as `u_has_item_category` or `npc_has_item_category`.
-`"u_has_bionics"`
`"npc_has_bionics"` | string | `true` if the player or NPC has an installed bionic with an `bionic_id` matching `"u_has_bionics"` or `"npc_has_bionics"`. The special string "ANY" returns true if the player or NPC has any installed bionics.
-`"u_has_effect"`
`"npc_has_effect"` | string | `true` if the player character or NPC is under the effect with `u_has_effect` or `npc_has_effect`'s `effect_id`.
-`"u_can_stow_weapon"`
`"npc_can_stow_weapon"` | simple string | `true` if the player character or NPC is wielding a weapon and has enough space to put it away.
-`"u_has_weapon"`
`"npc_has_weapon"` | simple string | `true` if the player character or NPC is wielding a weapon.
-`"u_driving"`
`"npc_driving"` | simple string | `true` if the player character or NPC is operating a vehicle. Note NPCs cannot currently operate vehicles.
-`"u_has_skill"`
`"npc_has_skill"` | dictionary | `u_has_skill` or `npc_has_skill` must be a dictionary with a `skill` string and a `level` int.
`true` if the player character or NPC has at least the value of `level` in `skill`.
-`"u_know_recipe"` | string | `true` if the player character knows the recipe specified in `u_know_recipe`. It only counts as known if it is actually memorized--holding a book with the recipe in it will not count.
+
+These conditions can be tested for the player using the `"u_"` form, and for the NPC using the
+`"npc_"` form.
+
+| Condition | Type | Description |
+| ------------------------------------------------------ | ------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `"u_male"`
`"npc_male"` | simple string | `true` if the player character or NPC is male. |
+| `"u_female"`
`"npc_female"` | simple string | `true` if the player character or NPC is female. |
+| `"u_at_om_location"`
`"npc_at_om_location"` | string | `true` if the player character or NPC is standing on an overmap tile with `u_at_om_location`'s id. The special string `"FACTION_CAMP_ANY"` changes it to return true if the player or NPC is standing on a faction camp overmap tile. The special string `"FACTION_CAMP_START"` changes it to return true if the overmap tile that the player or NPC is standing on can be turned into a faction camp overmap tile. |
+| `"u_has_trait"`
`"npc_has_trait"` | string | `true` if the player character or NPC has a specific trait. Simpler versions of `u_has_any_trait` and `npc_has_any_trait` that only checks for one trait. |
+| `"u_has_trait_flag"`
`"npc_has_trait_flag"` | string | `true` if the player character or NPC has any traits with the specific trait flag. More robust versions of `u_has_any_trait` and `npc_has_any_trait`. The special trait flag `"MUTATION_THRESHOLD"` checks to see if the player or NPC has crossed a mutation threshold. |
+| `"u_has_any_trait"`
`"npc_has_any_trait"` | array | `true` if the player character or NPC has any trait or mutation in the array. Used to check multiple specific traits. |
+| `"u_has_var"`, `"npc_has_var"` | string | `"type": type_str`, `"context": context_str`, and `"value": value_str` are required fields in the same dictionary as `"u_has_var"` or `"npc_has_var"`.
`true` is the player character or NPC has a variable set by `"u_add_var"` or `"npc_add_var"` with the string, `type_str`, `context_str`, and `value_str`. |
+| `"u_compare_var"`, `"npc_compare_var"` | dictionary | `"type": type_str`, `"context": context_str`, `"op": op_str`, `"value": value_num` are required fields, referencing a var as in `"u_add_var"` or `"npc_add_var"`.
`true` if the player character or NPC has a stored variable that is true for the provided operator `op_str` (one of `==`, `!=`, `<`, `>`, `<=`, `>=`) and value. |
+| `"u_has_strength"`
`"npc_has_strength"` | int | `true` if the player character's or NPC's strength is at least the value of `u_has_strength` or `npc_has_strength`. |
+| `"u_has_dexterity"`
`"npc_has_dexterity"` | int | `true` if the player character's or NPC's dexterity is at least the value of `u_has_dexterity` or `npc_has_dexterity`. |
+| `"u_has_intelligence"`
`"npc_has_intelligence"` | int | `true` if the player character's or NPC's intelligence is at least the value of `u_has_intelligence` or `npc_has_intelligence`. |
+| `"u_has_perception"`
`"npc_has_perception"` | int | `true` if the player character's or NPC's perception is at least the value of `u_has_perception` or `npc_has_perception`. |
+| `"u_has_item"`
`"npc_has_item"` | string | `true` if the player character or NPC has something with `u_has_item`'s or `npc_has_item`'s `item_id` in their inventory. |
+| `"u_has_items"`
`"npc_has_item"` | dictionary | `u_has_items` or `npc_has_items` must be a dictionary with an `item` string and a `count` int.
`true` if the player character or NPC has at least `count` charges or counts of `item` in their inventory. |
+| `"u_has_item_category"`
`"npc_has_item_category"` | string | `"count": item_count` is an optional field that must be in the same dictionary and defaults to 1 if not specified. `true` if the player or NPC has `item_count` items with the same category as `u_has_item_category` or `npc_has_item_category`. |
+| `"u_has_bionics"`
`"npc_has_bionics"` | string | `true` if the player or NPC has an installed bionic with an `bionic_id` matching `"u_has_bionics"` or `"npc_has_bionics"`. The special string "ANY" returns true if the player or NPC has any installed bionics. |
+| `"u_has_effect"`
`"npc_has_effect"` | string | `true` if the player character or NPC is under the effect with `u_has_effect` or `npc_has_effect`'s `effect_id`. |
+| `"u_can_stow_weapon"`
`"npc_can_stow_weapon"` | simple string | `true` if the player character or NPC is wielding a weapon and has enough space to put it away. |
+| `"u_has_weapon"`
`"npc_has_weapon"` | simple string | `true` if the player character or NPC is wielding a weapon. |
+| `"u_driving"`
`"npc_driving"` | simple string | `true` if the player character or NPC is operating a vehicle. Note NPCs cannot currently operate vehicles. |
+| `"u_has_skill"`
`"npc_has_skill"` | dictionary | `u_has_skill` or `npc_has_skill` must be a dictionary with a `skill` string and a `level` int.
`true` if the player character or NPC has at least the value of `level` in `skill`. |
+| `"u_know_recipe"` | string | `true` if the player character knows the recipe specified in `u_know_recipe`. It only counts as known if it is actually memorized--holding a book with the recipe in it will not count. |
#### Player Only conditions
`"u_has_mission"` | string | `true` if the mission is assigned to the player character.
-`"u_has_ecash"` | int | `true` if the player character has at least `u_has_ecash` ecash available in his pre-cataclysm bank account. NPCs should **not** deal in e-cash, only personal debts and items (including faction currency).
-`"u_are_owed"` | int | `true` if the NPC's op_of_u.owed is at least `u_are_owed`. Can be used to check if the player can buy something from the NPC without needing to barter anything.
-`"u_has_camp"` | simple string | `true` is the player has one or more active base camps.
+`"u_has_ecash"` | int | `true` if the player character has at least `u_has_ecash` ecash available in
+his pre-cataclysm bank account. NPCs should **not** deal in e-cash, only personal debts and items
+(including faction currency). `"u_are_owed"` | int | `true` if the NPC's op_of_u.owed is at least
+`u_are_owed`. Can be used to check if the player can buy something from the NPC without needing to
+barter anything. `"u_has_camp"` | simple string | `true` is the player has one or more active base
+camps.
#### Player and NPC interaction conditions
-Condition | Type | Description
---- | --- | ---
-`"at_safe_space"` | simple string | `true` if the NPC's current overmap location passes the `is_safe()` test.
-`"has_assigned_mission"` | simple string | `true` if the player character has exactly one mission from the NPC. Can be used for texts like "About that job...".
-`"has_many_assigned_missions"` | simple string | `true` if the player character has several mission from the NPC (more than one). Can be used for texts like "About one of those jobs..." and to switch to the `"TALK_MISSION_LIST_ASSIGNED"` topic.
-`"has_no_available_mission"` | simple string | `true` if the NPC has no jobs available for the player character.
-`"has_available_mission"` | simple string | `true` if the NPC has one job available for the player character.
-`"has_many_available_missions"` | simple string | `true` if the NPC has several jobs available for the player character.
-`"mission_goal"` | string | `true` if the NPC's current mission has the same goal as `mission_goal`.
-`"mission_complete"` | simple string | `true` if the player has completed the NPC's current mission.
-`"mission_incomplete"` | simple string | `true` if the player hasn't completed the NPC's current mission.
-`"mission_has_generic_rewards"` | simple string | `true` if the NPC's current mission is flagged as having generic rewards.
-`"npc_service"` | simple string | `true` if the NPC does not have the `"currently_busy"` effect. Useful to check if the player character can hire an NPC to perform a task that would take time to complete. Functionally, this is identical to `not": { "npc_has_effect": "currently_busy" }`. Same as `npc_available`.
-`"npc_allies"` | int | `true` if the player character has at least `npc_allies` number of NPC allies.
-`"npc_following"` | simple string | `true` if the NPC is following the player character.
-`"is_by_radio"` | simple string | `true` if the player is talking to the NPC over a radio.
+| Condition | Type | Description |
+| ------------------------------- | ------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `"at_safe_space"` | simple string | `true` if the NPC's current overmap location passes the `is_safe()` test. |
+| `"has_assigned_mission"` | simple string | `true` if the player character has exactly one mission from the NPC. Can be used for texts like "About that job...". |
+| `"has_many_assigned_missions"` | simple string | `true` if the player character has several mission from the NPC (more than one). Can be used for texts like "About one of those jobs..." and to switch to the `"TALK_MISSION_LIST_ASSIGNED"` topic. |
+| `"has_no_available_mission"` | simple string | `true` if the NPC has no jobs available for the player character. |
+| `"has_available_mission"` | simple string | `true` if the NPC has one job available for the player character. |
+| `"has_many_available_missions"` | simple string | `true` if the NPC has several jobs available for the player character. |
+| `"mission_goal"` | string | `true` if the NPC's current mission has the same goal as `mission_goal`. |
+| `"mission_complete"` | simple string | `true` if the player has completed the NPC's current mission. |
+| `"mission_incomplete"` | simple string | `true` if the player hasn't completed the NPC's current mission. |
+| `"mission_has_generic_rewards"` | simple string | `true` if the NPC's current mission is flagged as having generic rewards. |
+| `"npc_service"` | simple string | `true` if the NPC does not have the `"currently_busy"` effect. Useful to check if the player character can hire an NPC to perform a task that would take time to complete. Functionally, this is identical to `not": { "npc_has_effect": "currently_busy" }`. Same as `npc_available`. |
+| `"npc_allies"` | int | `true` if the player character has at least `npc_allies` number of NPC allies. |
+| `"npc_following"` | simple string | `true` if the NPC is following the player character. |
+| `"is_by_radio"` | simple string | `true` if the player is talking to the NPC over a radio. |
#### NPC only conditions
-Condition | Type | Description
---- | --- | ---
-`"npc_available"` | simple string | `true` if the NPC does not have effect `"currently_busy"`.
-`"npc_following"` | simple string | `true` if the NPC is following the player character.
-`"npc_friend"` | simple string | `true` if the NPC is friendly to the player character.
-`"npc_hostile"` | simple string | `true` if the NPC is an enemy of the player character.
-`"npc_train_skills"` | simple string | `true` if the NPC has one or more skills with more levels than the player.
-`"npc_train_styles"` | simple string | `true` if the NPC knows one or more martial arts styles that the player does not know.
-`"npc_has_class"` | array | `true` if the NPC is a member of an NPC class.
-`"npc_role_nearby"` | string | `true` if there is an NPC with the same companion mission role as `npc_role_nearby` within 100 tiles.
-`"has_reason"` | simple string | `true` if a previous effect set a reason for why an effect could not be completed.
+| Condition | Type | Description |
+| -------------------- | ------------- | ----------------------------------------------------------------------------------------------------- |
+| `"npc_available"` | simple string | `true` if the NPC does not have effect `"currently_busy"`. |
+| `"npc_following"` | simple string | `true` if the NPC is following the player character. |
+| `"npc_friend"` | simple string | `true` if the NPC is friendly to the player character. |
+| `"npc_hostile"` | simple string | `true` if the NPC is an enemy of the player character. |
+| `"npc_train_skills"` | simple string | `true` if the NPC has one or more skills with more levels than the player. |
+| `"npc_train_styles"` | simple string | `true` if the NPC knows one or more martial arts styles that the player does not know. |
+| `"npc_has_class"` | array | `true` if the NPC is a member of an NPC class. |
+| `"npc_role_nearby"` | string | `true` if there is an NPC with the same companion mission role as `npc_role_nearby` within 100 tiles. |
+| `"has_reason"` | simple string | `true` if a previous effect set a reason for why an effect could not be completed. |
#### NPC Follower AI rules
-Condition | Type | Description
---- | --- | ---
-`"npc_aim_rule"` | string | `true` if the NPC follower AI rule for aiming matches the string.
-`"npc_engagement_rule"` | string | `true` if the NPC follower AI rule for engagement matches the string.
-`"npc_rule"` | string | `true` if the NPC follower AI rule for that matches string is set.
-#### Environment
+| Condition | Type | Description |
+| ----------------------- | ------ | --------------------------------------------------------------------- |
+| `"npc_aim_rule"` | string | `true` if the NPC follower AI rule for aiming matches the string. |
+| `"npc_engagement_rule"` | string | `true` if the NPC follower AI rule for engagement matches the string. |
+| `"npc_rule"` | string | `true` if the NPC follower AI rule for that matches string is set. |
-Condition | Type | Description
---- | --- | ---
-`"days_since_cataclysm"` | int | `true` if at least `days_since_cataclysm` days have passed since the Cataclysm.
-`"is_season"` | string | `true` if the current season matches `is_season`, which must be one of "`spring"`, `"summer"`, `"autumn"`, or `"winter"`.
-`"is_day"` | simple string | `true` if it is currently daytime.
-`"is_outside"` | simple string | `true` if the NPC is on a tile without a roof.
+#### Environment
+| Condition | Type | Description |
+| ------------------------ | ------------- | ------------------------------------------------------------------------------------------------------------------------- |
+| `"days_since_cataclysm"` | int | `true` if at least `days_since_cataclysm` days have passed since the Cataclysm. |
+| `"is_season"` | string | `true` if the current season matches `is_season`, which must be one of "`spring"`, `"summer"`, `"autumn"`, or `"winter"`. |
+| `"is_day"` | simple string | `true` if it is currently daytime. |
+| `"is_outside"` | simple string | `true` if the NPC is on a tile without a roof. |
#### Sample responses with conditions and effects
+
```json
{
"text": "Understood. I'll get those antibiotics.",
diff --git a/doc/OVERMAP.md b/doc/OVERMAP.md
index fb436d74f9ee..2d05939bb3db 100644
--- a/doc/OVERMAP.md
+++ b/doc/OVERMAP.md
@@ -34,16 +34,16 @@ player directly interacts with, providing the necessary context for local map ge
By example, the overmap tells the game:
-* where the cities are
-* where the roads are
-* where the buildings are
-* what types the buildings are
+- where the cities are
+- where the roads are
+- where the buildings are
+- what types the buildings are
but it does not tell the game:
-* what the actual road terrain looks like
-* what the actual building layout looks like
-* what items are in the building
+- what the actual road terrain looks like
+- what the actual building layout looks like
+- what items are in the building
It can sometimes be useful to think of the overmap as the outline for the game world which will then
be filled in as the player explores. The rest of this document is a discussion of how we can create
@@ -51,14 +51,14 @@ that outline.
## Terminology and Types
-First we need to briefly discuss some of the data types used by the
-overmap.
+First we need to briefly discuss some of the data types used by the overmap.
### overmap_terrain
-The fundamental unit of the overmap is the **overmap_terrain**, which defines the id, name, symbol, color
-(and more, but we'll get to that) used to represent a single location on the overmap. The overmap is 180
-overmap terrains wide, 180 overmap terrains tall, and 21 overmap terrains deep (these are the z-levels).
+The fundamental unit of the overmap is the **overmap_terrain**, which defines the id, name, symbol,
+color (and more, but we'll get to that) used to represent a single location on the overmap. The
+overmap is 180 overmap terrains wide, 180 overmap terrains tall, and 21 overmap terrains deep (these
+are the z-levels).
In this example snippet of an overmap, each character corresponds one entry in the overmap which
references a given overmap terrain:
@@ -76,11 +76,11 @@ So for example, the `F` is a forest which has a definition like this:
```json
{
- "type": "overmap_terrain",
- "id": "forest",
- "name": "forest",
- "sym": "F",
- "color": "green"
+ "type": "overmap_terrain",
+ "id": "forest",
+ "name": "forest",
+ "sym": "F",
+ "color": "green"
}
```
@@ -88,53 +88,55 @@ and the `^` is a house which has a definition like this:
```json
{
- "type": "overmap_terrain",
- "id": "house",
- "name": "house",
- "sym": "^",
- "color": "light_green"
+ "type": "overmap_terrain",
+ "id": "house",
+ "name": "house",
+ "sym": "^",
+ "color": "light_green"
}
```
-It's important to note that a single overmap terrain definition is referenced for every usage of that
-overmap terrain in the overmap--if we were to change the color of our forest from `green` to `red`, every
-forest in the overmap would be red.
+It's important to note that a single overmap terrain definition is referenced for every usage of
+that overmap terrain in the overmap--if we were to change the color of our forest from `green` to
+`red`, every forest in the overmap would be red.
### overmap_special / city_building
-The next important concept for the overmap is that of the **overmap_special** and
-**city_building**. These types, which are similar in structure, are the mechanism by which multiple
-overmap terrains can be collected together into one conceptual entity for placement on the overmap.
-If you wanted to have a sprawling mansion, a two story house, a large pond, or a secret lair placed
-under an unassuming bookstore, you would need to use an **overmap_special** or a **city_building**.
+The next important concept for the overmap is that of the **overmap_special** and **city_building**.
+These types, which are similar in structure, are the mechanism by which multiple overmap terrains
+can be collected together into one conceptual entity for placement on the overmap. If you wanted to
+have a sprawling mansion, a two story house, a large pond, or a secret lair placed under an
+unassuming bookstore, you would need to use an **overmap_special** or a **city_building**.
-These types are effectively a list of the overmap terrains that compose them, the placement of
-those overmap terrains relative to each other, and some data used to drive the placement of the
-overmap special / city building (e.g. how far from a city, should be it connected to a road, can it
-be placed in a forest/field/river, etc).
+These types are effectively a list of the overmap terrains that compose them, the placement of those
+overmap terrains relative to each other, and some data used to drive the placement of the overmap
+special / city building (e.g. how far from a city, should be it connected to a road, can it be
+placed in a forest/field/river, etc).
### overmap_connection
-Speaking of roads, the concept of linear features like roads, sewers, subways, railroads, and
-forest trails are managed through a combination of overmap terrain attributes and another type
-called an **overmap_connection**.
+Speaking of roads, the concept of linear features like roads, sewers, subways, railroads, and forest
+trails are managed through a combination of overmap terrain attributes and another type called an
+**overmap_connection**.
An overmap connection effectively defines the types of overmap terrain on which a given connection
can be made, the "cost" to make that connection, and the type of terrain to be placed when making
the connection. This, for example, is what allows us to say that:
-* a road may be placed on fields, forests, swamps and rivers
-* fields are preferred over forests, which are preferred over swamps, which are preferred over rivers
-* a road crossing a river will be a bridge
+- a road may be placed on fields, forests, swamps and rivers
+- fields are preferred over forests, which are preferred over swamps, which are preferred over
+ rivers
+- a road crossing a river will be a bridge
-The **overmap_connection** data will then be used to create a linear road feature connecting two points,
-applying the previously defined rules.
+The **overmap_connection** data will then be used to create a linear road feature connecting two
+points, applying the previously defined rules.
### overmap_location
-We've previously mentioned defining valid overmap terrain types for the placement of overmap specials,
-city buildings, and overmap connections, but one thing to clarify is these actually leverage another
-type called an **overmap_location** rather than referencing **overmap_terrain** values directly.
+We've previously mentioned defining valid overmap terrain types for the placement of overmap
+specials, city buildings, and overmap connections, but one thing to clarify is these actually
+leverage another type called an **overmap_location** rather than referencing **overmap_terrain**
+values directly.
Simply put, an **overmap_location** is just a named collection of **overmap_terrain** values.
@@ -179,69 +181,69 @@ update these definitions as follows:
### Rotation
-If an overmap terrain can be rotated (i.e. it does not have the `NO_ROTATE` flag), then when
-the game loads the definition from JSON, it will create the rotated definitions automatically,
-suffixing the `id` defined here with `_north`, `_east`, `_south` or `_west`. This will be
-particularly relevant if the overmap terrains are used in **overmap_special** or **city_building**
-definitions, because if those are allowed to rotate, it's desirable to specify a particular
-rotation for the referenced overmap terrains (e.g. the `_north` version for all).
+If an overmap terrain can be rotated (i.e. it does not have the `NO_ROTATE` flag), then when the
+game loads the definition from JSON, it will create the rotated definitions automatically, suffixing
+the `id` defined here with `_north`, `_east`, `_south` or `_west`. This will be particularly
+relevant if the overmap terrains are used in **overmap_special** or **city_building** definitions,
+because if those are allowed to rotate, it's desirable to specify a particular rotation for the
+referenced overmap terrains (e.g. the `_north` version for all).
### Fields
-| Identifier | Description |
-| ----------------- | ------------------------------------------------------------------------------------------------ |
-| `type` | Must be "overmap_terrain". |
-| `id` | Unique id. |
-| `name` | Name for the location shown in game. |
-| `sym` | Symbol used when drawing the location, like `"F"` (or you may use an ASCII value like `70`). |
-| `color` | Color to draw the symbol in. See [COLOR.md](COLOR.md). |
-| `looks_like` | Id of another overmap terrain to be used for the graphical tile, if this doesn't have one. |
-| `connect_group` | Specify that this overmap terrain might be graphically connected to its neighbours, should a tileset wish to. It will connect to any other `overmap_terrain` with the same `connect_group`. |
-| `see_cost` | Affects player vision on overmap. Higher values obstruct vision more. |
-| `travel_cost` | Affects pathfinding cost. Higher values are harder to travel through (reference: Forest = 10 ) |
-| `extras` | Reference to a named `map_extras` in region_settings, defines which map extras can be applied. |
-| `mondensity` | Summed with values for adjacent overmap terrains to influence density of monsters spawned here. |
-| `spawns` | Spawns added once at mapgen. Monster group, % chance, population range (min/max). |
-| `flags` | See `Overmap terrains` in [JSON_FLAGS.md](JSON_FLAGS.md). |
-| `mapgen` | Specify a C++ mapgen function. Don't do this--use JSON. |
-| `mapgen_straight` | Specify a C++ mapgen function for a LINEAR feature variation. |
-| `mapgen_curved` | Specify a C++ mapgen function for a LINEAR feature variation. |
-| `mapgen_end` | Specify a C++ mapgen function for a LINEAR feature variation. |
-| `mapgen_tee` | Specify a C++ mapgen function for a LINEAR feature variation. |
-| `mapgen_four_way` | Specify a C++ mapgen function for a LINEAR feature variation. |
+| Identifier | Description |
+| ----------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `type` | Must be "overmap_terrain". |
+| `id` | Unique id. |
+| `name` | Name for the location shown in game. |
+| `sym` | Symbol used when drawing the location, like `"F"` (or you may use an ASCII value like `70`). |
+| `color` | Color to draw the symbol in. See [COLOR.md](COLOR.md). |
+| `looks_like` | Id of another overmap terrain to be used for the graphical tile, if this doesn't have one. |
+| `connect_group` | Specify that this overmap terrain might be graphically connected to its neighbours, should a tileset wish to. It will connect to any other `overmap_terrain` with the same `connect_group`. |
+| `see_cost` | Affects player vision on overmap. Higher values obstruct vision more. |
+| `travel_cost` | Affects pathfinding cost. Higher values are harder to travel through (reference: Forest = 10 ) |
+| `extras` | Reference to a named `map_extras` in region_settings, defines which map extras can be applied. |
+| `mondensity` | Summed with values for adjacent overmap terrains to influence density of monsters spawned here. |
+| `spawns` | Spawns added once at mapgen. Monster group, % chance, population range (min/max). |
+| `flags` | See `Overmap terrains` in [JSON_FLAGS.md](JSON_FLAGS.md). |
+| `mapgen` | Specify a C++ mapgen function. Don't do this--use JSON. |
+| `mapgen_straight` | Specify a C++ mapgen function for a LINEAR feature variation. |
+| `mapgen_curved` | Specify a C++ mapgen function for a LINEAR feature variation. |
+| `mapgen_end` | Specify a C++ mapgen function for a LINEAR feature variation. |
+| `mapgen_tee` | Specify a C++ mapgen function for a LINEAR feature variation. |
+| `mapgen_four_way` | Specify a C++ mapgen function for a LINEAR feature variation. |
### Example
-A real `overmap_terrain` wouldn't have all these defined at the same time, but in the interest of
-an exhaustive example...
+A real `overmap_terrain` wouldn't have all these defined at the same time, but in the interest of an
+exhaustive example...
```json
{
- "type": "overmap_terrain",
- "id": "field",
- "name": "field",
- "sym": ".",
- "color": "brown",
- "looks_like": "forest",
- "see_cost": 2,
- "extras": "field",
- "mondensity": 2,
- "spawns": { "group": "GROUP_FOREST", "population": [ 0, 1 ], "chance": 13 },
- "flags": [ "NO_ROTATE" ],
- "mapgen": [ { "method": "builtin", "name": "bridge" } ],
- "mapgen_straight": [ { "method": "builtin", "name": "road_straight" } ],
- "mapgen_curved": [ { "method": "builtin", "name": "road_curved" } ],
- "mapgen_end": [ { "method": "builtin", "name": "road_end" } ],
- "mapgen_tee": [ { "method": "builtin", "name": "road_tee" } ],
- "mapgen_four_way": [ { "method": "builtin", "name": "road_four_way" } ]
+ "type": "overmap_terrain",
+ "id": "field",
+ "name": "field",
+ "sym": ".",
+ "color": "brown",
+ "looks_like": "forest",
+ "see_cost": 2,
+ "extras": "field",
+ "mondensity": 2,
+ "spawns": { "group": "GROUP_FOREST", "population": [0, 1], "chance": 13 },
+ "flags": ["NO_ROTATE"],
+ "mapgen": [{ "method": "builtin", "name": "bridge" }],
+ "mapgen_straight": [{ "method": "builtin", "name": "road_straight" }],
+ "mapgen_curved": [{ "method": "builtin", "name": "road_curved" }],
+ "mapgen_end": [{ "method": "builtin", "name": "road_end" }],
+ "mapgen_tee": [{ "method": "builtin", "name": "road_tee" }],
+ "mapgen_four_way": [{ "method": "builtin", "name": "road_four_way" }]
}
```
## Overmap Special
An overmap special is an entity that is placed in the overmap after the city generation process has
-completed--they are the "non-city" counterpart to the **city_building** type. They commonly are
-made up of multiple overmap terrains (though not always), may have overmap connections (e.g. roads,
+completed--they are the "non-city" counterpart to the **city_building** type. They commonly are made
+up of multiple overmap terrains (though not always), may have overmap connections (e.g. roads,
sewers, subways), and have JSON-defined rules guiding their placement.
### Mandatory Overmap Specials
@@ -260,9 +262,9 @@ minimum should be 0.
In general, it is desirable to define your overmap special as allowing rotation--this will provide
variety in the game world and allow for more possible valid locations when attempting to place the
overmap special. A consequence of the relationship between rotating an overmap special and rotating
-the underlying overmap terrains that make up the special is that the overmap special should reference
-a specific rotated version of the associated overmap terrain--generally, this is the `_north` rotation
-as it corresponds to the way in which the JSON for mapgen is defined.
+the underlying overmap terrains that make up the special is that the overmap special should
+reference a specific rotated version of the associated overmap terrain--generally, this is the
+`_north` rotation as it corresponds to the way in which the JSON for mapgen is defined.
### Locations
@@ -276,7 +278,7 @@ level value and then only specify it for individual entries that differ.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| --------------- | ----------------------------------------------------------------------------------------------------- |
| `type` | Must be "overmap_special". |
| `id` | Unique id. |
@@ -297,25 +299,25 @@ level value and then only specify it for individual entries that differ.
"type": "overmap_special",
"id": "campground",
"overmaps": [
- { "point": [ 0, 0, 0 ], "overmap": "campground_1a_north", "locations": [ "forest_edge" ] },
- { "point": [ 1, 0, 0 ], "overmap": "campground_1b_north" },
- { "point": [ 0, 1, 0 ], "overmap": "campground_2a_north" },
- { "point": [ 1, 1, 0 ], "overmap": "campground_2b_north" }
+ { "point": [0, 0, 0], "overmap": "campground_1a_north", "locations": ["forest_edge"] },
+ { "point": [1, 0, 0], "overmap": "campground_1b_north" },
+ { "point": [0, 1, 0], "overmap": "campground_2a_north" },
+ { "point": [1, 1, 0], "overmap": "campground_2b_north" }
],
- "connections": [ { "point": [ 1, -1, 0 ], "connection": "local_road", "from": [ 1, 0, 0 ] } ],
- "locations": [ "forest" ],
- "city_distance": [ 10, -1 ],
- "city_sizes": [ 3, 12 ],
- "occurrences": [ 0, 5 ],
- "flags": [ "CLASSIC" ],
- "rotate" : true
+ "connections": [{ "point": [1, -1, 0], "connection": "local_road", "from": [1, 0, 0] }],
+ "locations": ["forest"],
+ "city_distance": [10, -1],
+ "city_sizes": [3, 12],
+ "occurrences": [0, 5],
+ "flags": ["CLASSIC"],
+ "rotate": true
}
]
```
### Overmaps
-| Identifier | Description |
+| Identifier | Description |
| ----------- | -------------------------------------------------------------------------- |
| `point` | `[ x, y, z]` of the overmap terrain within the special. |
| `overmap` | Id of the `overmap_terrain` to place at the location. |
@@ -323,7 +325,7 @@ level value and then only specify it for individual entries that differ.
### Connections
-| Identifier | Description |
+| Identifier | Description |
| ------------ | -------------------------------------------------------------------------------------------------- |
| `point` | `[ x, y, z]` of the connection end point. Cannot overlap an overmap terrain entry for the special. |
| `connection` | Id of the `overmap_connection` to build. |
@@ -332,19 +334,19 @@ level value and then only specify it for individual entries that differ.
## City Building
A city building is an entity that is placed in the overmap during the city generation process--they
-are the "city" counterpart to the **overmap_special** type. The definition for a city building is a subset
-of that for an overmap special, and consequently will not be repeated in detail here.
+are the "city" counterpart to the **overmap_special** type. The definition for a city building is a
+subset of that for an overmap special, and consequently will not be repeated in detail here.
### Mandatory Overmap Specials / Region Settings
-City buildings are not subject to the same quantity limitations as overmap specials, and in fact
-the occurrences attribute does not apply at all. Instead, the placement of city buildings is driven
-by the frequency assigned to the city building within the `region_settings`. Consult
+City buildings are not subject to the same quantity limitations as overmap specials, and in fact the
+occurrences attribute does not apply at all. Instead, the placement of city buildings is driven by
+the frequency assigned to the city building within the `region_settings`. Consult
[REGION_SETTINGS.md](REGION_SETTINGS.md) for more details.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ----------- | -------------------------------------------------------------------------------- |
| `type` | Must be "city_building". |
| `id` | Unique id. |
@@ -360,20 +362,20 @@ by the frequency assigned to the city building within the `region_settings`. Con
{
"type": "city_building",
"id": "zoo",
- "locations": [ "land" ],
+ "locations": ["land"],
"overmaps": [
- { "point": [ 0, 0, 0 ], "overmap": "zoo_0_0_north" },
- { "point": [ 1, 0, 0 ], "overmap": "zoo_1_0_north" },
- { "point": [ 2, 0, 0 ], "overmap": "zoo_2_0_north" },
- { "point": [ 0, 1, 0 ], "overmap": "zoo_0_1_north" },
- { "point": [ 1, 1, 0 ], "overmap": "zoo_1_1_north" },
- { "point": [ 2, 1, 0 ], "overmap": "zoo_2_1_north" },
- { "point": [ 0, 2, 0 ], "overmap": "zoo_0_2_north" },
- { "point": [ 1, 2, 0 ], "overmap": "zoo_1_2_north" },
- { "point": [ 2, 2, 0 ], "overmap": "zoo_2_2_north" }
+ { "point": [0, 0, 0], "overmap": "zoo_0_0_north" },
+ { "point": [1, 0, 0], "overmap": "zoo_1_0_north" },
+ { "point": [2, 0, 0], "overmap": "zoo_2_0_north" },
+ { "point": [0, 1, 0], "overmap": "zoo_0_1_north" },
+ { "point": [1, 1, 0], "overmap": "zoo_1_1_north" },
+ { "point": [2, 1, 0], "overmap": "zoo_2_1_north" },
+ { "point": [0, 2, 0], "overmap": "zoo_0_2_north" },
+ { "point": [1, 2, 0], "overmap": "zoo_1_2_north" },
+ { "point": [2, 2, 0], "overmap": "zoo_2_2_north" }
],
- "flags": [ "CLASSIC" ],
- "rotate" : true
+ "flags": ["CLASSIC"],
+ "rotate": true
}
]
```
@@ -382,14 +384,13 @@ by the frequency assigned to the city building within the `region_settings`. Con
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ----------------- | ----------------------------------------------------------------------------------------------- |
| `type` | Must be "overmap_connection". |
| `id` | Unique id. |
| `default_terrain` | Default `overmap_terrain` to use for undirected connections and existance checks. |
| `subtypes` | List of entries used to determine valid locations, terrain cost, and resulting overmap terrain. |
-
### Example
```json
@@ -398,25 +399,27 @@ by the frequency assigned to the city building within the `region_settings`. Con
"type": "overmap_connection",
"id": "local_road",
"subtypes": [
- { "terrain": "road", "locations": [ "field", "road" ] },
- { "terrain": "road", "locations": [ "forest_without_trail" ], "basic_cost": 20 },
- { "terrain": "road", "locations": [ "forest_trail" ], "basic_cost": 25 },
- { "terrain": "road", "locations": [ "swamp" ], "basic_cost": 40 },
- { "terrain": "road_nesw_manhole", "locations": [ ] },
- { "terrain": "bridge", "locations": [ "water" ], "basic_cost": 120 }
+ { "terrain": "road", "locations": ["field", "road"] },
+ { "terrain": "road", "locations": ["forest_without_trail"], "basic_cost": 20 },
+ { "terrain": "road", "locations": ["forest_trail"], "basic_cost": 25 },
+ { "terrain": "road", "locations": ["swamp"], "basic_cost": 40 },
+ { "terrain": "road_nesw_manhole", "locations": [] },
+ { "terrain": "bridge", "locations": ["water"], "basic_cost": 120 }
]
},
{
"type": "overmap_connection",
"id": "subway_tunnel",
- "subtypes": [ { "terrain": "subway", "locations": [ "subterranean_subway" ], "flags": [ "ORTHOGONAL" ] } ]
+ "subtypes": [
+ { "terrain": "subway", "locations": ["subterranean_subway"], "flags": ["ORTHOGONAL"] }
+ ]
}
]
```
### Subtypes
-| Identifier | Description |
+| Identifier | Description |
| ------------ | ---------------------------------------------------------------------------------------------------------- |
| `terrain` | `overmap_terrain` to be placed when the placement location matches `locations`. |
| `locations` | List of `overmap_location` that this subtype applies to. Can be empty; signifies `terrain` is valid as is. |
@@ -427,13 +430,12 @@ by the frequency assigned to the city building within the `region_settings`. Con
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ---------- | ----------------------------------------------------------------------- |
| `type` | Must be "overmap_location". |
| `id` | Unique id. |
| `terrains` | List of `overmap_terrain` that can be considered part of this location. |
-
### Example
```json
@@ -441,8 +443,7 @@ by the frequency assigned to the city building within the `region_settings`. Con
{
"type": "overmap_location",
"id": "wilderness",
- "terrains": [ "forest", "forest_thick", "field", "forest_trail" ]
+ "terrains": ["forest", "forest_thick", "field", "forest_trail"]
}
]
-
```
diff --git a/doc/PLAYER_ACTIVITY.md b/doc/PLAYER_ACTIVITY.md
index a67f993a4be7..831fc79698ce 100644
--- a/doc/PLAYER_ACTIVITY.md
+++ b/doc/PLAYER_ACTIVITY.md
@@ -1,71 +1,63 @@
# Activities
-Activities are long term actions, that can be interrupted and (optionally)
-continued. This allows the avatar or an npc to react to events (like
-monsters appearing, getting hurt) while doing something that takes more
-than just one turn.
+Activities are long term actions, that can be interrupted and (optionally) continued. This allows
+the avatar or an npc to react to events (like monsters appearing, getting hurt) while doing
+something that takes more than just one turn.
## Adding new activities
-1. `player_activities.json` Define properties that apply to all instances
-of the activity in question.
+1. `player_activities.json` Define properties that apply to all instances of the activity in
+ question.
-2. `activity_actor.h` Create a new subclass of `activity_actor`
-(e.g. move_items_activity_actor) to store state and handle turns of the
-new activity.
+2. `activity_actor.h` Create a new subclass of `activity_actor` (e.g. move_items_activity_actor) to
+ store state and handle turns of the new activity.
-3. `activity_actor.cpp` Define the `start`, `do_turn`, and `finish`
-functions needed for the new actor as well as the required serialization
-functions. Don't forget to add the deserialization function of your new
-activity actor to the `deserialize_functions` map towards the bottom of
-`activity_actor.cpp`. Define `canceled` function if activity modifies
-some complex state that should be restored upon cancellation / interruption.
+3. `activity_actor.cpp` Define the `start`, `do_turn`, and `finish` functions needed for the new
+ actor as well as the required serialization functions. Don't forget to add the deserialization
+ function of your new activity actor to the `deserialize_functions` map towards the bottom of
+ `activity_actor.cpp`. Define `canceled` function if activity modifies some complex state that
+ should be restored upon cancellation / interruption.
-4. If this activity is resumable, `override`
-`activity_actor::can_resume_with_internal`
+4. If this activity is resumable, `override` `activity_actor::can_resume_with_internal`
-5. Construct your activity actor and then pass it to the constructor for
-`player_activity`. The newly constructed activity can then be assigned
-to the character and started using `Character::assign_activity`.
+5. Construct your activity actor and then pass it to the constructor for `player_activity`. The
+ newly constructed activity can then be assigned to the character and started using
+ `Character::assign_activity`.
## JSON Properties
-* verb: A descriptive term to describe the activity to be used in the
-query to stop the activity, and strings that describe it, for example:
-`"verb": "mining"` or
-`"verb": { "ctxt": "instrument", "str": "playing" }`.
+- verb: A descriptive term to describe the activity to be used in the query to stop the activity,
+ and strings that describe it, for example: `"verb": "mining"` or
+ `"verb": { "ctxt": "instrument", "str": "playing" }`.
-* suspendable (true): If true, the activity can be continued without
-starting from scratch again. This is only possible if `can_resume_with()`
-returns true.
+- suspendable (true): If true, the activity can be continued without starting from scratch again.
+ This is only possible if `can_resume_with()` returns true.
-* rooted (false): If true, then during the activity, recoil is reduced,
-and plant mutants sink their roots into the ground. Should be true if the
-activity lasts longer than a few minutes, and can always be accomplished
-without moving your feet.
+- rooted (false): If true, then during the activity, recoil is reduced, and plant mutants sink their
+ roots into the ground. Should be true if the activity lasts longer than a few minutes, and can
+ always be accomplished without moving your feet.
-* based_on: Can be 'time', 'speed', or 'neither'.
+- based_on: Can be 'time', 'speed', or 'neither'.
- * time: The amount that `player_activity::moves_left` is
- decremented by is independent from the character's speed.
+ - time: The amount that `player_activity::moves_left` is decremented by is independent from the
+ character's speed.
- * speed: `player_activity::moves_left` may be decremented faster
- or slower, depending on the character's speed.
+ - speed: `player_activity::moves_left` may be decremented faster or slower, depending on the
+ character's speed.
- * neither: `moves_left` will not be decremented. Thus you must
- define a do_turn function; otherwise the activity will never end!
+ - neither: `moves_left` will not be decremented. Thus you must define a do_turn function;
+ otherwise the activity will never end!
-* no_resume (false): Rather than resuming, you must always restart the
-activity from scratch.
+- no_resume (false): Rather than resuming, you must always restart the activity from scratch.
-* multi_activity(false): This activity will repeat until it cannot do
-any more work, used for NPC and avatar zone activities.
+- multi_activity(false): This activity will repeat until it cannot do any more work, used for NPC
+ and avatar zone activities.
-* refuel_fires( false ): If true, the character will automatically refuel
-fires during the long activity.
+- refuel_fires( false ): If true, the character will automatically refuel fires during the long
+ activity.
-* auto_needs( false ) : If true, the character will automatically eat and
-drink from specific auto_consume zones during long activities.
+- auto_needs( false ) : If true, the character will automatically eat and drink from specific
+ auto_consume zones during long activities.
## Termination
@@ -73,71 +65,61 @@ There are several ways an activity can be ended:
1. Call `player_activity::set_to_null()`
- This can be called if it finished early or something went wrong,
- such as corrupt data, disappearing items, etc. The activity will
- not be put into the backlog.
+ This can be called if it finished early or something went wrong, such as corrupt data,
+ disappearing items, etc. The activity will not be put into the backlog.
2. `moves_left` <= 0
- Once this condition is true, the finish function, if there is one,
- is called. The finish function should call `set_to_null()`. If
- there isn't a finish function, `set_to_null()` will be called
- for you (from activity_actor::do_turn).
+ Once this condition is true, the finish function, if there is one, is called. The finish function
+ should call `set_to_null()`. If there isn't a finish function, `set_to_null()` will be called for
+ you (from activity_actor::do_turn).
3. `Character::cancel_activity`
- Canceling an activity prevents the `activity_actor::finish`
- function from running, and the activity does therefore not yield a
- result. Instead, `activity_actor::canceled` is called. If activity is
- suspendable, a copy of it is written to `Character::backlog`.
+ Canceling an activity prevents the `activity_actor::finish` function from running, and the
+ activity does therefore not yield a result. Instead, `activity_actor::canceled` is called. If
+ activity is suspendable, a copy of it is written to `Character::backlog`.
## Notes
-While the character performs the activity,
-`activity_actor::do_turn` is called on each turn. Depending on the
-JSON properties, this will do some stuff. It will also call the do_turn
-function, and if `moves_left` is non-positive, the finish function.
+While the character performs the activity, `activity_actor::do_turn` is called on each turn.
+Depending on the JSON properties, this will do some stuff. It will also call the do_turn function,
+and if `moves_left` is non-positive, the finish function.
-Some activities (like playing music on a mp3 player) don't have an end
-result but instead do something each turn (playing music on the mp3
-player decreases its batteries and gives a morale bonus).
+Some activities (like playing music on a mp3 player) don't have an end result but instead do
+something each turn (playing music on the mp3 player decreases its batteries and gives a morale
+bonus).
-If the activity needs any information during its execution or when
-it's finished (like *where* to do something, *which* item to use to do
-something, ...), simply add data members to your activity actor as well
-as serialization functions for the data so that the activity can persist
-across save/load cycles.
+If the activity needs any information during its execution or when it's finished (like _where_ to do
+something, _which_ item to use to do something, ...), simply add data members to your activity actor
+as well as serialization functions for the data so that the activity can persist across save/load
+cycles.
-Be careful when storing coordinates as the activity may be carried out
-by NPCS. If its, the coordinates must be absolute not local as local
-coordinates are based on the avatars position.
+Be careful when storing coordinates as the activity may be carried out by NPCS. If its, the
+coordinates must be absolute not local as local coordinates are based on the avatars position.
### `activity_actor::start`
-This function is called exactly once when the activity
-is assigned to a character. It is useful for setting
-`player_activity::moves_left`/`player_activity::moves_total` in the case
-of an activity based on time or speed.
+This function is called exactly once when the activity is assigned to a character. It is useful for
+setting `player_activity::moves_left`/`player_activity::moves_total` in the case of an activity
+based on time or speed.
### `activity_actor::do_turn`
-To prevent an infinite loop, ensure that one of the following is
-satisfied:
+To prevent an infinite loop, ensure that one of the following is satisfied:
- The `based_on` JSON property is `speed` or `time`
- The `player_activity::moves_left` is decreased in `do_turn`
-- The the activity is stopped in `do_turn` (see 'Termination' above)
+- The the activity is stopped in `do_turn` (see 'Termination' above)
-For example, `move_items_activity_actor::do_turn` will either move as
-many items as possible given the character's current moves or, if there
-are no target items remaining, terminate the activity.
+For example, `move_items_activity_actor::do_turn` will either move as many items as possible given
+the character's current moves or, if there are no target items remaining, terminate the activity.
### `activity_actor::finish`
-This function is called when the activity has been completed
-(`moves_left` <= 0). It must call `player_activity::set_to_null()` or
-assign a new activity. One should not call `Character::cancel_activity`
-for the finished activity, as this could make a copy of the activity in
+This function is called when the activity has been completed (`moves_left` <= 0). It must call
+`player_activity::set_to_null()` or assign a new activity. One should not call
+`Character::cancel_activity` for the finished activity, as this could make a copy of the activity in
`Character::backlog`, which allows resuming an already finished activity.
diff --git a/doc/POINTS_COORDINATES.md b/doc/POINTS_COORDINATES.md
index 35ca4d9bf2d9..eb3fa00444a0 100644
--- a/doc/POINTS_COORDINATES.md
+++ b/doc/POINTS_COORDINATES.md
@@ -3,127 +3,113 @@
## Axes
The game is three-dimensional, with the axes oriented as follows:
-* The **x-axis** goes from left to right across the display (in non-isometric
- views).
-* The **y-axis** goes from top to bottom of the display.
-* The **z-axis** is vertical, with negative z pointing underground and positive
- z pointing to the sky.
+
+- The **x-axis** goes from left to right across the display (in non-isometric views).
+- The **y-axis** goes from top to bottom of the display.
+- The **z-axis** is vertical, with negative z pointing underground and positive z pointing to the
+ sky.
## Coordinate systems
-CDDA uses a variety of coordinate systems for different purposes. These differ
-by scale and origin.
+CDDA uses a variety of coordinate systems for different purposes. These differ by scale and origin.
-The most precise coordinates are **map square** (ms) coordinates. These refer to
-the tiles you see normally when playing the game.
+The most precise coordinates are **map square** (ms) coordinates. These refer to the tiles you see
+normally when playing the game.
Two origins for map square coordinates are common:
-* **Absolute** coordinates, sometimes called global, which are a global system
- for the whole game, relative to a fixed origin.
-* **Local** coordinates, which are relative to the corner of the current "reality
- bubble", or `map` roughly centered on the avatar. In local map square
- coordinates, `x` and `y` values will both fall in the range [0,`MAPSIZE_X`).
-The next scale is **submap** (sm) coordinates. One submap is 12x12
-(`SEEX`x`SEEY`) map squares. Submaps are the scale at which chunks of the map
-are loaded or saved as they enter or leave the reality bubble.
+- **Absolute** coordinates, sometimes called global, which are a global system for the whole game,
+ relative to a fixed origin.
+- **Local** coordinates, which are relative to the corner of the current "reality bubble", or `map`
+ roughly centered on the avatar. In local map square coordinates, `x` and `y` values will both fall
+ in the range [0,`MAPSIZE_X`).
+
+The next scale is **submap** (sm) coordinates. One submap is 12x12 (`SEEX`x`SEEY`) map squares.
+Submaps are the scale at which chunks of the map are loaded or saved as they enter or leave the
+reality bubble.
-Next comes **overmap terrain** (omt) coordinates. One overmap terrain is 2x2
-submaps. Overmap terrains correspond to a single tile on the map view in-game,
-and are the scale of chunk of mapgen.
+Next comes **overmap terrain** (omt) coordinates. One overmap terrain is 2x2 submaps. Overmap
+terrains correspond to a single tile on the map view in-game, and are the scale of chunk of mapgen.
-Largest are **overmap** (om) coordinates. One overmap is 180x180
-(`OMAPX`x`OMAPY`) overmap terrains. Large-scale mapgen (e.g. city layout)
-happens one overmap at a time.
+Largest are **overmap** (om) coordinates. One overmap is 180x180 (`OMAPX`x`OMAPY`) overmap terrains.
+Large-scale mapgen (e.g. city layout) happens one overmap at a time.
-Lastly, these is a system called **segment** (seg) coordinates. These are only
-used in saving/loading submaps and you are unlikely to encounter them.
+Lastly, these is a system called **segment** (seg) coordinates. These are only used in
+saving/loading submaps and you are unlikely to encounter them.
-As well as absolute and local coordinates, sometimes we need to use coordinates
-relative so some larger scale. For example, when performing mapgen for a
-single overmap, we want to work with coordinates within that overmap. This
-will be an overmap terrain-scale point relative to the corner of its containing
-overmap, and so typically take `x` and `y` values in the range [0,180).
+As well as absolute and local coordinates, sometimes we need to use coordinates relative so some
+larger scale. For example, when performing mapgen for a single overmap, we want to work with
+coordinates within that overmap. This will be an overmap terrain-scale point relative to the corner
+of its containing overmap, and so typically take `x` and `y` values in the range [0,180).
## Vertical coordinates
-Although `x` and `y` coordinates work at all these various scales, `z`
-coordinates are consistent across all contexts. They lie in the range
-[-`OVERMAP_DEPTH`,`OVERMAP_HEIGHT`].
+Although `x` and `y` coordinates work at all these various scales, `z` coordinates are consistent
+across all contexts. They lie in the range [-`OVERMAP_DEPTH`,`OVERMAP_HEIGHT`].
## Vehicle coordinates
-Each vehicle has its own origin point, which will be at a particular part of
-the vehicle (e.g. it might be at the driver's seat). The origin can move if
-the vehicle is damaged and all the vehicle parts at that location are
-destroyed.
+Each vehicle has its own origin point, which will be at a particular part of the vehicle (e.g. it
+might be at the driver's seat). The origin can move if the vehicle is damaged and all the vehicle
+parts at that location are destroyed.
Vehicles use two systems of coordinates relative to their origin:
-* **mount** coordinates provide a location for vehicle parts that does not
- change as the vehicle moves. It is the map square of that part, relative to
- the vehicle origin, when the vehicle is facing due east.
+- **mount** coordinates provide a location for vehicle parts that does not change as the vehicle
+ moves. It is the map square of that part, relative to the vehicle origin, when the vehicle is
+ facing due east.
-* **map square** is the map square, relative to the origin, but accounting for
- the vehicle's current facing.
+- **map square** is the map square, relative to the origin, but accounting for the vehicle's current
+ facing.
-Vehicle facing is implemented via a combination of rotations (by quarter turns)
-and shearing to interpolate between quarter turns. The logic to convert
-between vehicle mount and map square coordinates is complicated and handled by
-the `vehicle::coord_translate()` and `vehicle::mount_to_tripoint()` families of
-functions.
+Vehicle facing is implemented via a combination of rotations (by quarter turns) and shearing to
+interpolate between quarter turns. The logic to convert between vehicle mount and map square
+coordinates is complicated and handled by the `vehicle::coord_translate()` and
+`vehicle::mount_to_tripoint()` families of functions.
-Currently, vehicle mount coordinates do not have a z-level component, but
-vehicle map square coordinates do. The z coordinate is relative to the vehicle
-origin.
+Currently, vehicle mount coordinates do not have a z-level component, but vehicle map square
+coordinates do. The z coordinate is relative to the vehicle origin.
## Point types
-To work with these coordinate systems we have a variety of types. These are
-defined in [`coordinates.h`](../src/coordinates.h). For example, we have
-`point_abs_ms` for absolute map-square coordinates. The three parts of the
-type name are *dimension*`_`*origin*`_`*scale*.
-
-* **dimension** is either `point` for two-dimensional or `tripoint` for
- three-dimensional.
-* **origin** specifies what the value is relative to, and can be:
- * `rel` means relative to some arbitrary point. This is the result of
- subtracting two points with a common origin. It would be used for example
- to represent the offset between the avatar and a monster they are shooting
- at.
- * `abs` means global absolute coordinates.
- * `sm` means relative to a corner of a submap.
- * `omt` means relative to a corner of an overmap terrain.
- * `om` means relative to a corner of an overmap.
- * `veh` means relative to a vehicle origin.
-* **scale** means the scale as discussed above.
- * `ms` for map square.
- * `sm` for submap.
- * `omt` for overmap terrain.
- * `seg` for segment.
- * `om` for overmap.
- * `mnt` for vehicle mount coordinates (only relevant for the `veh` origin).
+To work with these coordinate systems we have a variety of types. These are defined in
+[`coordinates.h`](../src/coordinates.h). For example, we have `point_abs_ms` for absolute map-square
+coordinates. The three parts of the type name are _dimension_ `_` _origin_ `_` _scale_.
+
+- **dimension** is either `point` for two-dimensional or `tripoint` for three-dimensional.
+- **origin** specifies what the value is relative to, and can be:
+ - `rel` means relative to some arbitrary point. This is the result of subtracting two points with
+ a common origin. It would be used for example to represent the offset between the avatar and a
+ monster they are shooting at.
+ - `abs` means global absolute coordinates.
+ - `sm` means relative to a corner of a submap.
+ - `omt` means relative to a corner of an overmap terrain.
+ - `om` means relative to a corner of an overmap.
+ - `veh` means relative to a vehicle origin.
+- **scale** means the scale as discussed above.
+ - `ms` for map square.
+ - `sm` for submap.
+ - `omt` for overmap terrain.
+ - `seg` for segment.
+ - `om` for overmap.
+ - `mnt` for vehicle mount coordinates (only relevant for the `veh` origin).
## Raw point types
-As well as these types with origin and scale encoded into the type, there are
-simple raw point types called just `point` and `tripoint`. These can be used
-when no particular game scale is intended.
+As well as these types with origin and scale encoded into the type, there are simple raw point types
+called just `point` and `tripoint`. These can be used when no particular game scale is intended.
-At time of writing we are still in the process of transitioning the codebase
-away from using these raw point types everywhere, so you are likely to see
-legacy code using them in places where the more type-safe points might seem
-appropriate.
+At time of writing we are still in the process of transitioning the codebase away from using these
+raw point types everywhere, so you are likely to see legacy code using them in places where the more
+type-safe points might seem appropriate.
-New code should prefer to use the types which include their coordinate system
-where feasible.
+New code should prefer to use the types which include their coordinate system where feasible.
## Converting between point types
### Changing scale
-To change the scale of a point without changing its origin, use `project_to`.
-For example:
+To change the scale of a point without changing its origin, use `project_to`. For example:
```c++
point_abs_ms pos_ms = get_avatar()->global_square_location().xy();
@@ -131,17 +117,15 @@ point_abs_omt pos_omt = project_to( pos_ms );
assert( pos_omt == get_avatar()->global_omt_location().xy() );
```
-The same function `project_to` can be used for scaling up or down. When
-converting to a coarser coordinate system precision is of course lost. If you
-care about the remainder then you must instead use `project_remain`.
+The same function `project_to` can be used for scaling up or down. When converting to a coarser
+coordinate system precision is of course lost. If you care about the remainder then you must instead
+use `project_remain`.
-`project_remain` allows you to convert to a coarser coordinate system and also
-capture the remainder relative to that coarser point. It returns a helper
-struct intended to be used with
-[`std::tie`](https://en.cppreference.com/w/cpp/utility/tuple/tie) to capture
-the two parts of the result. For example, suppose you want to know which
-overmap the avatar is in, and which overmap terrain they are in within that
-overmap.
+`project_remain` allows you to convert to a coarser coordinate system and also capture the remainder
+relative to that coarser point. It returns a helper struct intended to be used with
+[`std::tie`](https://en.cppreference.com/w/cpp/utility/tuple/tie) to capture the two parts of the
+result. For example, suppose you want to know which overmap the avatar is in, and which overmap
+terrain they are in within that overmap.
```c++
point_abs_omt abs_pos = get_avatar()->global_omt_location().xy();
@@ -150,14 +134,12 @@ point_om_omt omt_within_overmap;
std::tie( overmap, omt_within_overmap ) = project_remain( abs_pos );
```
-That makes sense for two-dimensional `point` types, but how does it handle
-`tripoint`? Recall that the z-coordinates do not scale along with the
-horizontal dimensions, so `z` values are unchanged by `project_to` and
-`project_remain`. However, for `project_remain` we don't want to duplicate the
-z-coordinate in both parts of the result, so you must choose exactly one to be
-a `tripoint`. In the example above, z-coodinates do not have much meaning at
-the overmap scale, so you probably want the z-coordinate in
-`omt_within_overmap`. That can be done as follows:
+That makes sense for two-dimensional `point` types, but how does it handle `tripoint`? Recall that
+the z-coordinates do not scale along with the horizontal dimensions, so `z` values are unchanged by
+`project_to` and `project_remain`. However, for `project_remain` we don't want to duplicate the
+z-coordinate in both parts of the result, so you must choose exactly one to be a `tripoint`. In the
+example above, z-coodinates do not have much meaning at the overmap scale, so you probably want the
+z-coordinate in `omt_within_overmap`. That can be done as follows:
```c++
tripoint_abs_omt abs_pos = get_avatar()->global_omt_location();
@@ -166,11 +148,10 @@ tripoint_om_omt omt_within_overmap;
std::tie( overmap, omt_within_overmap ) = project_remain( abs_pos );
```
-The last available operation for rescaling points is `project_combine`. This
-performs the opposite operation from `project_remain`. Given two points where
-the origin of the second matches the scale of the first, you can combine them
-into a single value. As you might expect from the above discussion, one of
-these two can be a `tripoint`, but not both.
+The last available operation for rescaling points is `project_combine`. This performs the opposite
+operation from `project_remain`. Given two points where the origin of the second matches the scale
+of the first, you can combine them into a single value. As you might expect from the above
+discussion, one of these two can be a `tripoint`, but not both.
```c++
tripoint_abs_omt abs_pos = get_avatar()->global_omt_location();
@@ -183,25 +164,23 @@ assert( abs_pos == abs_pos_again );
### Changing origin
-`project_remain` and `project_combine` facilitate some changes of origin, but
-only those origins specifically related to rescaling. To convert to or from
-local or vehicle coordinates requires a specific `map` or `vehicle` object.
+`project_remain` and `project_combine` facilitate some changes of origin, but only those origins
+specifically related to rescaling. To convert to or from local or vehicle coordinates requires a
+specific `map` or `vehicle` object.
TODO: write some examples once this is implemented.
## Point operations
-We provide standard arithmetic operations as overloaded operators, but limit
-them to prevent bugs. For example, most point types cannot be multiplied by a
-constant, but ones with the `rel` origin can (it makes sense to say "half as
-far in the same direction").
+We provide standard arithmetic operations as overloaded operators, but limit them to prevent bugs.
+For example, most point types cannot be multiplied by a constant, but ones with the `rel` origin can
+(it makes sense to say "half as far in the same direction").
-Similarly, you can't generally add two points together, but you can when one of
-them has the `rel` origin, or if one of them is a raw point type.
+Similarly, you can't generally add two points together, but you can when one of them has the `rel`
+origin, or if one of them is a raw point type.
-For computing distances a variety of functions are available, depending on your
-requirements: `square_dist`, `trig_dist`, `rl_dist`, `manhattan_dist`. Other
-related utility functions include `direction_from` and `line_to`.
+For computing distances a variety of functions are available, depending on your requirements:
+`square_dist`, `trig_dist`, `rl_dist`, `manhattan_dist`. Other related utility functions include
+`direction_from` and `line_to`.
-To iterate over nearby points of the same type you can use
-`closest_points_first`.
+To iterate over nearby points of the same type you can use `closest_points_first`.
diff --git a/doc/POSTAPOC_PRICE_GUIDE.md b/doc/POSTAPOC_PRICE_GUIDE.md
index 21936530b661..a9856a2f764c 100644
--- a/doc/POSTAPOC_PRICE_GUIDE.md
+++ b/doc/POSTAPOC_PRICE_GUIDE.md
@@ -1,23 +1,26 @@
# How to give items a sensible postapoc_price value
### Pricing philosophy
-Prices were based on the philosophy that there is a decent portion of humanity unwilling to regularly
-brave the risks of city scavenging, that are always on the lookout for supplies to stay alive.
-This mean food and medicine are relatively quite valuable compared to the goods you would use to acquire them.
-**The standard unit of currency**, the free merchants note, is priced to be the equivalent to
-**one piece of meat jerky (250 cents)**.
+Prices were based on the philosophy that there is a decent portion of humanity unwilling to
+regularly brave the risks of city scavenging, that are always on the lookout for supplies to stay
+alive. This mean food and medicine are relatively quite valuable compared to the goods you would use
+to acquire them.
+
+**The standard unit of currency**, the free merchants note, is priced to be the equivalent to **one
+piece of meat jerky (250 cents)**.
One piece of meat jerky will stay fresh for 24 days, a decent time, and contains 347 calories.
-A survivor that bring back 2000 cents worth of goods back to the merchants every day is
-doing well enough to eke out a subsistence living.
+A survivor that bring back 2000 cents worth of goods back to the merchants every day is doing well
+enough to eke out a subsistence living.
A survivor that brings back 4000 cents worth of goods every day is doing quite well for himself.
#### Items are priced on a combination of their availability and their utility.
-Raw materials tend to either be worthless, or only somewhat worthwhile if you can get them in bulk ~(0-25)/kg.
+Raw materials tend to either be worthless, or only somewhat worthwhile if you can get them in bulk
+~(0-25)/kg.
Common items with no utility are worthless.
@@ -31,10 +34,12 @@ Rare items with some theoretical utility are worth ~(100-500).
Rare items with considerable use tend to have the highest variety in prices ~(1000-15000).
-**No item should be worth more than 150000.** They might be worth more if the economy was larger but that's
-pretty much the most anyone will be willing to spend on any one thing, no matter how nice it is.
+**No item should be worth more than 150000.** They might be worth more if the economy was larger but
+that's pretty much the most anyone will be willing to spend on any one thing, no matter how nice it
+is.
#### Currently Implemented faction currencies
+
"id": "FMCNote",
"description": "The Free Merchant Certified Note, also known by names such as a 'c-note' or 'merch', is a currency based on old American bills. Fifty dollar bills and larger are printed with a promissory note signed by the treasurer of the Free Merchants, along with a complex design. The note explains that this can be exchanged for food, water, and other services through the Free Merchants in the Refugee Center.",
"name": { "str": "Merch" },
@@ -45,8 +50,8 @@ pretty much the most anyone will be willing to spend on any one thing, no matter
"name": { "str": "Hub 01 Gold Coin" },
"description": "This is a small but surprisingly heavy gold coin. One side is etched with circuitry and the other side reads 'Hub 01 exchange currency'.",
"price_postapoc": 5000
-
-
+
+
"id": "FlatCoin",
"name": { "str": "FlatCoin" },
"description": "This is a coin that has been flattened in a novelty coin flattening machine. The machine has been somewhat crudely altered so that the design - which appears to once have been Mickey Mouse - is overlaid with a handwritten emblem of a book. There is some text that faintly reads 'Campus Exchange Token'.",
@@ -63,43 +68,44 @@ pretty much the most anyone will be willing to spend on any one thing, no matter
"name": { "str": "icon" },
"description": "This is a small picture, about the same size as an ID card, symbolizing a religious figure. On the back, there is a text that faintly reads 'New England Church Community'.",
"price_postapoc": 250
-
-
+
##### Some benchmark prices
```
- {
- "id": "antibiotics",
- "name": { "str_sp": "antibiotics" },
- "description": "A strong antibacterial medication designed to prevent or stop the spread of infection. It's the safest way to cure any infections you might have. One dose lasts twelve hours.",
- "price_postapoc": 40000,
- "charges": 15,
- "stack_size": 200,
- "flags": [ "NPC_SAFE", "IRREPLACEABLE_CONSUMABLE" ]
- },
- {
- "id": "boltcutters",
- "type": "TOOL",
- "name": { "str": "pair of bolt cutters", "str_pl": "pairs of bolt cutters" },
- "description": "This is a large pair of bolt cutters. You could use them to cut padlocks or heavy gauge wire.",
- "price_postapoc": 250,
- },
- {
- "id": "armor_lightplate",
- "name": { "str": "plate armor" },
- "description": "A suit of Gothic plate armor.",
- "price_postapoc": 12000,
- },
- {
- "id": "bat_metal",
- "name": { "str": "aluminum bat" },
- "description": "An aluminum baseball bat, lighter than a wooden bat and a little easier to swing as a result.",
- "price_postapoc": 1250,
- },
+{
+ "id": "antibiotics",
+ "name": { "str_sp": "antibiotics" },
+ "description": "A strong antibacterial medication designed to prevent or stop the spread of infection. It's the safest way to cure any infections you might have. One dose lasts twelve hours.",
+ "price_postapoc": 40000,
+ "charges": 15,
+ "stack_size": 200,
+ "flags": [ "NPC_SAFE", "IRREPLACEABLE_CONSUMABLE" ]
+},
+{
+ "id": "boltcutters",
+ "type": "TOOL",
+ "name": { "str": "pair of bolt cutters", "str_pl": "pairs of bolt cutters" },
+ "description": "This is a large pair of bolt cutters. You could use them to cut padlocks or heavy gauge wire.",
+ "price_postapoc": 250,
+},
+{
+ "id": "armor_lightplate",
+ "name": { "str": "plate armor" },
+ "description": "A suit of Gothic plate armor.",
+ "price_postapoc": 12000,
+},
+{
+ "id": "bat_metal",
+ "name": { "str": "aluminum bat" },
+ "description": "An aluminum baseball bat, lighter than a wooden bat and a little easier to swing as a result.",
+ "price_postapoc": 1250,
+},
```
#### Food pricing
-Food is priced mostly for a combination of calories, how preservable they are, and how nutritious they are.
+
+Food is priced mostly for a combination of calories, how preservable they are, and how nutritious
+they are.
Meat jerky is 250.
@@ -108,14 +114,19 @@ Food that has more calories, is canned, and is very nutritious might be 500.
Food that has less calories, goes bad quickly, and is junk food, might be 50.
#### Stack size
-For items that have a stack size, you need to **divide** the price by stack size find the price per unit of the item.
+
+For items that have a stack size, you need to **divide** the price by stack size find the price per
+unit of the item.
#### IRREPLACEABLE_CONSUMABLE Flag
+
Pre cataclysm consumables that can not be replaced can be given the flag:
+
```
- "flags": [ "IRREPLACEABLE_CONSUMABLE" ],
+"flags": [ "IRREPLACEABLE_CONSUMABLE" ],
```
-This is used for things such as unreloaded ammo, medicine, and luxury consumables such as coffee and tea.
+This is used for things such as unreloaded ammo, medicine, and luxury consumables such as coffee and
+tea.
In the future we can implement a feature that will allow this items to increase in price over time.
diff --git a/doc/REGION_SETTINGS.md b/doc/REGION_SETTINGS.md
index 90fb07918cd7..9e95659ade61 100644
--- a/doc/REGION_SETTINGS.md
+++ b/doc/REGION_SETTINGS.md
@@ -1,10 +1,10 @@
# Region Settings
-The **region_settings** define the attributes for map generation that apply to an entire region.
-The general settings define the default overmap terrain and ground cover. Additional sections are
-as follows:
+The **region_settings** define the attributes for map generation that apply to an entire region. The
+general settings define the default overmap terrain and ground cover. Additional sections are as
+follows:
-| Section | Description |
+| Section | Description |
| ------------------------------- | --------------------------------------------------------------------- |
| `region_terrain_and_furniture` | Defines the resolution of regional terrain/furniture to actual types. |
| `field_coverage` | Defines the flora that cover the `field` overmap terrain. |
@@ -21,78 +21,78 @@ Note that for the default region, all attributes and sections are required.
### Fields
-| Identifier | Description |
-| ----------------------- | ------------------------------------------------------------------ |
-| `type` | Type identifier. Must be "region_settings". |
-| `id` | Unique identfier for this region. |
-| `default_oter` | Default overmap terrain for this region. |
-| `default_groundcover` | List of terrain types and weights applied as default ground cover. |
-
+| Identifier | Description |
+| --------------------- | ------------------------------------------------------------------ |
+| `type` | Type identifier. Must be "region_settings". |
+| `id` | Unique identfier for this region. |
+| `default_oter` | Default overmap terrain for this region. |
+| `default_groundcover` | List of terrain types and weights applied as default ground cover. |
### Example
+
```json
{
- "type": "region_settings",
- "id": "default",
- "default_oter": "field",
- "default_groundcover": [
- ["t_grass", 4],
- ["t_dirt", 1]
- ]
+ "type": "region_settings",
+ "id": "default",
+ "default_oter": "field",
+ "default_groundcover": [
+ ["t_grass", 4],
+ ["t_dirt", 1]
+ ]
}
```
## Region Terrain / Furniture
-The **region_terrain_and_furniture** section defines the resolution of regional terrain/furniture
-to their actual terrain and furniture types for the region, with a weighted list for
-terrain/furniture entry that defines the relative weight of a given entry when mapgen resolves the
-regional entry to an actual entry.
+The **region_terrain_and_furniture** section defines the resolution of regional terrain/furniture to
+their actual terrain and furniture types for the region, with a weighted list for terrain/furniture
+entry that defines the relative weight of a given entry when mapgen resolves the regional entry to
+an actual entry.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ----------- | ------------------------------------------------------------------ |
| `terrain` | List of regional terrain and their corresponding weighted lists. |
| `furniture` | List of regional furniture and their corresponding weighted lists. |
### Example
+
```json
{
- "region_terrain_and_furniture": {
- "terrain": {
- "t_region_groundcover": {
- "t_grass": 4,
- "t_grass_long": 2,
- "t_dirt": 1
- }
- },
- "furniture": {
- "f_region_flower": {
- "f_black_eyed_susan": 1,
- "f_lily": 1,
- "f_flower_tulip": 1,
- "f_flower_spurge": 1,
- "f_chamomile": 1,
- "f_dandelion": 1,
- "f_datura": 1,
- "f_dahlia": 1,
- "f_bluebell": 1
- }
- }
- }
+ "region_terrain_and_furniture": {
+ "terrain": {
+ "t_region_groundcover": {
+ "t_grass": 4,
+ "t_grass_long": 2,
+ "t_dirt": 1
+ }
+ },
+ "furniture": {
+ "f_region_flower": {
+ "f_black_eyed_susan": 1,
+ "f_lily": 1,
+ "f_flower_tulip": 1,
+ "f_flower_spurge": 1,
+ "f_chamomile": 1,
+ "f_dandelion": 1,
+ "f_datura": 1,
+ "f_dahlia": 1,
+ "f_bluebell": 1
+ }
+ }
+ }
}
```
-
## Field Coverage
-The **field_coverage** section defines the furniture and terrain that make up the flora that
-cover the `field` overmap terrain.
+The **field_coverage** section defines the furniture and terrain that make up the flora that cover
+the `field` overmap terrain.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| -------------------------- | ---------------------------------------------------------------------------- |
| `percent_coverage` | % of tiles in the overmap terrain that have a plant. |
| `default_ter` | Default terrain feature for plants. |
@@ -103,24 +103,25 @@ cover the `field` overmap terrain.
| `boosted_other_percent` | % of `boosted_percent_coverage` that will be covered by `boosted_other`. |
### Example
+
```json
{
- "field_coverage": {
- "percent_coverage": 0.9333,
- "default_ter": "t_shrub",
- "other": {
- "t_shrub_blueberry": 0.4166,
- "t_shrub_strawberry": 0.4166,
- "f_mutpoppy": 8.3333
- },
- "boost_chance": 0.833,
- "boosted_percent_coverage": 2.5,
- "boosted_other": {
- "t_shrub_blueberry": 40.0,
- "f_dandelion": 6.6
- },
- "boosted_other_percent": 50.0
- }
+ "field_coverage": {
+ "percent_coverage": 0.9333,
+ "default_ter": "t_shrub",
+ "other": {
+ "t_shrub_blueberry": 0.4166,
+ "t_shrub_strawberry": 0.4166,
+ "f_mutpoppy": 8.3333
+ },
+ "boost_chance": 0.833,
+ "boosted_percent_coverage": 2.5,
+ "boosted_other": {
+ "t_shrub_blueberry": 40.0,
+ "f_dandelion": 6.6
+ },
+ "boosted_other_percent": 50.0
+ }
}
```
@@ -128,12 +129,12 @@ cover the `field` overmap terrain.
The **overmap_lake_settings** section defines the attributes used in generating lakes on the
overmap. The actual placement of these features is determined globally across all overmaps so that
-the edges of the features align, and these parameters are mostly about how those global features
-are interpreted.
+the edges of the features align, and these parameters are mostly about how those global features are
+interpreted.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ------------------------------------------ | --------------------------------------------------------------------------- |
| `noise_threshold_lake` | [0, 1], x > value spawns a `lake_surface` or `lake_shore`. |
| `lake_size_min` | Minimum size of the lake in overmap terrains for it to actually spawn. |
@@ -145,28 +146,28 @@ are interpreted.
```json
{
- "overmap_lake_settings": {
- "noise_threshold_lake": 0.25,
- "lake_size_min": 20,
- "lake_depth": -5,
- "shore_extendable_overmap_terrain": ["forest_thick", "forest_water", "field"],
- "shore_extendable_overmap_terrain_aliases": [
- { "om_terrain": "forest", "om_terrain_match_type": "TYPE", "alias": "forest_thick" }
- ]
- }
+ "overmap_lake_settings": {
+ "noise_threshold_lake": 0.25,
+ "lake_size_min": 20,
+ "lake_depth": -5,
+ "shore_extendable_overmap_terrain": ["forest_thick", "forest_water", "field"],
+ "shore_extendable_overmap_terrain_aliases": [
+ { "om_terrain": "forest", "om_terrain_match_type": "TYPE", "alias": "forest_thick" }
+ ]
+ }
}
```
## Overmap Forest Settings
The **overmap_forest_settings** section defines the attributes used in generating forest and swamps
-on the overmap. The actual placement of these features is determined globally across all overmaps
-so that the edges of the features align, and these parameters are mostly about how those global
+on the overmap. The actual placement of these features is determined globally across all overmaps so
+that the edges of the features align, and these parameters are mostly about how those global
features are interpreted.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| -------------------------------------- | ---------------------------------------------------------------------- |
| `noise_threshold_forest` | [0, 1], x > value spawns `forest`. |
| `noise_threshold_forest_thick` | [0, 1], x > value spawns `forest_thick`. |
@@ -179,37 +180,36 @@ features are interpreted.
```json
{
- "overmap_forest_settings": {
- "noise_threshold_forest": 0.25,
- "noise_threshold_forest_thick": 0.3,
- "noise_threshold_swamp_adjacent_water": 0.3,
- "noise_threshold_swamp_isolated": 0.6,
- "river_floodplain_buffer_distance_min": 3,
- "river_floodplain_buffer_distance_max": 15
- }
+ "overmap_forest_settings": {
+ "noise_threshold_forest": 0.25,
+ "noise_threshold_forest_thick": 0.3,
+ "noise_threshold_swamp_adjacent_water": 0.3,
+ "noise_threshold_swamp_isolated": 0.6,
+ "river_floodplain_buffer_distance_min": 3,
+ "river_floodplain_buffer_distance_max": 15
+ }
}
```
## Forest Map Generation Settings
The **forest_mapgen_settings** section defines the attributes used in generating forest (`forest`,
-`forest_thick`, `forest_water`) terrains, including their items, groundcover, terrain and
-furniture.
+`forest_thick`, `forest_water`) terrains, including their items, groundcover, terrain and furniture.
### General Structure
At the top level, the `forest_mapgen_settings` is a collection of named configurations where each
entry has the name of the overmap terrain that it applies to, e.g. `forest`, `forest_thick`,
-`forest_water`. It is possible to define settings for overmap terrains that are not rendered by
-the forest mapgen, but will be used when blending forest terrains with other terrain types.
+`forest_water`. It is possible to define settings for overmap terrains that are not rendered by the
+forest mapgen, but will be used when blending forest terrains with other terrain types.
```json
{
- "forest_mapgen_settings": {
- "forest": {},
- "forest_thick": {},
- "forest_water": {}
- }
+ "forest_mapgen_settings": {
+ "forest": {},
+ "forest_thick": {},
+ "forest_water": {}
+ }
}
```
@@ -217,7 +217,7 @@ Each terrain then has an independent set of configuration values that control th
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ----------------------------- | ---------------------------------------------------------------------------- |
| `sparseness_adjacency_factor` | Value relative to neighbors controls how sparse the overmap terrain will be. |
| `item_group` | Item group used to place items randomly within the overmap terrain. |
@@ -234,21 +234,21 @@ Each terrain then has an independent set of configuration values that control th
```json
{
- "forest": {
- "sparseness_adjacency_factor": 3,
- "item_group": "forest",
- "item_group_chance": 60,
- "item_spawn_iterations": 1,
- "clear_groundcover": false,
- "groundcover": {
- "t_grass": 3,
- "t_dirt": 1
- },
- "clear_components": false,
- "components": {},
- "clear_terrain_furniture": false,
- "terrain_furniture": {}
- }
+ "forest": {
+ "sparseness_adjacency_factor": 3,
+ "item_group": "forest",
+ "item_group_chance": 60,
+ "item_spawn_iterations": 1,
+ "clear_groundcover": false,
+ "groundcover": {
+ "t_grass": 3,
+ "t_dirt": 1
+ },
+ "clear_components": false,
+ "components": {},
+ "clear_terrain_furniture": false,
+ "terrain_furniture": {}
+ }
}
```
@@ -260,7 +260,7 @@ for the components are only relevant for the purposes of overriding them in regi
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ------------- | -------------------------------------------------------------------- |
| `sequence` | Sequence in which components are processed. |
| `chance` | One in X chance that something from this component will be placed. |
@@ -271,63 +271,62 @@ for the components are only relevant for the purposes of overriding them in regi
```json
{
- "trees": {
- "sequence": 0,
- "chance": 12,
- "clear_types": false,
- "types": {
- "t_tree_young": 128,
- "t_tree": 32,
- "t_tree_birch": 32,
- "t_tree_pine": 32,
- "t_tree_maple": 32,
- "t_tree_willow": 32,
- "t_tree_hickory": 32,
- "t_tree_blackjack": 8,
- "t_tree_coffee": 8,
- "t_tree_apple": 2,
- "t_tree_apricot": 2,
- "t_tree_cherry": 2,
- "t_tree_peach": 2,
- "t_tree_pear": 2,
- "t_tree_plum": 2,
- "t_tree_deadpine": 1,
- "t_tree_hickory_dead": 1,
- "t_tree_dead": 1
- }
- },
- "shrubs_and_flowers": {
- "sequence": 1,
- "chance": 10,
- "clear_types": false,
- "types": {
- "t_underbrush": 8,
- "t_shrub_blueberry": 1,
- "t_shrub_strawberry": 1,
- "t_shrub": 1,
- "f_chamomile": 1,
- "f_dandelion": 1,
- "f_datura": 1,
- "f_dahlia": 1,
- "f_bluebell": 1,
- "f_mutpoppy": 1
- }
- }
+ "trees": {
+ "sequence": 0,
+ "chance": 12,
+ "clear_types": false,
+ "types": {
+ "t_tree_young": 128,
+ "t_tree": 32,
+ "t_tree_birch": 32,
+ "t_tree_pine": 32,
+ "t_tree_maple": 32,
+ "t_tree_willow": 32,
+ "t_tree_hickory": 32,
+ "t_tree_blackjack": 8,
+ "t_tree_coffee": 8,
+ "t_tree_apple": 2,
+ "t_tree_apricot": 2,
+ "t_tree_cherry": 2,
+ "t_tree_peach": 2,
+ "t_tree_pear": 2,
+ "t_tree_plum": 2,
+ "t_tree_deadpine": 1,
+ "t_tree_hickory_dead": 1,
+ "t_tree_dead": 1
+ }
+ },
+ "shrubs_and_flowers": {
+ "sequence": 1,
+ "chance": 10,
+ "clear_types": false,
+ "types": {
+ "t_underbrush": 8,
+ "t_shrub_blueberry": 1,
+ "t_shrub_strawberry": 1,
+ "t_shrub": 1,
+ "f_chamomile": 1,
+ "f_dandelion": 1,
+ "f_datura": 1,
+ "f_dahlia": 1,
+ "f_bluebell": 1,
+ "f_mutpoppy": 1
+ }
+ }
}
```
### Terrain Furniture
-The terrain furniture are a collection of terrain ids with a chance of having furniture
-picked from a weighted list for that given terrain and placed on it during mapgen after
-the normal mapgen has completed. This is used, for example, to place cattails on fresh
-water in swamps. Cattails could be simply placed in the `components` section and placed
-during the normal forest mapgen, but that would not guarantee their placement on fresh
-water only, while this does.
+The terrain furniture are a collection of terrain ids with a chance of having furniture picked from
+a weighted list for that given terrain and placed on it during mapgen after the normal mapgen has
+completed. This is used, for example, to place cattails on fresh water in swamps. Cattails could be
+simply placed in the `components` section and placed during the normal forest mapgen, but that would
+not guarantee their placement on fresh water only, while this does.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ----------------- | ------------------------------------------------------------------ |
| `chance` | One in X chance that furniture from this component will be placed. |
| `clear_furniture` | Clear all previously defined `furniture` for this terrain. |
@@ -337,13 +336,13 @@ water only, while this does.
```json
{
- "t_water_sh" : {
- "chance": 2,
- "clear_furniture": false,
- "furniture": {
- "f_cattails": 1
- }
- }
+ "t_water_sh": {
+ "chance": 2,
+ "clear_furniture": false,
+ "furniture": {
+ "f_cattails": 1
+ }
+ }
}
```
@@ -355,7 +354,7 @@ trailheads, and some general tuning of the actual trail width/position in mapgen
### Fields
-| Identifier | Description |
+| Identifier | Description |
| -------------------------- | ------------------------------------------------------------------------------------------- |
| `chance` | One in X chance a contiguous forest will have a trail system. |
| `border_point_chance` | One in X chance that the N/S/E/W-most point of the forest will be part of the trail system. |
@@ -376,26 +375,26 @@ trailheads, and some general tuning of the actual trail width/position in mapgen
```json
{
- "forest_trail_settings": {
- "chance": 2,
- "border_point_chance": 2,
- "minimum_forest_size": 100,
- "random_point_min": 4,
- "random_point_max": 50,
- "random_point_size_scalar": 100,
- "trailhead_chance": 1,
- "trailhead_road_distance": 6,
- "trail_center_variance": 3,
- "trail_width_offset_min": 1,
- "trail_width_offset_max": 3,
- "clear_trail_terrain": false,
- "trail_terrain": {
- "t_dirt": 1
- },
- "trailheads": {
- "trailhead_basic": 50
- }
- }
+ "forest_trail_settings": {
+ "chance": 2,
+ "border_point_chance": 2,
+ "minimum_forest_size": 100,
+ "random_point_min": 4,
+ "random_point_max": 50,
+ "random_point_size_scalar": 100,
+ "trailhead_chance": 1,
+ "trailhead_road_distance": 6,
+ "trail_center_variance": 3,
+ "trail_width_offset_min": 1,
+ "trail_width_offset_max": 3,
+ "clear_trail_terrain": false,
+ "trail_terrain": {
+ "t_dirt": 1
+ },
+ "trailheads": {
+ "trailhead_basic": 50
+ }
+ }
}
```
@@ -407,46 +406,47 @@ relative placements of various classes of buildings.
### Fields
-| Identifier | Description |
-| ----------------------- | ------------------------------------------------------------------ |
-| `type` | City type identifier--currently unused. |
-| `shop_radius` | Radial frequency of shop placement. Smaller number = more shops. |
-| `park_radius` | Radial frequency of park placement. Smaller number = more parks. |
-| `houses` | Weighted list of overmap terrains and specials used for houses. |
-| `parks` | Weighted list of overmap terrains and specials used for parks. |
-| `shops` | Weighted list of overmap terrains and specials used for shops. |
+| Identifier | Description |
+| ------------- | ---------------------------------------------------------------- |
+| `type` | City type identifier--currently unused. |
+| `shop_radius` | Radial frequency of shop placement. Smaller number = more shops. |
+| `park_radius` | Radial frequency of park placement. Smaller number = more parks. |
+| `houses` | Weighted list of overmap terrains and specials used for houses. |
+| `parks` | Weighted list of overmap terrains and specials used for parks. |
+| `shops` | Weighted list of overmap terrains and specials used for shops. |
### Placing shops, parks, and houses
When picking a building to place in a given location, the game considers the city size, the
location's distance from the city center, and finally the `shop_radius` and `park_radius` values for
the region. It then tries to place a shop, then a park, and finally a house, where the chance to
-place the shop or park are based on the formula `rng( 0, 99 ) > X_radius * distance from city center
-/ city size`.
+place the shop or park are based on the formula
+`rng( 0, 99 ) > X_radius * distance from city center / city size`.
### Example
+
```json
{
- "city": {
- "type": "town",
- "shop_radius": 80,
- "park_radius": 90,
- "houses": {
- "house_two_story_basement": 1,
- "house": 1000,
- "house_base": 333,
- "emptyresidentiallot": 20
- },
- "parks": {
- "park": 4,
- "pool": 1
- },
- "shops": {
- "s_gas": 5,
- "s_pharm": 3,
- "s_grocery": 15
- }
- }
+ "city": {
+ "type": "town",
+ "shop_radius": 80,
+ "park_radius": 90,
+ "houses": {
+ "house_two_story_basement": 1,
+ "house": 1000,
+ "house_base": 333,
+ "emptyresidentiallot": 20
+ },
+ "parks": {
+ "park": 4,
+ "pool": 1
+ },
+ "shops": {
+ "s_gas": 5,
+ "s_pharm": 3,
+ "s_grocery": 15
+ }
+ }
}
```
@@ -458,7 +458,7 @@ terrain. This includes both the chance of an extra occurring as well as the weig
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ---------- | ---------------------------------------------------------------- |
| `chance` | One in X chance that the overmap terrain will spawn a map extra. |
| `extras` | Weighted list of map extras that can spawn. |
@@ -467,15 +467,15 @@ terrain. This includes both the chance of an extra occurring as well as the weig
```json
{
- "map_extras": {
- "field": {
- "chance": 90,
- "extras": {
- "mx_helicopter": 40,
- "mx_portal_in": 1
- }
- }
- }
+ "map_extras": {
+ "field": {
+ "chance": 90,
+ "extras": {
+ "mx_helicopter": 40,
+ "mx_portal_in": 1
+ }
+ }
+ }
}
```
@@ -485,20 +485,19 @@ The **weather** section defines the base weather attributes used for the region.
### Fields
-| Identifier | Description |
-| ------------------------------ | --------------------------------------------------------------------- |
-| `spring_temp` | Mid spring temperature for the region in degrees Celsius |
-| `summer_temp` | Mid summer temperature for the region in degrees Celsius |
-| `autumn_temp` | Mid autumn temperature for the region in degrees Celsius |
-| `winter_temp` | Mid winter temperature for the region in degrees Celsius |
-| `base_humidity` | Base humidity for the region in relative humidity % |
-| `base_pressure` | Base pressure for the region in millibars. |
-| `base_acid` | Base acid for the region in ? units. Value >= 1 is considered acidic. |
-| `base_wind` | Base wind for the region in mph units. Roughly the yearly average. |
-| `base_wind_distrib_peaks` | How high the wind peaks can go. Higher values produce windier days. |
-| `base_wind_season_variation` | How the wind varies with season. Lower values produce more variation |
-| `weather_types` | Ids of the weather types allowed in this region. First value will be the default weather type. Declaration order will affect weather selection, see [WEATHER_TYPE.md](WEATHER_TYPE.md) for details. |
-
+| Identifier | Description |
+| ---------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| `spring_temp` | Mid spring temperature for the region in degrees Celsius |
+| `summer_temp` | Mid summer temperature for the region in degrees Celsius |
+| `autumn_temp` | Mid autumn temperature for the region in degrees Celsius |
+| `winter_temp` | Mid winter temperature for the region in degrees Celsius |
+| `base_humidity` | Base humidity for the region in relative humidity % |
+| `base_pressure` | Base pressure for the region in millibars. |
+| `base_acid` | Base acid for the region in ? units. Value >= 1 is considered acidic. |
+| `base_wind` | Base wind for the region in mph units. Roughly the yearly average. |
+| `base_wind_distrib_peaks` | How high the wind peaks can go. Higher values produce windier days. |
+| `base_wind_season_variation` | How the wind varies with season. Lower values produce more variation |
+| `weather_types` | Ids of the weather types allowed in this region. First value will be the default weather type. Declaration order will affect weather selection, see [WEATHER_TYPE.md](WEATHER_TYPE.md) for details. |
### Example
@@ -538,12 +537,13 @@ The **weather** section defines the base weather attributes used for the region.
## Overmap Feature Flag Settings
-The **overmap_feature_flag_settings** section defines operations that operate on the flags assigned to overmap features.
-This is currently used to provide a mechanism for whitelisting and blacklisting locations on a per-region basis.
+The **overmap_feature_flag_settings** section defines operations that operate on the flags assigned
+to overmap features. This is currently used to provide a mechanism for whitelisting and blacklisting
+locations on a per-region basis.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ----------------- | ------------------------------------------------------------------------------------------ |
| `clear_blacklist` | Clear all previously defined `blacklist`. |
| `blacklist` | List of flags. Any location with a matching flag will be excluded from overmap generation. |
@@ -554,12 +554,12 @@ This is currently used to provide a mechanism for whitelisting and blacklisting
```json
{
- "overmap_feature_flag_settings": {
- "clear_blacklist": false,
- "blacklist": [ "FUNGAL" ],
- "clear_whitelist": false,
- "whitelist": []
- }
+ "overmap_feature_flag_settings": {
+ "clear_blacklist": false,
+ "blacklist": ["FUNGAL"],
+ "clear_whitelist": false,
+ "whitelist": []
+ }
}
```
@@ -571,7 +571,7 @@ those values which should be changed.
### Fields
-| Identifier | Description |
+| Identifier | Description |
| ---------- | ------------------------------------------------------------------------------------------- |
| `type` | Type identifier. Must be "region_overlay". |
| `id` | Unique identfier for this region overlay. |
@@ -580,15 +580,16 @@ those values which should be changed.
All additional fields and sections are as defined for a `region_overlay`.
### Example
+
```json
[{
- "type": "region_overlay",
- "id": "example_overlay",
- "regions": ["all"],
- "city": {
- "parks": {
- "examplepark": 1
- }
- }
+ "type": "region_overlay",
+ "id": "example_overlay",
+ "regions": ["all"],
+ "city": {
+ "parks": {
+ "examplepark": 1
+ }
+ }
}]
-```
\ No newline at end of file
+```
diff --git a/doc/RELICS.md b/doc/RELICS.md
index d62e50bec020..b2ffb54f3397 100644
--- a/doc/RELICS.md
+++ b/doc/RELICS.md
@@ -7,10 +7,11 @@
Relics are regular items with special relic data attached to them.
-Relic data is defined in JSON in `relic_data` field of corresponding item type definition,
-and whenever the item is spawned a copy of relic data is attached to the item instance.
+Relic data is defined in JSON in `relic_data` field of corresponding item type definition, and
+whenever the item is spawned a copy of relic data is attached to the item instance.
Relic data object can contain the following fields:
+
```c++
{
"name": "Boots of Haste", // Overrides default item name
@@ -24,8 +25,9 @@ Relic data object can contain the following fields:
## Relic recharge
-Relics can recharge under certain conditions.
-Recharge method is defined as follows (all fields optional):
+Relics can recharge under certain conditions. Recharge method is defined as follows (all fields
+optional):
+
```c++
{
"type": "time", // Defines what resource is consumed. Default: time
@@ -42,24 +44,24 @@ Recharge method is defined as follows (all fields optional):
### Recharge type
-| ID | Description
-|----------------|--------------------------------
-|`time` | Needs no additional resources
-|`solar` | Consumes sunlight (character must be in sunlight)
-|`pain` | Causes pain to recharge. Intensity controlled by `int_min` and `int_max`
-|`hp` | Causes damage to all body parts. Intensity controlled by `int_min` and `int_max`
-|`fatigue` | Causes fatigue and drains stamina. Fatigue drain controlled by `int_min` and `int_max`, stamina drain rolled as `[ int_min*100, int_max*100 ]`
-|`field` | Consumes adjacent field. Allowed field intensity controlled by `int_min` and `int_max`
-|`trap` | Consumes adjacent trap.
+| ID | Description |
+| --------- | ---------------------------------------------------------------------------------------------------------------------------------------------- |
+| `time` | Needs no additional resources |
+| `solar` | Consumes sunlight (character must be in sunlight) |
+| `pain` | Causes pain to recharge. Intensity controlled by `int_min` and `int_max` |
+| `hp` | Causes damage to all body parts. Intensity controlled by `int_min` and `int_max` |
+| `fatigue` | Causes fatigue and drains stamina. Fatigue drain controlled by `int_min` and `int_max`, stamina drain rolled as `[ int_min*100, int_max*100 ]` |
+| `field` | Consumes adjacent field. Allowed field intensity controlled by `int_min` and `int_max` |
+| `trap` | Consumes adjacent trap. |
### Recharge requirements
-| ID | Description
-|----------------|--------------------------------
-|`none` | No additional requirements (always works)
-|`equipped` | Must be worn if armor, wielded if weapon.
-|`close_to_skin` | Must be worn underneath all other clothing, or be wielded with bare hands
-|`sleep` | Character must be asleep
-|`rad` | Character or map tile must be irradiated
-|`wet` | Character must be wet, or it's raining
-|`sky` | Character must be above z=0
+| ID | Description |
+| --------------- | ------------------------------------------------------------------------- |
+| `none` | No additional requirements (always works) |
+| `equipped` | Must be worn if armor, wielded if weapon. |
+| `close_to_skin` | Must be worn underneath all other clothing, or be wielded with bare hands |
+| `sleep` | Character must be asleep |
+| `rad` | Character or map tile must be irradiated |
+| `wet` | Character must be wet, or it's raining |
+| `sky` | Character must be above z=0 |
diff --git a/doc/SOUNDPACKS.md b/doc/SOUNDPACKS.md
index 0e757dc8d116..6f21ca751a1b 100644
--- a/doc/SOUNDPACKS.md
+++ b/doc/SOUNDPACKS.md
@@ -1,10 +1,13 @@
# Soundpacks
-A soundpack can be installed in the `data/sound` directory. It has to be a subdirectory that contains at least a file named `soundpack.txt`. It can include any number of json files which add any number of sound_effect or playlist.
+A soundpack can be installed in the `data/sound` directory. It has to be a subdirectory that
+contains at least a file named `soundpack.txt`. It can include any number of json files which add
+any number of sound_effect or playlist.
## soundpack.txt format
-The `soundpack.txt` file format needs 2 values, a NAME and a VIEW. The NAME value should be unique. The VIEW value will be shown in the options menu. Every line starting with a `#` is a comment line.
+The `soundpack.txt` file format needs 2 values, a NAME and a VIEW. The NAME value should be unique.
+The VIEW value will be shown in the options menu. Every line starting with a `#` is a comment line.
```
#Basic provided soundpack
@@ -21,53 +24,59 @@ VIEW: Basic
Sound effects can be included with a format like this:
```javascript
-[
- {
- "type": "sound_effect",
- "id" : "menu_move",
- "volume" : 100,
- "files" : [
- "nenadsimic_menu_selection_click.wav"
- ]
- },
- {
- "type": "sound_effect",
- "id" : "fire_gun",
- "volume" : 90,
- "variant" : "bio_laser_gun",
- "files" : [
- "guns/energy_generic/weapon_fire_laser.ogg"
- ]
- }
+;[
+ {
+ "type": "sound_effect",
+ "id": "menu_move",
+ "volume": 100,
+ "files": [
+ "nenadsimic_menu_selection_click.wav",
+ ],
+ },
+ {
+ "type": "sound_effect",
+ "id": "fire_gun",
+ "volume": 90,
+ "variant": "bio_laser_gun",
+ "files": [
+ "guns/energy_generic/weapon_fire_laser.ogg",
+ ],
+ },
]
```
-Adding variety: If for a certain `id`'s `variant` multiple `files` are defined, they will be chosen at random when `variant` is played.
+
+Adding variety: If for a certain `id`'s `variant` multiple `files` are defined, they will be chosen
+at random when `variant` is played.
The volume key may range from 0-100.
-Cataclysm has its own set of user-controllable volumes that will additionally affect the sound. These range from 0-128, and the default is 100. This means that at default volume, any sound that Cataclysm plays will default to playing at about 78% of the maximum; if you are working on sounds in an external audio editor, expect Cataclysm at default volume settings to play that sound file back more-quietly than your editor does.
+Cataclysm has its own set of user-controllable volumes that will additionally affect the sound.
+These range from 0-128, and the default is 100. This means that at default volume, any sound that
+Cataclysm plays will default to playing at about 78% of the maximum; if you are working on sounds in
+an external audio editor, expect Cataclysm at default volume settings to play that sound file back
+more-quietly than your editor does.
### Preloading SFX
Sound effects can be included for preloading with a format like this:
```javascript
-[
- {
- "type": "sound_effect_preload",
- "preload": [
- { "id": "fire_gun", "variant": "all" },
- { "id": "environment", "variant": "daytime" },
- { "id": "environment" }
- ]
- }
+;[
+ {
+ "type": "sound_effect_preload",
+ "preload": [
+ { "id": "fire_gun", "variant": "all" },
+ { "id": "environment", "variant": "daytime" },
+ { "id": "environment" },
+ ],
+ },
]
```
-`"variant": "all"` will be treated specially and load all variants of the given id.
+`"variant": "all"` will be treated specially and load all variants of the given id.
-> [!WARNING]
-> `"variant": "all"` uses unoptimal algorithm (because the devs were dumb and lazy and used hacks) and will slow down game loading time.
+> [!WARNING] `"variant": "all"` uses unoptimal algorithm (because the devs were dumb and lazy and
+> used hacks) and will slow down game loading time.
If `"variant"` is omitted, it defaults to `"default"`.
@@ -76,188 +85,201 @@ If `"variant"` is omitted, it defaults to `"default"`.
A playlist can be included with a format like this:
```javascript
-[
- {
- "type": "playlist",
- "playlists":
- [
- {
- "id" : "title",
- "shuffle" : false,
- "files" : [
- {
- "file": "Dark_Days_Ahead_demo_2.wav",
- "volume": 100
- },
- {
- "file": "cataclysmthemeREV6.wav",
- "volume": 90
- }
- ]
- }
- ]
- }
+;[
+ {
+ "type": "playlist",
+ "playlists": [
+ {
+ "id": "title",
+ "shuffle": false,
+ "files": [
+ {
+ "file": "Dark_Days_Ahead_demo_2.wav",
+ "volume": 100,
+ },
+ {
+ "file": "cataclysmthemeREV6.wav",
+ "volume": 90,
+ },
+ ],
+ },
+ ],
+ },
]
```
-Each sound effect is identified by an id and a variant. If a sound effect is played with a variant that does not exist in the json files, but a variant "default" exists, then the "default" variant is played instead. The file name of the sound effect is relative to the soundpack directory, so if the file name is set to "sfx.wav" and your soundpack is in `data/sound/mypack`, the file must be placed at `data/sound/mypack/sfx.wav`.
+Each sound effect is identified by an id and a variant. If a sound effect is played with a variant
+that does not exist in the json files, but a variant "default" exists, then the "default" variant is
+played instead. The file name of the sound effect is relative to the soundpack directory, so if the
+file name is set to "sfx.wav" and your soundpack is in `data/sound/mypack`, the file must be placed
+at `data/sound/mypack/sfx.wav`.
## JSON Format Sound Effects List
-A full list of sound effect id's and variants is given in the following. Each line in the list has the following format:
+A full list of sound effect id's and variants is given in the following. Each line in the list has
+the following format:
`id variant1|variant2`
-Where id describes the id of the sound effect, and a list of variants separated by | follows. When the variants are omitted, the variant "default" is assumed. Where the variants do not represent literal strings, but variables, they will be enclosed in `<` `>`. For instance, `` is a placeholder for any valid furniture ID (as in the furniture definition JSON).
+Where id describes the id of the sound effect, and a list of variants separated by | follows. When
+the variants are omitted, the variant "default" is assumed. Where the variants do not represent
+literal strings, but variables, they will be enclosed in `<` `>`. For instance, `` is a
+placeholder for any valid furniture ID (as in the furniture definition JSON).
# open/close doors
-* `open_door default||`
-* `close_door default||`
-
- # smashing attempts and results, few special ones and furniture/terrain specific
-* `bash default`
-* `smash wall|door|door_boarded|glass|swing|web|paper_torn|metal`
-* `smash_success hit_vehicle|smash_glass_contents|smash_cloth||`
-* `smash_fail default||`
-
- # melee sounds
-* `melee_swing default|small_bash|small_cutting|small_stabbing|big_bash|big_cutting|big_stabbing`
-* `melee_hit_flesh default|small_bash|small_cutting|small_stabbing|big_bash|big_cutting|big_stabbing|`
-* `melee_hit_metal default|small_bash|small_cutting|small_stabbing|big_bash|big_cutting|big_stabbing!`
-* `melee_hit ` # note: use weapon id "null" for unarmed attacks
-
- # firearm/ranged weapon sounds
-* `fire_gun |brass_eject|empty`
-* `fire_gun_distant `
-* `reload `
-* `bullet_hit hit_flesh|hit_wall|hit_metal|hit_glass|hit_water`
-
- # environmental sfx, here divided by sections for clarity
-* `environment thunder_near|thunder_far`
-* `environment daytime|nighttime`
-* `environment indoors|indoors_rain|underground`
-* `environment ` # examples: `WEATHER_DRIZZLE|WEATHER_RAINY|WEATHER_THUNDER|WEATHER_FLURRIES|WEATHER_SNOW|WEATHER_SNOWSTORM`
-* `environment alarm|church_bells|police_siren`
-* `environment deafness_shock|deafness_tone_start|deafness_tone_light|deafness_tone_medium|deafness_tone_heavy`
-
- # misc environmental sounds
-* `footstep default|light|clumsy|bionic`
-* `explosion default|small|huge`
-
- # ambient danger theme for seeing large numbers of zombies
-* `danger_low`
-* `danger_medium`
-* `danger_high`
-* `danger_extreme`
-
- # chainsaw pack
-* `chainsaw_cord chainsaw_on`
-* `chainsaw_start chainsaw_on`
-* `chainsaw_start chainsaw_on`
-* `chainsaw_stop chainsaw_on`
-* `chainsaw_idle chainsaw_on`
-* `melee_swing_start chainsaw_on`
-* `melee_swing_end chainsaw_on`
-* `melee_swing chainsaw_on`
-* `melee_hit_flesh chainsaw_on`
-* `melee_hit_metal chainsaw_on`
-* `weapon_theme chainsaw`
-
- # monster death and bite attacks
-* `mon_death zombie_death|zombie_gibbed`
-* `mon_bite bite_miss|bite_hit`
-
-* `melee_attack monster_melee_hit`
-
-* `player_laugh laugh_f|laugh_m`
-
- # player movement sfx
- important: `plmove ` has priority over default `plmove|walk_` (excluding `|barefoot`)
- example: if `plmove|t_grass_long` is defined it will be played before default `plmove|walk_grass` default for all grassy terrains
-
-* `plmove |`
-* `plmove walk_grass|walk_dirt|walk_metal|walk_water|walk_tarmac|walk_barefoot|clear_obstacle`
-
- # fatigue
-* `plmove fatigue_m_low|fatigue_m_med|fatigue_m_high|fatigue_f_low|fatigue_f_med|fatigue_f_high`
-
- # player hurt sounds
-* `deal_damage hurt_f|hurt_m`
-
- # player death and end-game sounds
-* `clean_up_at_end game_over|death_m|death_f`
-
- # variuos bionic sounds
-* `bionic elec_discharge|elec_crackle_low|elec_crackle_med|elec_crackle_high|elec_blast|elec_blast_muffled|acid_discharge|pixelated`
-* `bionic bio_resonator|bio_hydraulics|`
-
- # various tools/traps being used (including some associated terrain/furniture)
-* `tool alarm_clock|jackhammer|pickaxe|oxytorch|hacksaw|axe|shovel|crowbar|boltcutters|compactor|gaspump|noise_emitter|repair_kit|camera_shutter|handcuffs`
-* `tool geiger_low|geiger_medium|geiger_high`
-* `trap bubble_wrap|bear_trap|snare|teleport|dissector|glass_caltrop|glass`
-
- # various activities
-* `activity burrow`
-
- # musical instruments, `_bad` is used when you fail to play it well
-* `musical_instrument `
-* `musical_instrument_bad `
-
- # various shouts and screams
-* `shout default|scream|scream_tortured|roar|squeak|shriek|wail|howl`
-
- # speach, it is currently linked with either item or monster id, or is special `NPC` or `NPC_loud`
- # TODO: full vocalization of speech.json
-* `speech ` # examples: talking_doll, creepy_doll, Granade,
-* `speech ` # examples: eyebot, minitank, mi-go, many robots
-* `speech NPC_m|NPC_f|NPC_m_loud|NPC_f_loud` # special for NPCs
-* `speech robot` # special for robotic voice from a machine etc.
-
- # radio chatter
-* `radio static|inaudible_chatter`
-
- # humming sounds of various origin
-* `humming electric|machinery`
-
- # sounds related to (burning) fire
-* `fire ignition`
-
- # vehicle sounds - engine and other parts in action
- # note: defaults are executed when specific option is not defined
-* `engine_start ` # note: specific engine start (id of any engine/motor/steam_engine/paddle/oar/sail/etc. )
-* `engine_start combustion|electric|muscle|wind` # default engine starts groups
-* `engine_stop ` # note: specific engine stop (id of any engine/motor/steam_engine/paddle/oar/sail/etc. )
-* `engine_stop combustion|electric|muscle|wind` # default engine stop groups
-
- # note: internal engine sound is dynamically pitch shifted depending on vehicle speed
- # it is an ambient looped sound with dedicated channel
-* `engine_working_internal ` # note: sound of engine working heard inside vehicle
-* `engine_working_internal combustion|electric|muscle|wind` # default engine working (inside) groups
-
- # note: external engine sound volume and pan is dynamically shifted depending on distance and angle to vehicle
- # volume heard at given distance is linked to engine's `noise_factor` and stress to the engine (see `vehicle::noise_and_smoke()` )
- # it is an ambient looped sound with dedicated channel
- # this is a single-channel solution (TODO: multi-channel for every heard vehicle); it picks loudest heard vehicle
- # there is no pitch shift here (may be introduced when need for it emerges)
-* `engine_working_external ` # note: sound of engine working heard outside vehicle
-* `engine_working_external combustion|electric|muscle|wind` # default engine working (outside) groups
-
- # note: gear_up/gear_down is done automatically by pitch manipulation
- # gear shift is dependant on max safe speed, and works in assumption, that there are
- # 6 forward gears, gear 0 = neutral, and gear -1 = reverse
-* `vehicle gear_shift`
-
-
-* `vehicle engine_backfire|engine_bangs_start|fault_immobiliser_beep|engine_single_click_fail|engine_multi_click_fail|engine_stutter_fail|engine_clanking_fail`
-* `vehicle horn_loud|horn_medium|horn_low|rear_beeper|chimes|car_alarm`
-* `vehicle reaper|scoop|scoop_thump`
-
-* `vehicle_open ` # note: id of: doors, trunks, hatches, etc.
-* `vehicle_close `
-
- # miscellaneous sounds
-* `misc flashbang|flash|shockwave|earthquake|stairs_movement|stones_grinding|bomb_ticking|lit_fuse|cow_bell|bell|timber`
-* `misc rc_car_hits_obstacle|rc_car_drives`
-* `misc default|whistle|airhorn|horn_bicycle|servomotor`
-* `misc beep|ding|`
-* `misc rattling|spitting|coughing|heartbeat|puff|inhale|exhale|insect_wings|snake_hiss` # mostly organic noises
+
+- `open_door default||`
+- `close_door default||`
+
+ # smashing attempts and results, few special ones and furniture/terrain specific
+- `bash default`
+- `smash wall|door|door_boarded|glass|swing|web|paper_torn|metal`
+- `smash_success hit_vehicle|smash_glass_contents|smash_cloth||`
+- `smash_fail default||`
+
+ # melee sounds
+- `melee_swing default|small_bash|small_cutting|small_stabbing|big_bash|big_cutting|big_stabbing`
+- `melee_hit_flesh default|small_bash|small_cutting|small_stabbing|big_bash|big_cutting|big_stabbing|`
+- `melee_hit_metal default|small_bash|small_cutting|small_stabbing|big_bash|big_cutting|big_stabbing!`
+- `melee_hit ` # note: use weapon id "null" for unarmed attacks
+
+ # firearm/ranged weapon sounds
+- `fire_gun |brass_eject|empty`
+- `fire_gun_distant `
+- `reload `
+- `bullet_hit hit_flesh|hit_wall|hit_metal|hit_glass|hit_water`
+
+ # environmental sfx, here divided by sections for clarity
+- `environment thunder_near|thunder_far`
+- `environment daytime|nighttime`
+- `environment indoors|indoors_rain|underground`
+- `environment ` # examples:
+ `WEATHER_DRIZZLE|WEATHER_RAINY|WEATHER_THUNDER|WEATHER_FLURRIES|WEATHER_SNOW|WEATHER_SNOWSTORM`
+- `environment alarm|church_bells|police_siren`
+- `environment deafness_shock|deafness_tone_start|deafness_tone_light|deafness_tone_medium|deafness_tone_heavy`
+
+ # misc environmental sounds
+- `footstep default|light|clumsy|bionic`
+- `explosion default|small|huge`
+
+ # ambient danger theme for seeing large numbers of zombies
+- `danger_low`
+- `danger_medium`
+- `danger_high`
+- `danger_extreme`
+
+ # chainsaw pack
+- `chainsaw_cord chainsaw_on`
+- `chainsaw_start chainsaw_on`
+- `chainsaw_start chainsaw_on`
+- `chainsaw_stop chainsaw_on`
+- `chainsaw_idle chainsaw_on`
+- `melee_swing_start chainsaw_on`
+- `melee_swing_end chainsaw_on`
+- `melee_swing chainsaw_on`
+- `melee_hit_flesh chainsaw_on`
+- `melee_hit_metal chainsaw_on`
+- `weapon_theme chainsaw`
+
+ # monster death and bite attacks
+- `mon_death zombie_death|zombie_gibbed`
+- `mon_bite bite_miss|bite_hit`
+
+- `melee_attack monster_melee_hit`
+
+- `player_laugh laugh_f|laugh_m`
+
+ # player movement sfx
+ important: `plmove ` has priority over default `plmove|walk_` (excluding
+ `|barefoot`) example: if `plmove|t_grass_long` is defined it will be played before default
+ `plmove|walk_grass` default for all grassy terrains
+
+- `plmove |`
+- `plmove walk_grass|walk_dirt|walk_metal|walk_water|walk_tarmac|walk_barefoot|clear_obstacle`
+
+ # fatigue
+- `plmove fatigue_m_low|fatigue_m_med|fatigue_m_high|fatigue_f_low|fatigue_f_med|fatigue_f_high`
+
+ # player hurt sounds
+- `deal_damage hurt_f|hurt_m`
+
+ # player death and end-game sounds
+- `clean_up_at_end game_over|death_m|death_f`
+
+ # variuos bionic sounds
+- `bionic elec_discharge|elec_crackle_low|elec_crackle_med|elec_crackle_high|elec_blast|elec_blast_muffled|acid_discharge|pixelated`
+- `bionic bio_resonator|bio_hydraulics|`
+
+ # various tools/traps being used (including some associated terrain/furniture)
+- `tool alarm_clock|jackhammer|pickaxe|oxytorch|hacksaw|axe|shovel|crowbar|boltcutters|compactor|gaspump|noise_emitter|repair_kit|camera_shutter|handcuffs`
+- `tool geiger_low|geiger_medium|geiger_high`
+- `trap bubble_wrap|bear_trap|snare|teleport|dissector|glass_caltrop|glass`
+
+ # various activities
+- `activity burrow`
+
+ # musical instruments, `_bad` is used when you fail to play it well
+- `musical_instrument `
+- `musical_instrument_bad `
+
+ # various shouts and screams
+- `shout default|scream|scream_tortured|roar|squeak|shriek|wail|howl`
+
+ # speach, it is currently linked with either item or monster id, or is special `NPC` or `NPC_loud`
+ # TODO: full vocalization of speech.json
+- `speech ` # examples: talking_doll, creepy_doll, Granade,
+- `speech ` # examples: eyebot, minitank, mi-go, many robots
+- `speech NPC_m|NPC_f|NPC_m_loud|NPC_f_loud` # special for NPCs
+- `speech robot` # special for robotic voice from a machine etc.
+
+ # radio chatter
+- `radio static|inaudible_chatter`
+
+ # humming sounds of various origin
+- `humming electric|machinery`
+
+ # sounds related to (burning) fire
+- `fire ignition`
+
+ # vehicle sounds - engine and other parts in action
+ # note: defaults are executed when specific option is not defined
+- `engine_start ` # note: specific engine start (id of any
+ engine/motor/steam_engine/paddle/oar/sail/etc. )
+- `engine_start combustion|electric|muscle|wind` # default engine starts groups
+- `engine_stop ` # note: specific engine stop (id of any
+ engine/motor/steam_engine/paddle/oar/sail/etc. )
+- `engine_stop combustion|electric|muscle|wind` # default engine stop groups
+
+ # note: internal engine sound is dynamically pitch shifted depending on vehicle speed
+ # it is an ambient looped sound with dedicated channel
+- `engine_working_internal ` # note: sound of engine working heard inside vehicle
+- `engine_working_internal combustion|electric|muscle|wind` # default engine working (inside) groups
+
+ # note: external engine sound volume and pan is dynamically shifted depending on distance and angle to vehicle
+ # volume heard at given distance is linked to engine's `noise_factor` and stress to the engine (see `vehicle::noise_and_smoke()` )
+ # it is an ambient looped sound with dedicated channel
+ # this is a single-channel solution (TODO: multi-channel for every heard vehicle); it picks loudest heard vehicle
+ # there is no pitch shift here (may be introduced when need for it emerges)
+- `engine_working_external ` # note: sound of engine working heard outside vehicle
+- `engine_working_external combustion|electric|muscle|wind` # default engine working (outside)
+ groups
+
+ # note: gear_up/gear_down is done automatically by pitch manipulation
+ # gear shift is dependant on max safe speed, and works in assumption, that there are
+ # 6 forward gears, gear 0 = neutral, and gear -1 = reverse
+- `vehicle gear_shift`
+
+- `vehicle engine_backfire|engine_bangs_start|fault_immobiliser_beep|engine_single_click_fail|engine_multi_click_fail|engine_stutter_fail|engine_clanking_fail`
+- `vehicle horn_loud|horn_medium|horn_low|rear_beeper|chimes|car_alarm`
+- `vehicle reaper|scoop|scoop_thump`
+
+- `vehicle_open ` # note: id of: doors, trunks, hatches, etc.
+- `vehicle_close `
+
+ # miscellaneous sounds
+- `misc flashbang|flash|shockwave|earthquake|stairs_movement|stones_grinding|bomb_ticking|lit_fuse|cow_bell|bell|timber`
+- `misc rc_car_hits_obstacle|rc_car_drives`
+- `misc default|whistle|airhorn|horn_bicycle|servomotor`
+- `misc beep|ding|`
+- `misc rattling|spitting|coughing|heartbeat|puff|inhale|exhale|insect_wings|snake_hiss` # mostly
+ organic noises
diff --git a/doc/TER_FURN_TRANSFORM.md b/doc/TER_FURN_TRANSFORM.md
index 043a68df950d..5cfd6f9e9688 100644
--- a/doc/TER_FURN_TRANSFORM.md
+++ b/doc/TER_FURN_TRANSFORM.md
@@ -1,6 +1,7 @@
# ter_furn_transform
-A ter_furn_transform is a type of json object that allows you to specify a transformation of a tile from one terrain to another terrain, and from one furniture to another furniture.
+A ter_furn_transform is a type of json object that allows you to specify a transformation of a tile
+from one terrain to another terrain, and from one furniture to another furniture.
```json
[
@@ -10,7 +11,7 @@ A ter_furn_transform is a type of json object that allows you to specify a trans
"terrain": [
{
"result": "t_dirt",
- "valid_terrain": [ "t_sand" ],
+ "valid_terrain": ["t_sand"],
"message": "sandy!",
"message_good": true
}
@@ -19,8 +20,9 @@ A ter_furn_transform is a type of json object that allows you to specify a trans
]
```
-The example above turns "sand" into "dirt". It does so by comparing the direct terrain ids. In addition, we can add a fail message to the transform.
-If, however, we wanted to turn sand into "dirt or grass" we can do:
+The example above turns "sand" into "dirt". It does so by comparing the direct terrain ids. In
+addition, we can add a fail message to the transform. If, however, we wanted to turn sand into "dirt
+or grass" we can do:
```json
"terrain": [
@@ -33,8 +35,8 @@ If, however, we wanted to turn sand into "dirt or grass" we can do:
]
```
-message_good is optional and defaults to true.
-This example chooses either dirt or grass at a 1:1 ratio. But, if you want a 4:1 ratio:
+message_good is optional and defaults to true. This example chooses either dirt or grass at a 1:1
+ratio. But, if you want a 4:1 ratio:
```json
"terrain": [
@@ -46,7 +48,8 @@ This example chooses either dirt or grass at a 1:1 ratio. But, if you want a 4:1
]
```
-As you can see, you can mix and match arrays with weights with single strings. Each single string has a weight of 1.
+As you can see, you can mix and match arrays with weights with single strings. Each single string
+has a weight of 1.
All of the above applies to furniture as well.
@@ -72,4 +75,5 @@ You can also use flags instead of specific IDs for both furniture and terrain.
]
```
-A ter_furn_transform can have both terrain and furniture fields. It treats them separately, so no "if dirt, add chair."
+A ter_furn_transform can have both terrain and furniture fields. It treats them separately, so no
+"if dirt, add chair."
diff --git a/doc/TESTING.md b/doc/TESTING.md
index 7fc661d7636a..9cbdeaf8b93f 100644
--- a/doc/TESTING.md
+++ b/doc/TESTING.md
@@ -1,47 +1,42 @@
# Testing Cataclysm
-When you `make` Cataclysm from source, an executable `tests/cata_test` is built
-from test cases found in the `tests/` directory. These tests are written in the
+When you `make` Cataclysm from source, an executable `tests/cata_test` is built from test cases
+found in the `tests/` directory. These tests are written in the
[Catch2 framework](https://github.com/catchorg/Catch2).
-Run `tests/cata_test --help` to see the available command-line options, and/or
-consult the [Catch2 tutorial](https://github.com/catchorg/Catch2/blob/master/docs/tutorial.md)
-for a more thorough introduction.
-
+Run `tests/cata_test --help` to see the available command-line options, and/or consult the
+[Catch2 tutorial](https://github.com/catchorg/Catch2/blob/master/docs/tutorial.md) for a more
+thorough introduction.
## Guidelines
-When creating tests, ensure that all objects used (directly or indirectly) are
-fully reset before testing. Several tests have been rendered flaky by
-properties of randomly generated objects or interactions between tests via
-global objects (often the player object). As a general guideline, test cases
-should be standalone (one test should not rely on the output of another).
-
-When generating objects with json definitions, use REQUIRE statements to assert
-the properties of the objects that the test needs. This protects the test from
-shifting json definitions by making it apparent what about the object changed
-to cause the test to break.
+When creating tests, ensure that all objects used (directly or indirectly) are fully reset before
+testing. Several tests have been rendered flaky by properties of randomly generated objects or
+interactions between tests via global objects (often the player object). As a general guideline,
+test cases should be standalone (one test should not rely on the output of another).
+When generating objects with json definitions, use REQUIRE statements to assert the properties of
+the objects that the test needs. This protects the test from shifting json definitions by making it
+apparent what about the object changed to cause the test to break.
## Writing test cases
-You can choose several ways to organize and express your tests, but the basic
-unit is a `TEST_CASE`. Each test `.cpp` file should define at least one test
-case, with a name, and optional (but strongly encouraged) list of tags:
+You can choose several ways to organize and express your tests, but the basic unit is a `TEST_CASE`.
+Each test `.cpp` file should define at least one test case, with a name, and optional (but strongly
+encouraged) list of tags:
```cpp
- TEST_CASE( "sweet junk food", "[food][junk][sweet]" )
- {
- // ...
- }
+TEST_CASE( "sweet junk food", "[food][junk][sweet]" )
+{
+ // ...
+}
```
-Within the `TEST_CASE`, the Catch2 framework allows a number of different
-macros for logically grouping related parts of the test together. One approach
-that encourages a high level of readability is the
-[BDD](https://en.wikipedia.org/wiki/Behavior-driven_development)
-(behavior-driven-development) style using `GIVEN`, `WHEN`, and `THEN` sections.
-Here's an outline of what a test might look like using those:
+Within the `TEST_CASE`, the Catch2 framework allows a number of different macros for logically
+grouping related parts of the test together. One approach that encourages a high level of
+readability is the [BDD](https://en.wikipedia.org/wiki/Behavior-driven_development)
+(behavior-driven-development) style using `GIVEN`, `WHEN`, and `THEN` sections. Here's an outline of
+what a test might look like using those:
```cpp
TEST_CASE( "sweet junk food", "[food][junk][sweet]" )
@@ -57,11 +52,10 @@ Here's an outline of what a test might look like using those:
}
```
-Thinking in these terms may help you understand the logical progression from
-setting up the test and initializing the test data (usually expressed by the
-`GIVEN` part), performing some operation that generates a result you want to
-test (often contained in the `WHEN` part), and verifying this result meets your
-expectations (the `THEN` part, naturally).
+Thinking in these terms may help you understand the logical progression from setting up the test and
+initializing the test data (usually expressed by the `GIVEN` part), performing some operation that
+generates a result you want to test (often contained in the `WHEN` part), and verifying this result
+meets your expectations (the `THEN` part, naturally).
Filling in the above with actual test code might look like this:
@@ -86,76 +80,70 @@ Filling in the above with actual test code might look like this:
}
```
-Let's look at each part in turn to see what's going on. First, we declare an
-`avatar`, representing the character or player. This test is going to check the
-player's morale, so we clear it to ensure a clean slate:
+Let's look at each part in turn to see what's going on. First, we declare an `avatar`, representing
+the character or player. This test is going to check the player's morale, so we clear it to ensure a
+clean slate:
```cpp
- avatar dummy;
- dummy.clear_morale();
+avatar dummy;
+dummy.clear_morale();
```
-Inside the `GIVEN`, we want some code that implements what the `GIVEN` is
-saying - that the character has a sweet tooth. In the game's code, this is
-represented with the `PROJUNK` trait, so we can set that using `toggle_trait`:
+Inside the `GIVEN`, we want some code that implements what the `GIVEN` is saying - that the
+character has a sweet tooth. In the game's code, this is represented with the `PROJUNK` trait, so we
+can set that using `toggle_trait`:
```cpp
- GIVEN( "character has a sweet tooth" ) {
- dummy.toggle_trait( trait_PROJUNK );
+GIVEN( "character has a sweet tooth" ) {
+ dummy.toggle_trait( trait_PROJUNK );
```
-Now, notice we are nested inside the `GIVEN` - for the rest of the scope of
-that `GIVEN`, the `dummy` will have this trait. For this simple test it will
-only affect a couple more lines, but when your tests become larger and more
-complex (which they will), you will need to be aware of these nested scopes and
-how you can use them to avoid cross-pollution between your tests.
+Now, notice we are nested inside the `GIVEN` - for the rest of the scope of that `GIVEN`, the
+`dummy` will have this trait. For this simple test it will only affect a couple more lines, but when
+your tests become larger and more complex (which they will), you will need to be aware of these
+nested scopes and how you can use them to avoid cross-pollution between your tests.
-Anyway, now that our `dummy` has a sweet tooth, we want them to eat something
-sweet, so we can spawn the `neccowafers` item and tell them to eat some:
+Anyway, now that our `dummy` has a sweet tooth, we want them to eat something sweet, so we can spawn
+the `neccowafers` item and tell them to eat some:
```cpp
- WHEN( "they eat some junk food" ) {
- dummy.eat( item( "neccowafers" ) );
+WHEN( "they eat some junk food" ) {
+ dummy.eat( item( "neccowafers" ) );
```
-The function(s) you invoke at this point are often the focus of your testing;
-the goal is to exercise some pathway through those function(s) in such a way
-that your code will be reached, and thus covered by the test. The `eat`
-function is used as an example here, but that is quite a high-level, complex
-function itself, with many behaviors and sub-behaviors. Since this test case is
-only interested in the morale effect, a better test would invoke a lower-level
-function that `eat` invokes, such as `modify_morale`.
+The function(s) you invoke at this point are often the focus of your testing; the goal is to
+exercise some pathway through those function(s) in such a way that your code will be reached, and
+thus covered by the test. The `eat` function is used as an example here, but that is quite a
+high-level, complex function itself, with many behaviors and sub-behaviors. Since this test case is
+only interested in the morale effect, a better test would invoke a lower-level function that `eat`
+invokes, such as `modify_morale`.
-Our `dummy` has eaten the `neccowafers`, but did it do anything? Because they
-have a sweet tooth, they should get a specific morale bonus known as
-`MORALE_SWEETTOOTH`, and it should be at least `5` in magnitude:
+Our `dummy` has eaten the `neccowafers`, but did it do anything? Because they have a sweet tooth,
+they should get a specific morale bonus known as `MORALE_SWEETTOOTH`, and it should be at least `5`
+in magnitude:
```cpp
- THEN( "they get a morale bonus from its sweetness" ) {
- CHECK( dummy.has_morale( MORALE_SWEETTOOTH ) >= 5 );
- }
+THEN( "they get a morale bonus from its sweetness" ) {
+ CHECK( dummy.has_morale( MORALE_SWEETTOOTH ) >= 5 );
+}
```
-This `CHECK` macro takes a boolean expression, failing the test if the
-expression is false. Likewise, you can use `CHECK_FALSE`, which will fail if
-the expression is true.
-
+This `CHECK` macro takes a boolean expression, failing the test if the expression is false.
+Likewise, you can use `CHECK_FALSE`, which will fail if the expression is true.
## Requiring or Checking
-While the `CHECK` and `CHECK_FALSE` macros make assertions about the truth or
-falsity of expressions, they still allow the test to continue, even when they
-fail. This lets you do several `CHECK`s, and be informed if one *or more* of
-them do not meet your expectations.
+While the `CHECK` and `CHECK_FALSE` macros make assertions about the truth or falsity of
+expressions, they still allow the test to continue, even when they fail. This lets you do several
+`CHECK`s, and be informed if one _or more_ of them do not meet your expectations.
-Another kind of assertion is the `REQUIRE` (and its counterpart
-`REQUIRE_FALSE`). Unlike the `CHECK` assertions, `REQUIRE` will not continue if
-it fails - this assertion is considered essential for the test to continue.
+Another kind of assertion is the `REQUIRE` (and its counterpart `REQUIRE_FALSE`). Unlike the `CHECK`
+assertions, `REQUIRE` will not continue if it fails - this assertion is considered essential for the
+test to continue.
-A `REQUIRE` is useful when you wish to double-check your assumptions after
-making some change to the system state. For example, here are a couple of
-`REQUIRE`s added to the sweet-tooth test, to ensure our `dummy` really has the
-desired trait, and that the `neccowafers` really are junk food:
+A `REQUIRE` is useful when you wish to double-check your assumptions after making some change to the
+system state. For example, here are a couple of `REQUIRE`s added to the sweet-tooth test, to ensure
+our `dummy` really has the desired trait, and that the `neccowafers` really are junk food:
```cpp
GIVEN( "character has a sweet tooth" ) {
@@ -175,11 +163,10 @@ desired trait, and that the `neccowafers` really are junk food:
}
```
-We use `REQUIRE` here, because there is no reason to continue the test if these
-fail. If our assumptions are wrong, nothing that follows is valid. Clearly, if
-`toggle_trait` failed to give the character the `PROJUNK` trait, or if the
-`neccowafers` turn out not to be made of sugar after all, then our test of the
-morale bonus is meaningless.
+We use `REQUIRE` here, because there is no reason to continue the test if these fail. If our
+assumptions are wrong, nothing that follows is valid. Clearly, if `toggle_trait` failed to give the
+character the `PROJUNK` trait, or if the `neccowafers` turn out not to be made of sugar after all,
+then our test of the morale bonus is meaningless.
-You can think of `REQUIRE` as being a prerequisite for the test, while `CHECK`
-is looking at the results of the test.
+You can think of `REQUIRE` as being a prerequisite for the test, while `CHECK` is looking at the
+results of the test.
diff --git a/doc/TILESET.md b/doc/TILESET.md
index 51dc6136ed09..0d04cddda02a 100644
--- a/doc/TILESET.md
+++ b/doc/TILESET.md
@@ -1,39 +1,78 @@
# TILESETS
-A tileset provides graphic images for the game. Each tileset has one or more tilesheets of image sprites and a `tile_config.json` file that describes how to map the contents of the sprite sheets to various entities in the game. It also has a `tileset.txt` file that provides metadata.
+
+A tileset provides graphic images for the game. Each tileset has one or more tilesheets of image
+sprites and a `tile_config.json` file that describes how to map the contents of the sprite sheets to
+various entities in the game. It also has a `tileset.txt` file that provides metadata.
## Compositing Tilesets
-Prior October 2019, tilesets had to be submitted to the repo with each tilesheet fully composited and the sprite indices in `tile_config.json` calculated by hand. After October 2019, tilesets can be submitted to repos as directories of individual sprite files and tile entry JSON files that used sprite file names, and a Python script that runs at compile time would merge the sprite images into tilesheets, convert the files names into sprite indices for the tile entries, and merge the tile entries into a `tile_config.json`.
-For the rest of this document, tilesets that are submitted as fully composited tilesheets are called legacy tilesets, and tilesets that submitted as individual sprite image files are compositing tilesets.
+Prior October 2019, tilesets had to be submitted to the repo with each tilesheet fully composited
+and the sprite indices in `tile_config.json` calculated by hand. After October 2019, tilesets can be
+submitted to repos as directories of individual sprite files and tile entry JSON files that used
+sprite file names, and a Python script that runs at compile time would merge the sprite images into
+tilesheets, convert the files names into sprite indices for the tile entries, and merge the tile
+entries into a `tile_config.json`.
+
+For the rest of this document, tilesets that are submitted as fully composited tilesheets are called
+legacy tilesets, and tilesets that submitted as individual sprite image files are compositing
+tilesets.
### tools/gfx_tools/decompose.py
-This is a Python script that will convert a legacy tileset into a compositing tileset. It reads the `tile_config.json` and assigns semi-arbitrary file names to each sprite index. Then it changes all the sprite indexes references to the file names. Then it breaks up `tile_config.json` into many small tile_entry JSON files with arbitrary file names, and pulls out each sprite and writes it to aseparate file.
+
+This is a Python script that will convert a legacy tileset into a compositing tileset. It reads the
+`tile_config.json` and assigns semi-arbitrary file names to each sprite index. Then it changes all
+the sprite indexes references to the file names. Then it breaks up `tile_config.json` into many
+small tile_entry JSON files with arbitrary file names, and pulls out each sprite and writes it to
+aseparate file.
It requires pyvips to do the image processing.
-It takes a single mandatory argument, which is the path to the tileset directory. For example:
-`python tools/gfx_tools/decompose.py gfx/ChestHole16Tileset` will convert the legacy ChestHole16 tileset to a compositing tileset.
+It takes a single mandatory argument, which is the path to the tileset directory. For example:
+`python tools/gfx_tools/decompose.py gfx/ChestHole16Tileset` will convert the legacy ChestHole16
+tileset to a compositing tileset.
-decompose.py creates a sufficient directory hierarchy and file names for a tileset to be compositing, but it is machine generated and badly organized. New compositing tilesets should use more sensible file names and a better organization.
+decompose.py creates a sufficient directory hierarchy and file names for a tileset to be
+compositing, but it is machine generated and badly organized. New compositing tilesets should use
+more sensible file names and a better organization.
-It shouldn't be necessary to run decompose.py very often. Legacy tilesets should only need to be converted to composite tilesets one time.
+It shouldn't be necessary to run decompose.py very often. Legacy tilesets should only need to be
+converted to composite tilesets one time.
### tools/gfx_tools/compose.py
-This is a Python script that creates the tilesheets for a compositing tileset. It reads all of the directories in a tileset's directory with names that start with `pngs_` for sprite files and `tile_entry` JSON files, creates mappings of sprite file names to indices, merges the sprite files into tilesheets, changes all of the sprite file name references in the `tile_entries` to indices, and merges the `tile_entries` into a `tile_config.json`.
+
+This is a Python script that creates the tilesheets for a compositing tileset. It reads all of the
+directories in a tileset's directory with names that start with `pngs_` for sprite files and
+`tile_entry` JSON files, creates mappings of sprite file names to indices, merges the sprite files
+into tilesheets, changes all of the sprite file name references in the `tile_entries` to indices,
+and merges the `tile_entries` into a `tile_config.json`.
Like decompose.py, it requires pyvips to the image processing.
The original sprite files and `tile_entry` JSON files are preserved.
### directory structure
-Each compositing tileset has one or more directories in it with a name that starts with `pngs_`, such as `pngs_tree_32x40` or `pngs_overlay`. These are the image directories. All of the sprites in an image directory must have the same height and width and will be merged into a single tilesheet.
-It is recommended that tileset developers include the sprite dimensions in the image directory name, but this is not required. `pngs_overlay_24x24` is preferred over `pngs_overlay` but both are allowed. As each image directory creates its own tilesheet, and tilesheets should be as large as possible for performance reasons, tileset developers are strongly encouraged to minimize the number of image directories.
+Each compositing tileset has one or more directories in it with a name that starts with `pngs_`,
+such as `pngs_tree_32x40` or `pngs_overlay`. These are the image directories. All of the sprites in
+an image directory must have the same height and width and will be merged into a single tilesheet.
+
+It is recommended that tileset developers include the sprite dimensions in the image directory name,
+but this is not required. `pngs_overlay_24x24` is preferred over `pngs_overlay` but both are
+allowed. As each image directory creates its own tilesheet, and tilesheets should be as large as
+possible for performance reasons, tileset developers are strongly encouraged to minimize the number
+of image directories.
-Each image directory contains a hierarchy of subdirectories, `tile_entry` JSON files, and sprite files. There is no restriction on the arrangement or names of these files, except for `tile_entry` JSON files for expansion tilesheets must be at the top level of the image directory. Subdirectories are not required but are recommended to keep things manageable.
+Each image directory contains a hierarchy of subdirectories, `tile_entry` JSON files, and sprite
+files. There is no restriction on the arrangement or names of these files, except for `tile_entry`
+JSON files for expansion tilesheets must be at the top level of the image directory. Subdirectories
+are not required but are recommended to keep things manageable.
#### `tile_entry` JSON
-Each `tile_entry` JSON is a dictionary that describes how to map one or more game entities to one or more sprites. The simplest version has a single game entity, a single foreground sprite, an *optional* background sprite, and a rotation value. For instance:
+
+Each `tile_entry` JSON is a dictionary that describes how to map one or more game entities to one or
+more sprites. The simplest version has a single game entity, a single foreground sprite, an
+_optional_ background sprite, and a rotation value. For instance:
+
```C++
{ // this is an object and doesn't require a list
"id": "mon_cat", // the game entity represented by this sprite
@@ -43,33 +82,56 @@ Each `tile_entry` JSON is a dictionary that describes how to map one or more gam
}
```
-The values in `"id"`, `"fg"`, and `"bg"` can be repeated within an image directory or in different image directories. `"fg"` and `"bg"` sprite images can be referenced across image directories, but the sprites must be stored in an image directory with other sprites of the same height and width.
+The values in `"id"`, `"fg"`, and `"bg"` can be repeated within an image directory or in different
+image directories. `"fg"` and `"bg"` sprite images can be referenced across image directories, but
+the sprites must be stored in an image directory with other sprites of the same height and width.
+
+`"id"` can also be a list of multiple game entities sharing the same sprite, like
+`"id": ["vp_door"], ["vp_hddoor"]`. `"id"` can be any vehicle part, terrain, furniture, item, or
+monster in the game. The special ids `"player_female", "player_male", "npc_female", "npc_male"` are
+used to identify the sprites for the player avatar and NPCs. The special id `"unknown"` provides a
+sprite that is displayed when an entity has no other sprite.
-`"id"` can also be a list of multiple game entities sharing the same sprite, like `"id": ["vp_door"], ["vp_hddoor"]`. `"id"` can be any vehicle part, terrain, furniture, item, or monster in the game. The special ids `"player_female", "player_male", "npc_female", "npc_male"` are used to identify the sprites for the player avatar and NPCs. The special id `"unknown"` provides a sprite that is displayed when an entity has no other sprite.
+The special suffixes `_season_spring`, `_season_summer`, `_season_autumn`, and `_season_winter` can
+be applied to any entity id to create a seasonal variant for that entity that will be displayed in
+the appropriate season like this `"id": "mon_wolf_season_winter"`.
-The special suffixes `_season_spring`, `_season_summer`, `_season_autumn`, and `_season_winter` can be applied to any entity id to create a seasonal variant for that entity that will be displayed in the appropriate season like this `"id": "mon_wolf_season_winter"`.
+The special prefixes `overlay_mutation_`, `overlay_female_mutation_`, `overlay_male_mutation_` can
+prefix any trait or bionic in the game to specify an overlay image that will be laid over the player
+and NPC sprites to indicate they have that mutation or bionic.
-The special prefixes `overlay_mutation_`, `overlay_female_mutation_`, `overlay_male_mutation_` can prefix any trait or bionic in the game to specify an overlay image that will be laid over the player and NPC sprites to indicate they have that mutation or bionic.
+The special prefixes `overlay_worn_`, `overlay_female_worn_`, `overlay_male_worn_` can prefix any
+item in the game to specify an overlay image that will be laid over the player and NPC sprites to
+indicate they are wearing that item.
-The special prefixes `overlay_worn_`, `overlay_female_worn_`, `overlay_male_worn_` can prefix any item in the game to specify an overlay image that will be laid over the player and NPC sprites to indicate they are wearing that item.
+The special prefixes `overlay_wielded_`, `overlay_female_wielded_`, `overlay_male_wielded_` can
+prefix any item in the game to specify an overlay image that will be laid over the player and NPC
+sprites to indicate they are holding that item.
-The special prefixes `overlay_wielded_`, `overlay_female_wielded_`, `overlay_male_wielded_` can prefix any item in the game to specify an overlay image that will be laid over the player and NPC sprites to indicate they are holding that item.
+`"fg"` and `"bg"` can also be a list of 2 or 4 pre-rotated rotational variants, like
+`"bg": ["t_wall_n", "t_wall_e", "t_wall_s", "t_wall_w"]` or
+`"fg": ["mon_dog_left", "mon_dog_right"]`.
-`"fg"` and `"bg"` can also be a list of 2 or 4 pre-rotated rotational variants, like `"bg": ["t_wall_n", "t_wall_e", "t_wall_s", "t_wall_w"]` or `"fg": ["mon_dog_left", "mon_dog_right"]`.
+`"fg"` and `"bg"` can also be a list of dictionaries of weighted, randomly chosen options, any of
+which can also be a rotated list:
-`"fg"` and `"bg"` can also be a list of dictionaries of weighted, randomly chosen options, any of which can also be a rotated list:
```C++
- "fg": [
- { "weight": 50, "sprite": "t_dirt_brown"}, // appears in 50 of 53 tiles
- { "weight": 1, "sprite": "t_dirt_black_specks"}, // appears 1 in 53 tiles
- { "weight": 1, "sprite": "t_dirt_specks_gray"},
- { "weight": 1, "sprite": "t_patchy_grass"} // file names are arbitrary
- ],
+"fg": [
+ { "weight": 50, "sprite": "t_dirt_brown"}, // appears in 50 of 53 tiles
+ { "weight": 1, "sprite": "t_dirt_black_specks"}, // appears 1 in 53 tiles
+ { "weight": 1, "sprite": "t_dirt_specks_gray"},
+ { "weight": 1, "sprite": "t_patchy_grass"} // file names are arbitrary
+],
```
-`"multitle"` is an *optional* field. If it is present and `true`, there must be an `additional_tiles` list with 1 or more dictionaries for entities and sprites associated with this tile, such as broken versions of an item or wall connections. Each dictionary in the list has an `"id`" field, as above, and a `"fg"` field, which can be a single filename, a list of filenames, or a list of dictionaries as above.
+`"multitle"` is an _optional_ field. If it is present and `true`, there must be an
+`additional_tiles` list with 1 or more dictionaries for entities and sprites associated with this
+tile, such as broken versions of an item or wall connections. Each dictionary in the list has an
+`"id`" field, as above, and a `"fg"` field, which can be a single filename, a list of filenames, or
+a list of dictionaries as above.
Each `tile_entry.json` file can have a single object in it, or a list of 1 or more objects like so:
+
```C++
[
{ "id": "mon_zombie", "fg": "mon_zombie", "bg": "mon_zombie_bg", "rotates": false },
@@ -78,20 +140,33 @@ Each `tile_entry.json` file can have a single object in it, or a list of 1 or mo
]
```
-Having a list of tile entries in a file may be useful for organization, but completely unrelated entries may all exist in the same file without any complications.
+Having a list of tile entries in a file may be useful for organization, but completely unrelated
+entries may all exist in the same file without any complications.
#### expansion `tile_entry` JSON
-Tilesheets can have expansion tilesheets, which are tilesheets from mods. Each expansion tilesheet is a single `"id"` value, `"rotates": false"`, and `"fg": 0`. Expansion `tile_entry` JSON are the only `tile_entry` JSONs that use an integer value for `"fg"` and that value must be 0. Expansion `tile_entry` JSONs must be located at the top layer of each image directory.
+
+Tilesheets can have expansion tilesheets, which are tilesheets from mods. Each expansion tilesheet
+is a single `"id"` value, `"rotates": false"`, and `"fg": 0`. Expansion `tile_entry` JSON are the
+only `tile_entry` JSONs that use an integer value for `"fg"` and that value must be 0. Expansion
+`tile_entry` JSONs must be located at the top layer of each image directory.
#### Sprite Images
-Every sprite inside an image directory must have the same height and width as every other sprite in the image directory.
-Sprites can be organized into subdirectories within the image directory however the tileset developer prefers. Sprite filenames are completely arbitrary and should be chosen using a scheme that makes sense to the tileset developer.
+Every sprite inside an image directory must have the same height and width as every other sprite in
+the image directory.
+
+Sprites can be organized into subdirectories within the image directory however the tileset
+developer prefers. Sprite filenames are completely arbitrary and should be chosen using a scheme
+that makes sense to the tileset developer.
-After loading a tileset, config/debug.log will contain a space separated list of every entity missing a sprite in the tileset. Entities that have sprites because of a `"looks_like"` definition will not show up in the list.
+After loading a tileset, config/debug.log will contain a space separated list of every entity
+missing a sprite in the tileset. Entities that have sprites because of a `"looks_like"` definition
+will not show up in the list.
### `tile_info.json`
-Each compositing tileset *must* have a `tile_info.json`, laid out like so:
+
+Each compositing tileset _must_ have a `tile_info.json`, laid out like so:
+
```
[
{
@@ -118,125 +193,150 @@ Each compositing tileset *must* have a `tile_info.json`, laid out like so:
}
]
```
-The first dictionary is mandatory, and gives the default sprite width and sprite height for all tilesheets in the tileset. Each of the image directories must have a separate dictionary, containing the tilesheet png name as its key. If the tilesheet has the default sprite dimensions and no special offsets, it can have an empty dictionary as the value for the tilesheet name key. Otherwise, it should have a dictionary of the sprite offsets, height, and width.
-
-A special key is `"fallback"` which should be `true` if present. If a tilesheet is designated as fallback, it will be treated as a tilesheet of fallback ASCII characters. `compose.py` will also compose the fallback tilesheet to the end of the tileset, and will add a "fallback.png" to `tile_config.json` if there is no `"fallback"` entry in `tile_info.json`.
-A special is `"filler"` which should be `true` if present. If a tilesheet is designated as filler, entries from its directory will be ignored if an entry from a non-filler directory has already defined the same id. Entries will also be ignored if the id was already defined by in the filler directory. Also, pngs from a filler directory will be ignored if they share a name with a png from a non-filler directory. A filler tilesheet is useful when upgrading the art in a tileset: old, low-quality art can be placed on filler tilesheet and will be automatically replaced as better images are added to the non-filler tilesheets.
+The first dictionary is mandatory, and gives the default sprite width and sprite height for all
+tilesheets in the tileset. Each of the image directories must have a separate dictionary, containing
+the tilesheet png name as its key. If the tilesheet has the default sprite dimensions and no special
+offsets, it can have an empty dictionary as the value for the tilesheet name key. Otherwise, it
+should have a dictionary of the sprite offsets, height, and width.
+
+A special key is `"fallback"` which should be `true` if present. If a tilesheet is designated as
+fallback, it will be treated as a tilesheet of fallback ASCII characters. `compose.py` will also
+compose the fallback tilesheet to the end of the tileset, and will add a "fallback.png" to
+`tile_config.json` if there is no `"fallback"` entry in `tile_info.json`.
+
+A special is `"filler"` which should be `true` if present. If a tilesheet is designated as filler,
+entries from its directory will be ignored if an entry from a non-filler directory has already
+defined the same id. Entries will also be ignored if the id was already defined by in the filler
+directory. Also, pngs from a filler directory will be ignored if they share a name with a png from a
+non-filler directory. A filler tilesheet is useful when upgrading the art in a tileset: old,
+low-quality art can be placed on filler tilesheet and will be automatically replaced as better
+images are added to the non-filler tilesheets.
## Legacy tilesets
+
### tilesheets
-Each tilesheet contains 1 or more sprites with the same width and height. Each tilesheet contains one or more rows of exactly 16 sprites. Sprite index 0 is special and the first sprite of the first tilesheet in a tileset should be blank. Indices run sequentially through each sheet and continue incrementing for each new sheet without reseting, so index 32 is the first sprite in the third row of the first sheet. If the first sheet has 320 sprites in it, index 352 would be the first sprite of the third row of the second sheet.
+
+Each tilesheet contains 1 or more sprites with the same width and height. Each tilesheet contains
+one or more rows of exactly 16 sprites. Sprite index 0 is special and the first sprite of the first
+tilesheet in a tileset should be blank. Indices run sequentially through each sheet and continue
+incrementing for each new sheet without reseting, so index 32 is the first sprite in the third row
+of the first sheet. If the first sheet has 320 sprites in it, index 352 would be the first sprite of
+the third row of the second sheet.
### `tile_config`
-Each legacy tileset has a `tile_config.json` describing how to map the contents of a sprite sheet to various tile identifiers, different orientations, etc. The ordering of the overlays used for displaying mutations can be controlled as well. The ordering can be used to override the default ordering provided in `mutation_ordering.json`. Example:
+
+Each legacy tileset has a `tile_config.json` describing how to map the contents of a sprite sheet to
+various tile identifiers, different orientations, etc. The ordering of the overlays used for
+displaying mutations can be controlled as well. The ordering can be used to override the default
+ordering provided in `mutation_ordering.json`. Example:
```C++
- { // whole file is a single object
- "tile_info": [ // tile_info is mandatory
- {
- "height": 32,
- "width": 32,
- "iso" : true, // Optional. Indicates an isometric tileset. Defaults to false.
- "pixelscale" : 2 // Optional. Sets a multiplier for resizing a tileset. Defaults to 1.
- }
- ],
- "tiles-new": [ // tiles-new is an array of sprite sheets
- { // alternately, just one "tiles" array
- "file": "tiles.png", // file containing sprites in a grid
- "tiles": [ // array with one entry per tile
- {
- "id": "10mm", // id is how the game maps things to sprites
- "fg": 1, // lack of prefix mostly indicates items
- "bg": 632, // fg and bg can be sprite indexes in the image
- "rotates": false
- },
- {
- "id": "t_wall", // "t_" indicates terrain
- "fg": [2918, 2919, 2918, 2919], // 2 or 4 sprite numbers indicates pre-rotated
- "bg": 633,
- "rotates": true,
- "multitile": true,
- "additional_tiles": [ // connected/combined versions of sprite
- { // or variations, see below
- "id": "center",
- "fg": [2919, 2918, 2919, 2918]
- },
- {
- "id": "corner",
- "fg": [2924, 2922, 2922, 2923]
- },
- {
- "id": "end_piece",
- "fg": [2918, 2919, 2918, 2919]
- },
- {
- "id": "t_connection",
- "fg": [2919, 2918, 2919, 2918]
- },
- {
- "id": "unconnected",
- "fg": 2235
- }
- ]
- },
- {
- "id": "vp_atomic_lamp", // "vp_" vehicle part
- "fg": 3019,
- "bg": 632,
- "rotates": false,
- "multitile": true,
- "additional_tiles": [
- {
- "id": "broken", // variant sprite
- "fg": 3021
- }
- ]
- },
- {
- "id": "t_dirt",
- "rotates": false,
- "fg": [
- { "weight":50, "sprite":640}, // weighted random variants
- { "weight":1, "sprite":3620},
- { "weight":1, "sprite":3621},
- { "weight":1, "sprite":3622}
- ]
- },
- {
- "id": [
- "overlay_mutation_GOURMAND", // character overlay for mutation
- "overlay_mutation_male_GOURMAND", // overlay for specified gender
- "overlay_mutation_active_GOURMAND" // overlay for activated mutation
- ],
- "fg": 4040
- }
- ]
- },
- { // second entry in tiles-new
- "file": "moretiles.png", // another sprite sheet
- "tiles": [
- {
- "id": ["xxx","yyy"], // define two ids at once
- "fg": 1,
- "bg": 234
- }
- ]
- }d
- ],
- "overlay_ordering": [
- {
- "id" : "WINGS_BAT", // mutation name, in a string or array of strings
- "order" : 1000 // range from 0 - 9999, 9999 being the topmost layer
- },
- {
- "id" : [ "PLANTSKIN", "BARK" ], // mutation name, in a string or array of strings
- "order" : 3500 // order is applied to all items in the array
- },
- {
- "id" : "bio_armor_torso", // Overlay order of bionics is controlled in the same way
- "order" : 500
- }
- ]
- }
+{ // whole file is a single object
+ "tile_info": [ // tile_info is mandatory
+ {
+ "height": 32,
+ "width": 32,
+ "iso" : true, // Optional. Indicates an isometric tileset. Defaults to false.
+ "pixelscale" : 2 // Optional. Sets a multiplier for resizing a tileset. Defaults to 1.
+ }
+ ],
+ "tiles-new": [ // tiles-new is an array of sprite sheets
+ { // alternately, just one "tiles" array
+ "file": "tiles.png", // file containing sprites in a grid
+ "tiles": [ // array with one entry per tile
+ {
+ "id": "10mm", // id is how the game maps things to sprites
+ "fg": 1, // lack of prefix mostly indicates items
+ "bg": 632, // fg and bg can be sprite indexes in the image
+ "rotates": false
+ },
+ {
+ "id": "t_wall", // "t_" indicates terrain
+ "fg": [2918, 2919, 2918, 2919], // 2 or 4 sprite numbers indicates pre-rotated
+ "bg": 633,
+ "rotates": true,
+ "multitile": true,
+ "additional_tiles": [ // connected/combined versions of sprite
+ { // or variations, see below
+ "id": "center",
+ "fg": [2919, 2918, 2919, 2918]
+ },
+ {
+ "id": "corner",
+ "fg": [2924, 2922, 2922, 2923]
+ },
+ {
+ "id": "end_piece",
+ "fg": [2918, 2919, 2918, 2919]
+ },
+ {
+ "id": "t_connection",
+ "fg": [2919, 2918, 2919, 2918]
+ },
+ {
+ "id": "unconnected",
+ "fg": 2235
+ }
+ ]
+ },
+ {
+ "id": "vp_atomic_lamp", // "vp_" vehicle part
+ "fg": 3019,
+ "bg": 632,
+ "rotates": false,
+ "multitile": true,
+ "additional_tiles": [
+ {
+ "id": "broken", // variant sprite
+ "fg": 3021
+ }
+ ]
+ },
+ {
+ "id": "t_dirt",
+ "rotates": false,
+ "fg": [
+ { "weight":50, "sprite":640}, // weighted random variants
+ { "weight":1, "sprite":3620},
+ { "weight":1, "sprite":3621},
+ { "weight":1, "sprite":3622}
+ ]
+ },
+ {
+ "id": [
+ "overlay_mutation_GOURMAND", // character overlay for mutation
+ "overlay_mutation_male_GOURMAND", // overlay for specified gender
+ "overlay_mutation_active_GOURMAND" // overlay for activated mutation
+ ],
+ "fg": 4040
+ }
+ ]
+ },
+ { // second entry in tiles-new
+ "file": "moretiles.png", // another sprite sheet
+ "tiles": [
+ {
+ "id": ["xxx","yyy"], // define two ids at once
+ "fg": 1,
+ "bg": 234
+ }
+ ]
+ }d
+ ],
+ "overlay_ordering": [
+ {
+ "id" : "WINGS_BAT", // mutation name, in a string or array of strings
+ "order" : 1000 // range from 0 - 9999, 9999 being the topmost layer
+ },
+ {
+ "id" : [ "PLANTSKIN", "BARK" ], // mutation name, in a string or array of strings
+ "order" : 3500 // order is applied to all items in the array
+ },
+ {
+ "id" : "bio_armor_torso", // Overlay order of bionics is controlled in the same way
+ "order" : 500
+ }
+ ]
+}
```
diff --git a/doc/TRANSLATING.md b/doc/TRANSLATING.md
index 845d29f48071..9e704a7047c3 100644
--- a/doc/TRANSLATING.md
+++ b/doc/TRANSLATING.md
@@ -18,77 +18,71 @@
## Translators
-The official location for translating Cataclysm: BN is the
-[Transifex translations project][1].
+The official location for translating Cataclysm: BN is the [Transifex translations project][1].
-If you're looking for a way to translate mods not included
-in game repository, see [TRANSLATING_MODS.md](TRANSLATING_MODS.md).
+If you're looking for a way to translate mods not included in game repository, see
+[TRANSLATING_MODS.md](TRANSLATING_MODS.md).
Some of the currently supported languages are:
-* Arabic
-* Bulgarian
-* Chinese (Simplified)
-* Chinese (Traditional)
-* Dutch
-* Esperanto
-* French
-* German
-* Italian (Italy)
-* Japanese
-* Korean
-* Polish
-* Portuguese (Brazil)
-* Russian
-* Serbian
-* Spanish (Argentina)
-* Spanish (Spain)
-* Turkish
-
-Don't see your language in the list above? You can add it into the project at
-Transifex!
+- Arabic
+- Bulgarian
+- Chinese (Simplified)
+- Chinese (Traditional)
+- Dutch
+- Esperanto
+- French
+- German
+- Italian (Italy)
+- Japanese
+- Korean
+- Polish
+- Portuguese (Brazil)
+- Russian
+- Serbian
+- Spanish (Argentina)
+- Spanish (Spain)
+- Turkish
+
+Don't see your language in the list above? You can add it into the project at Transifex!
### Getting Started
-To begin translating, head over the [translation project][1] and click on the
-"Help Translate Cataclysm: BN" button.
-This should take you to a page where you can either create a free account on
+To begin translating, head over the [translation project][1] and click on the "Help Translate
+Cataclysm: BN" button. This should take you to a page where you can either create a free account on
Transifex, or login using GitHub, Google+ or LinkedIn.
![Start translating](img/translating-start.png)
-After you've created your account, return to the [translation project][1] and
-click on the "Join team" button.
-This will open a window where you can choose the language you are interested on
+After you've created your account, return to the [translation project][1] and click on the "Join
+team" button. This will open a window where you can choose the language you are interested on
translating, so pick one and click the "Join" button.
![Join project](img/translating-join.png)
-After this, the most straightforward thing to do is to reload the page,
-which should redirect you to the translation project's dashboard.
-Here, you can click the "Languages" link on the sidebar to see the list of
-supported languages and the current progress of the translation effort.
+After this, the most straightforward thing to do is to reload the page, which should redirect you to
+the translation project's dashboard. Here, you can click the "Languages" link on the sidebar to see
+the list of supported languages and the current progress of the translation effort.
-Note that you can request for the inclusion of additional languages,
-if the one you are interested in is not available on the list.
+Note that you can request for the inclusion of additional languages, if the one you are interested
+in is not available on the list.
![Language list](img/translating-list.png)
-From this list, you can click on the language of your choice, and then click on
-the "Translate" to get started right away. Otherwise, you can click on any
-other language and click on the "Join team" button, if you are interested in
-translating for that language as well.
+From this list, you can click on the language of your choice, and then click on the "Translate" to
+get started right away. Otherwise, you can click on any other language and click on the "Join team"
+button, if you are interested in translating for that language as well.
-After clicking on the "Translate" button, you will be taken to the web editor.
-To begin, you need to choose a resource to translate. Most of the in-game text
-is contained in the `cataclysm-bright-nights` resource, so click on it to start.
+After clicking on the "Translate" button, you will be taken to the web editor. To begin, you need to
+choose a resource to translate. Most of the in-game text is contained in the
+`cataclysm-bright-nights` resource, so click on it to start.
![Choose a resource](img/translating-resource.png)
-At this point, the editor should show you the list of text available for
-translation, now you only need to click on the string you want to translate and
-type your translation on the translation area on the right side of the screen.
-Click on the "Save" button when you are satisfied with your translation.
+At this point, the editor should show you the list of text available for translation, now you only
+need to click on the string you want to translate and type your translation on the translation area
+on the right side of the screen. Click on the "Save" button when you are satisfied with your
+translation.
![Web editor](img/translating-editor.png)
@@ -96,75 +90,68 @@ See [Transifex's documentation][2] for more information.
### Grammatical gender
-For NPC dialogue (and potentially other strings) some languages may wish to
-have alternate translations depending on the gender of the conversation
-participants. This two pieces of initial configuration.
+For NPC dialogue (and potentially other strings) some languages may wish to have alternate
+translations depending on the gender of the conversation participants. This two pieces of initial
+configuration.
-1. The dialogue must have the relevant genders listed in the json file defining
- it. See [the NPC docs](NPCs.md).
-2. Each language must specify the genders it wishes to use via `genders` list
- of the language's entry in `data/raw/languages.json`. Don't add genders there
- until you're sure you will need them, because it will make more work for
- you. Current choices are: `m` (male), `f` (female), `n` (neuter).
- If you need different genders than the ones currently supported, see
- relevant note in `src/language.h`.
+1. The dialogue must have the relevant genders listed in the json file defining it. See
+ [the NPC docs](NPCs.md).
+2. Each language must specify the genders it wishes to use via `genders` list of the language's
+ entry in `data/raw/languages.json`. Don't add genders there until you're sure you will need them,
+ because it will make more work for you. Current choices are: `m` (male), `f` (female), `n`
+ (neuter). If you need different genders than the ones currently supported, see relevant note in
+ `src/language.h`.
-Having done this, the relevant dialogue lines will appear multiple times for
-translation, with different genders specified in the message context. For
-example, a context of `npc:m` would indicate that the NPC participant in the
-conversation is male.
+Having done this, the relevant dialogue lines will appear multiple times for translation, with
+different genders specified in the message context. For example, a context of `npc:m` would indicate
+that the NPC participant in the conversation is male.
-Because of technical limitations, all supported genders will appear as
-contexts, but you only need to provide translations for the genders listed in
-the grammatical gender list for your language.
+Because of technical limitations, all supported genders will appear as contexts, but you only need
+to provide translations for the genders listed in the grammatical gender list for your language.
-Other parts of the game have various ad hoc solutions to grammatical gender, so
-don't be surprised to see other contexts appearing for other strings.
+Other parts of the game have various ad hoc solutions to grammatical gender, so don't be surprised
+to see other contexts appearing for other strings.
### Tips
-There are issues specific to Cataclysm: BN which translators should be aware of.
-These include the use of terms like `%s` and `%3$d` (leave them as they are),
-and the use of tags like ``, which shouldn't be translated.
+There are issues specific to Cataclysm: BN which translators should be aware of. These include the
+use of terms like `%s` and `%3$d` (leave them as they are), and the use of tags like ``, which
+shouldn't be translated.
-Information about these and any other issues specific to individual languages,
-can be found in Cataclysm: BN's [language notes folder][3].
+Information about these and any other issues specific to individual languages, can be found in
+Cataclysm: BN's [language notes folder][3].
-General notes for all translators are in `README_all_translators.txt`,
-and notes specific to a language may be stored as `.txt`,
-for example `de.txt` for German.
+General notes for all translators are in `README_all_translators.txt`, and notes specific to a
+language may be stored as `.txt`, for example `de.txt` for German.
-Cataclysm: BN has more than 14000 translatable strings, but don't be discouraged.
-The more translators there are, the easier it becomes 😄.
+Cataclysm: BN has more than 14000 translatable strings, but don't be discouraged. The more
+translators there are, the easier it becomes 😄.
## Developers
-Cataclysm: BN uses custom runtime library that works similarly to
-[GNU gettext][4] to display translated texts.
+Cataclysm: BN uses custom runtime library that works similarly to [GNU gettext][4] to display
+translated texts.
Using `gettext` requires two actions:
-* Marking strings that should be translated in the source code.
-* Calling translation functions at run time.
+- Marking strings that should be translated in the source code.
+- Calling translation functions at run time.
-Marking translatable string allows for their automatic extraction.
-This process generates a file that maps the original string (usually in English)
-as it appears in the source code to the translated string.
-These mappings are used at run time by the translation functions.
+Marking translatable string allows for their automatic extraction. This process generates a file
+that maps the original string (usually in English) as it appears in the source code to the
+translated string. These mappings are used at run time by the translation functions.
-Note that only extracted strings can get translated, since the original string
-is acting as the identifier used to request the translation.
-If a translation function can't find the translation, it returns the original
-string.
+Note that only extracted strings can get translated, since the original string is acting as the
+identifier used to request the translation. If a translation function can't find the translation, it
+returns the original string.
### Translation Functions
-In order to mark a string for translation and to obtain its translation at
-runtime, you should use one of the following functions and classes.
+In order to mark a string for translation and to obtain its translation at runtime, you should use
+one of the following functions and classes.
-String *literals* that are used in any of these functions are automatically
-extracted. Non-literal strings are still translated at run time, but they won't
-get extracted.
+String _literals_ that are used in any of these functions are automatically extracted. Non-literal
+strings are still translated at run time, but they won't get extracted.
#### `_()`
@@ -180,21 +167,18 @@ It also works directly:
add_msg( _( "You drop the %s." ), the_item_name );
```
-Strings from the JSON files are extracted by the `lang/extract_json_strings.py`
-script, and can be translated at run time using `_()`. If translation context
-is desired for a JSON string, `class translation` can be used instead, which is
-documented below.
+Strings from the JSON files are extracted by the `lang/extract_json_strings.py` script, and can be
+translated at run time using `_()`. If translation context is desired for a JSON string,
+`class translation` can be used instead, which is documented below.
#### `pgettext()`
-This function is useful when the original string's meaning is ambiguous in
-isolation. For example, the word "blue", which can mean either a color or an
-emotion.
+This function is useful when the original string's meaning is ambiguous in isolation. For example,
+the word "blue", which can mean either a color or an emotion.
-In addition to the translatable string, `pgettext` receives a context which is
-provided to the translators, but is not part of the translated string itself.
-This function's first parameter is the context, the second is the string to be
-translated:
+In addition to the translatable string, `pgettext` receives a context which is provided to the
+translators, but is not part of the translated string itself. This function's first parameter is the
+context, the second is the string to be translated:
```c++
const char *translated = pgettext( "The color", "blue" );
@@ -202,11 +186,10 @@ const char *translated = pgettext( "The color", "blue" );
#### `vgettext()`
-Some languages have complex rules for plural forms. `vgettext` can be used to
-translate these plurals correctly. Its first parameter is the untranslated
-string in singular form, the second parameter is the untranslated string in
-plural form and the third one is used to determine which one of the first two
-should be used at run time:
+Some languages have complex rules for plural forms. `vgettext` can be used to translate these
+plurals correctly. Its first parameter is the untranslated string in singular form, the second
+parameter is the untranslated string in plural form and the third one is used to determine which one
+of the first two should be used at run time:
```c++
const char *translated = vgettext( "%d zombie", "%d zombies", num_of_zombies );
@@ -222,10 +205,9 @@ const char *translated = vpgettext( "water source, not time of year", "%d spring
### `translation`
-There are times when you want to store a string for translation, maybe with
-translation context; Sometimes you may also want to store a string that needs no
-translation or has plural forms. `class translation` in `translations.h|cpp`
-offers these functionalities in a single wrapper:
+There are times when you want to store a string for translation, maybe with translation context;
+Sometimes you may also want to store a string that needs no translation or has plural forms.
+`class translation` in `translations.h|cpp` offers these functionalities in a single wrapper:
```c++
const translation text = to_translation( "Context", "Text" );
@@ -258,9 +240,9 @@ const std::string translated = text.translated();
const std::string translated = text.translated( 2 );
```
-`class translation` can also be read from JSON. The method `translation::deserialize()`
-handles deserialization from a `JsonIn` object, so translations can be read from
-JSON using the appropriate JSON functions. The JSON syntax is as follows:
+`class translation` can also be read from JSON. The method `translation::deserialize()` handles
+deserialization from a `JsonIn` object, so translations can be read from JSON using the appropriate
+JSON functions. The JSON syntax is as follows:
```JSON
"name": "bar"
@@ -276,22 +258,20 @@ or
"name": { "ctxt": "foo", "str_sp": "foo" }
```
-In the above code, `"ctxt"` and `"str_pl"` are both optional, whereas `"str_sp"`
-is equivalent to specifying `"str"` and `"str_pl"` with the same string. Additionally,
-`"str_pl"` and `"str_sp"` will only be read if the translation object is constructed using
-`plural_tag` or `pl_translation()`, or converted using `make_plural()`. Here's
-an example:
+In the above code, `"ctxt"` and `"str_pl"` are both optional, whereas `"str_sp"` is equivalent to
+specifying `"str"` and `"str_pl"` with the same string. Additionally, `"str_pl"` and `"str_sp"` will
+only be read if the translation object is constructed using `plural_tag` or `pl_translation()`, or
+converted using `make_plural()`. Here's an example:
```c++
translation name{ translation::plural_tag() };
jsobj.read( "name", name );
```
-If neither "str_pl" nor "str_sp" is specified, the plural form defaults to the
-singular form + "s".
+If neither "str_pl" nor "str_sp" is specified, the plural form defaults to the singular form + "s".
-You can also add comments for translators by writing it like below (the order
-of the entries does not matter):
+You can also add comments for translators by writing it like below (the order of the entries does
+not matter):
```JSON
"name": {
@@ -300,107 +280,112 @@ of the entries does not matter):
}
```
-Do note that currently the JSON syntax is only supported for some JSON values,
-which are listed below. If you want other json strings to use this format,
-refer to `translations.h|cpp` and migrate the corresponding code. Afterwards
-you may also want to test `update_pot.sh` to ensure that the strings are
-correctly extracted for translation, and run the unit test to fix text styling
-issues reported by the `translation` class.
-
-| Supported JSON values
-|---
-| Effect names
-| Item action names
-| Item category names
-| Activity verbs
-| Gate action messages
-| Spell names and descriptions
-| Terrain/furniture descriptions
-| Monster melee attack messages
-| Morale effect descriptions
-| Mutation names/descriptions
-| NPC class names/descriptions
-| Tool quality names
-| Score descriptions
-| Skill names/descriptions
-| Bionic names/descriptions
-| Terrain bash sound descriptions
-| Trap-vehicle collision sound descriptions
-| Vehicle part names/descriptions
-| Skill display type names
-| NPC dialogue u_buy_monster unique names
-| Spell messages and monster spell messages
-| Martial art names and descriptions
-| Mission names and descriptions
-| Fault names and descriptions
-| Plant names in item seed data
-| Transform use action messages and menu text
-| Template NPC names and name suffixes
-| NPC talk response text
-| Relic name overrides
-| Relic recharge messages
-| Speech text
-| Tutorial messages
-| Vitamin names
-| Recipe blueprint names
-| Recipe group recipe descriptions
-| Item names (plural supported) and descriptions
-| Recipe descriptions
-| Inscribe use action verbs/gerunds
-| Monster names (plural supported) and descriptions
-| Snippets
-| Bodypart names
-| Keybinding action names
-| Field level names
-
+Do note that currently the JSON syntax is only supported for some JSON values, which are listed
+below. If you want other json strings to use this format, refer to `translations.h|cpp` and migrate
+the corresponding code. Afterwards you may also want to test `update_pot.sh` to ensure that the
+strings are correctly extracted for translation, and run the unit test to fix text styling issues
+reported by the `translation` class.
+
+| Supported JSON values |
+| ------------------------------------------------- |
+| Effect names |
+| Item action names |
+| Item category names |
+| Activity verbs |
+| Gate action messages |
+| Spell names and descriptions |
+| Terrain/furniture descriptions |
+| Monster melee attack messages |
+| Morale effect descriptions |
+| Mutation names/descriptions |
+| NPC class names/descriptions |
+| Tool quality names |
+| Score descriptions |
+| Skill names/descriptions |
+| Bionic names/descriptions |
+| Terrain bash sound descriptions |
+| Trap-vehicle collision sound descriptions |
+| Vehicle part names/descriptions |
+| Skill display type names |
+| NPC dialogue u_buy_monster unique names |
+| Spell messages and monster spell messages |
+| Martial art names and descriptions |
+| Mission names and descriptions |
+| Fault names and descriptions |
+| Plant names in item seed data |
+| Transform use action messages and menu text |
+| Template NPC names and name suffixes |
+| NPC talk response text |
+| Relic name overrides |
+| Relic recharge messages |
+| Speech text |
+| Tutorial messages |
+| Vitamin names |
+| Recipe blueprint names |
+| Recipe group recipe descriptions |
+| Item names (plural supported) and descriptions |
+| Recipe descriptions |
+| Inscribe use action verbs/gerunds |
+| Monster names (plural supported) and descriptions |
+| Snippets |
+| Bodypart names |
+| Keybinding action names |
+| Field level names |
### Lua
-The 4 translation functions are exposed to the Lua code.
-See [LUA_SUPPORT.md](LUA_SUPPORT.md) for details.
+
+The 4 translation functions are exposed to the Lua code. See [LUA_SUPPORT.md](LUA_SUPPORT.md) for
+details.
### Recommendations
-In Cataclysm: BN, some classes, like `itype` and `mtype`, provide a wrapper
-for the translation functions, called `nname`.
+In Cataclysm: BN, some classes, like `itype` and `mtype`, provide a wrapper for the translation
+functions, called `nname`.
-When an empty string is marked for translation, it is always translated into
-debug information, rather than an empty string.
-On most cases, strings can be considered to be never empty, and thus always
-safe to mark for translation, however, when handling a string that can be empty
-and *needs* to remain empty after translation, the string should be checked for
-emptiness and only passed to a translation function when is non-empty.
+When an empty string is marked for translation, it is always translated into debug information,
+rather than an empty string. On most cases, strings can be considered to be never empty, and thus
+always safe to mark for translation, however, when handling a string that can be empty and _needs_
+to remain empty after translation, the string should be checked for emptiness and only passed to a
+translation function when is non-empty.
-Error and debug messages must not be marked for translation.
-When they appear, the player is expected to report them *exactly* as they are
-printed by the game.
+Error and debug messages must not be marked for translation. When they appear, the player is
+expected to report them _exactly_ as they are printed by the game.
See the [gettext manual][5] for more information.
## Maintainers
-Several steps need to be done in the correct order to correctly merge and maintain the translation files.
+Several steps need to be done in the correct order to correctly merge and maintain the translation
+files.
There are scripts available for these, so usually the process will be as follows:
1. Download the translations in `.po` format.
2. Put them in `lang/incoming/`, ensuring they are named consistently with the files in `lang/po/`.
-3. Run `lang/update_pot.sh` to update `lang/po/cataclysm-BN.pot` (requires python with `polib` and `luaparser` modules installed).
-4. Run `lang/merge_po.sh` to update `lang/po/*.po`. (This is only used to test translations locally as the project now uses Transifex for translation)
+3. Run `lang/update_pot.sh` to update `lang/po/cataclysm-BN.pot` (requires python with `polib` and
+ `luaparser` modules installed).
+4. Run `lang/merge_po.sh` to update `lang/po/*.po`. (This is only used to test translations locally
+ as the project now uses Transifex for translation)
- This will also merge the translations from `lang/incoming/`.
+ This will also merge the translations from `lang/incoming/`.
These steps should be enough to keep the translation files up-to-date.
-To compile the .po files into `.mo` files for use, run `lang/compile_mo.sh`. It will create a directory in `lang/mo/` for each language found.
-
-Also note that both `lang/merge_po.sh` and `lang/compile_mo.sh` accept arguments specifying which languages to merge or compile. So to compile only the translation for, say, Traditional Chinese (zh_TW), one would run `lang/compile_mo.sh zh_TW`.
+To compile the .po files into `.mo` files for use, run `lang/compile_mo.sh`. It will create a
+directory in `lang/mo/` for each language found.
-After compiling the appropriate .mo file, if the language has been selected in game settings, the translations will be automatically used when you run cataclysm.
+Also note that both `lang/merge_po.sh` and `lang/compile_mo.sh` accept arguments specifying which
+languages to merge or compile. So to compile only the translation for, say, Traditional Chinese
+(zh_TW), one would run `lang/compile_mo.sh zh_TW`.
-When `System language` is selected in settings, the game tries to use language that matches system language based on language definitions file `data/raw/languages.json`.
+After compiling the appropriate .mo file, if the language has been selected in game settings, the
+translations will be automatically used when you run cataclysm.
-If you're testing translations for a new language, or the language does not show up in settings, make sure it has its own entry in the definitions file.
+When `System language` is selected in settings, the game tries to use language that matches system
+language based on language definitions file `data/raw/languages.json`.
+If you're testing translations for a new language, or the language does not show up in settings,
+make sure it has its own entry in the definitions file.
[1]: https://app.transifex.com/bn-team/cataclysm-bright-nights
[2]: https://docs.transifex.com/
diff --git a/doc/TRANSLATING_MODS.md b/doc/TRANSLATING_MODS.md
index 95f1cf99f3f7..d4d12fb42649 100644
--- a/doc/TRANSLATING_MODS.md
+++ b/doc/TRANSLATING_MODS.md
@@ -27,42 +27,49 @@
- [What if 2 or more mods provide different translations for same string?](#what-if-2-or-more-mods-provide-different-translations-for-same-string)
## Intro
-This document aims to give a brief explanation on how to set up and operate
-mod translation workflow for Cataclysm: Bright Nights.
+
+This document aims to give a brief explanation on how to set up and operate mod translation workflow
+for Cataclysm: Bright Nights.
For mod localization the game uses custom localization system that is similar to
[GNU gettext](https://www.gnu.org/software/gettext/) and is compatible with GNU gettext MO files.
-While it's possible to use Transifex or any other platform or software that supports gettext,
-this document only gives examples on how to work with [Poedit](https://poedit.net/) and
-command-line [GNU gettext utilities](https://www.gnu.org/software/gettext/).
+While it's possible to use Transifex or any other platform or software that supports gettext, this
+document only gives examples on how to work with [Poedit](https://poedit.net/) and command-line
+[GNU gettext utilities](https://www.gnu.org/software/gettext/).
-If you desire an in-depth explanation on PO/POT/MO files or how to work with them using GNU gettext utilities,
-see [GNU gettext manual](https://www.gnu.org/software/gettext/manual/gettext.html).
+If you desire an in-depth explanation on PO/POT/MO files or how to work with them using GNU gettext
+utilities, see [GNU gettext manual](https://www.gnu.org/software/gettext/manual/gettext.html).
-To get some generic tips on translating strings for Cataclysm: Bright Nights and its mods,
-see [TRANSLATING.md](TRANSLATING.md).
+To get some generic tips on translating strings for Cataclysm: Bright Nights and its mods, see
+[TRANSLATING.md](TRANSLATING.md).
## A short glossary
+
### POT file
+
Portable Object Template file (`.pot`).
-This is a text file that contains original (English) strings extracted from mod's JSON and Lua source files.
-The POT file is a template used for creating empty or updating existing PO files of any language.
+This is a text file that contains original (English) strings extracted from mod's JSON and Lua
+source files. The POT file is a template used for creating empty or updating existing PO files of
+any language.
### PO file
+
Portable Object file (`.po`).
-This is a text file that contains translated strings for one language.
-The PO files are what translators work with, and what will be compiled into a MO file.
+This is a text file that contains translated strings for one language. The PO files are what
+translators work with, and what will be compiled into a MO file.
### MO file
+
Machine Object file (`.mo`).
-This is a binary file that contains translated strings for one language.
-The MO files are what the game loads, and where it gets translated strings from.
+This is a binary file that contains translated strings for one language. The MO files are what the
+game loads, and where it gets translated strings from.
## Workflow overview
+
The first translation workflow is as follows:
1. Extract strings from mod JSON and Lua source files into a POT file
@@ -71,8 +78,8 @@ The first translation workflow is as follows:
4. Compile PO into MO
5. Put the MO into your mod files
-As the mod changes with time, so may change its strings.
-Updating existing translations is done as follows:
+As the mod changes with time, so may change its strings. Updating existing translations is done as
+follows:
1. Extract strings from mod JSON and Lua source files into a new POT file
2. Update existing PO file from the new POT file
@@ -82,71 +89,87 @@ Updating existing translations is done as follows:
Step 1 in both workflows requires you to set up environment for string extraction (see below).
-Steps 2-4 can be done using translation software either by the mod author/maintainer, or by the translator.
+Steps 2-4 can be done using translation software either by the mod author/maintainer, or by the
+translator.
## Setting up environment for string extraction
+
You'll need Python 3 with `polib` and `luaparser` modules installed (available via `pip`).
Scripts for string extraction can be found in the `lang` subdirectory of the repository:
-* `extract_json_strings.py` - main string extraction routines
-* `dedup_pot_file.py` - fixes errors in POT file produces by the 1st script
-* `extract_mod_strings.bat` (`extract_mod_strings.sh` for Linux/MacOS) - to automate the other 2 scripts
+
+- `extract_json_strings.py` - main string extraction routines
+- `dedup_pot_file.py` - fixes errors in POT file produces by the 1st script
+- `extract_mod_strings.bat` (`extract_mod_strings.sh` for Linux/MacOS) - to automate the other 2
+ scripts
## Extracting strings
+
Copy these 3 scripts into the mod's folder and:
-* on Windows, double-click `extract_mod_strings.bat`
-* on Linux/MacOS, open terminal and run `./extract_mod_strings.sh`
-If the process completed without errors, you'll see a new `lang` folder
-with `extracted_strings.pot` file inside.
+- on Windows, double-click `extract_mod_strings.bat`
+- on Linux/MacOS, open terminal and run `./extract_mod_strings.sh`
+
+If the process completed without errors, you'll see a new `lang` folder with `extracted_strings.pot`
+file inside.
## Creating new PO
+
Before creating PO file, you need to choose language id.
Open `data/raw/languages.json` to see the list of languages supported by the game.
-In this list, each entry has its own id in form of `ln_LN`,
-where `ln` determines language and `LN` - dialect.
-You can either use full `ln_LN` for exact language+dialect match,
-or `ln` if you want the game to use your MO regardless of dialect.
+In this list, each entry has its own id in form of `ln_LN`, where `ln` determines language and
+`LN` - dialect. You can either use full `ln_LN` for exact language+dialect match, or `ln` if you
+want the game to use your MO regardless of dialect.
### Poedit
+
1. Open the POT file with Poedit
2. Press "Create new translation" button (should show up near the bottom)
3. In language selection dialog, enter language id you chose
4. Save the file as `path/to/mod/lang/LANG.po` where `LANG` is the same language id
### msginit
+
```bash
msginit -i lang/index.pot -o lang/LANG.po -l LANG.UTF-8 --no-translator
```
+
Where `LANG` is the language id you chose
## Updating existing PO
+
### Poedit
+
1. Open the PO file with Poedit
2. Choose `Catalog->Update from POT file...` and select the new POT file
3. Save the file
### msgmerge
+
```bash
msgmerge lang/LANG.po lang/index.pot
```
## Compiling PO into MO
+
### Poedit
+
1. Open the PO file with Poedit
-2. Make sure MO file will be encoded using UTF-8 (it should do so by default,
- you can double check in `Catalog->Properties->"Translation properties" tab->Charset`).
-3. By default, each time PO file is saved Poedit automatically compiles it into MO,
- but the same can also be done explicitly via `File->Compile to MO...`
+2. Make sure MO file will be encoded using UTF-8 (it should do so by default, you can double check
+ in `Catalog->Properties->"Translation properties" tab->Charset`).
+3. By default, each time PO file is saved Poedit automatically compiles it into MO, but the same can
+ also be done explicitly via `File->Compile to MO...`
### msgfmt
+
```
msgfmt -o lang/LANG.mo lang/LANG.po
```
## Adding MO file to the mod
+
Create `lang` directory in the mod files directory and put MOs there:
```
@@ -159,9 +182,8 @@ mods/
zh_CN.mo
```
-**Note:** Storing your POT/PO files
-in the same `lang` subdirectory may make it easier to keep track of them.
-The game ignores these files, and your mod folder structure will look like this:
+**Note:** Storing your POT/PO files in the same `lang` subdirectory may make it easier to keep track
+of them. The game ignores these files, and your mod folder structure will look like this:
```
mods/
@@ -178,23 +200,27 @@ mods/
```
## Miscellaneous notes
+
### Is it possible to use arbitrary location or names for MO files, like with JSONs?
-No. The game looks for MO files with specific names that are located in the
-`lang` subdirectory of the mod's `path` directory specified in `modinfo.json`
-(if not specified, `path` matches the mod's directory).
-However, any mod will automatically try to use any other mod's translation
-files to translate its strings. This makes it possible to create mods that are
-purely "translation packs" for other mods (or mod collections).
+No. The game looks for MO files with specific names that are located in the `lang` subdirectory of
+the mod's `path` directory specified in `modinfo.json` (if not specified, `path` matches the mod's
+directory).
+
+However, any mod will automatically try to use any other mod's translation files to translate its
+strings. This makes it possible to create mods that are purely "translation packs" for other mods
+(or mod collections).
### Reloading translations in a running game
-Open debug menu and select `Info...->Reload translations`,
-and the game will reload all MO files from disk.
-This makes it easy to see how translated string looks in game,
-provided the translator has a way to compile MO files.
+Open debug menu and select `Info...->Reload translations`, and the game will reload all MO files
+from disk.
+
+This makes it easy to see how translated string looks in game, provided the translator has a way to
+compile MO files.
Example workflow with Poedit:
+
1. Translate a string
2. Hit Ctrl+S
3. Alt+Tab into the game
@@ -202,35 +228,40 @@ Example workflow with Poedit:
5. The game now displays translated string
### MO load order
+
MO load order is as follows:
-1. First and always goes base game's MO file, which contains translation strings
- for UI, hardcoded functionality, base "mod" (`data/json/`) and in-repo mods.
+
+1. First and always goes base game's MO file, which contains translation strings for UI, hardcoded
+ functionality, base "mod" (`data/json/`) and in-repo mods.
2. Then MO files of mods are loaded, in same order as the mod load order.
### Dialects
-When loading MO files, the game first looks for the file with
-exact language and dialect match in its name.
-If such file is absent, then it looks for a file with no dialect.
+
+When loading MO files, the game first looks for the file with exact language and dialect match in
+its name. If such file is absent, then it looks for a file with no dialect.
For example, when using `Español (España)` the selection order is
+
1. `es_ES.mo`
2. `es.mo`
And when using `Español (Argentina)` the selection order is
+
1. `es_AR.mo`
2. `es.mo`
-Thus, `es.mo` would be loaded for either dialect of Spanish
-if the exact translation files are not present.
+Thus, `es.mo` would be loaded for either dialect of Spanish if the exact translation files are not
+present.
### What if 2 or more mods provide different translations for same string?
+
Then the game selects which one to use according to this set of rules:
-1. If string A's translation has plural forms, but string B's translation does not,
- then translation A is used for both single and plural forms.
-2. If both translation A and B have (or both don't have) plural forms, then
- the first loaded translation is used (see MO load order).
-
-If you want a different translation from the one in the base game,
-or don't want it to conflict with a string from some other mod,
-add a translation context to the string in the corresponding JSON object
-(see [TRANSLATING.md](TRANSLATING.md) for which fields support translation context).
+
+1. If string A's translation has plural forms, but string B's translation does not, then translation
+ A is used for both single and plural forms.
+2. If both translation A and B have (or both don't have) plural forms, then the first loaded
+ translation is used (see MO load order).
+
+If you want a different translation from the one in the base game, or don't want it to conflict with
+a string from some other mod, add a translation context to the string in the corresponding JSON
+object (see [TRANSLATING.md](TRANSLATING.md) for which fields support translation context).
diff --git a/doc/USER_INTERFACE.md b/doc/USER_INTERFACE.md
index 87f0ba27bcf4..7d58e9904f57 100644
--- a/doc/USER_INTERFACE.md
+++ b/doc/USER_INTERFACE.md
@@ -1,12 +1,11 @@
# User Interface
-Cataclysm: Bright Nights uses ncurses, or in the case of the tiles build, an
-ncurses port, for user interface. Window management is achieved by `ui_adaptor`,
-which requires a resizing callback and a redrawing callback for each UI to handle
-resizing and redrawing. Details on how to use `ui_adaptor` can be found within
-`ui_manager.h`.
+Cataclysm: Bright Nights uses ncurses, or in the case of the tiles build, an ncurses port, for user
+interface. Window management is achieved by `ui_adaptor`, which requires a resizing callback and a
+redrawing callback for each UI to handle resizing and redrawing. Details on how to use `ui_adaptor`
+can be found within `ui_manager.h`.
+
+Some good examples of the usage of `ui_adaptor` can be found within the following files:
-Some good examples of the usage of `ui_adaptor` can be found within the following
-files:
- `query_popup` and `static_popup` in `popup.h/cpp`
- `Messages::dialog` in `messages.cpp`
diff --git a/doc/VEHICLES_JSON.md b/doc/VEHICLES_JSON.md
index 962a3d2f6753..7cddfb36d046 100644
--- a/doc/VEHICLES_JSON.md
+++ b/doc/VEHICLES_JSON.md
@@ -1,6 +1,7 @@
# Vehicle prototypes JSON file contents
-Vehicle prototypes are used to spawn stock vehicles. After a vehicle has been spawned, it is saved in a different format.
+Vehicle prototypes are used to spawn stock vehicles. After a vehicle has been spawned, it is saved
+in a different format.
Vehicle prototypes do not currently accept copy-from
@@ -35,27 +36,32 @@ Vehicle prototypes do not currently accept copy-from
]
```
-.* Important! *. Vehicle parts must be defined in the same order you would install them in the game (ie, frames and mount points first). You also cannot break the normal rules of installation (you can't stack non-stackable part flags).
+.* Important! *. Vehicle parts must be defined in the same order you would install them in the game
+(ie, frames and mount points first). You also cannot break the normal rules of installation (you
+can't stack non-stackable part flags).
### Parts list
-The part list contains an arbitary number of lines. Each line is of the form:
- { "x": X, "y": Y, "part": PARTID, ... }
-or
- { "x": X, "y": Y, "parts": [ PARTID1, ... ] }
-In the first form, the line defines a single part at location X,Y of vehicle part type PARTID. It can have the optional "ammo", "ammo_types", "ammo_qty", or "fuel" keys with an appropriate value following.
+The part list contains an arbitary number of lines. Each line is of the form: { "x": X, "y": Y,
+"part": PARTID, ... } or { "x": X, "y": Y, "parts": [ PARTID1, ... ] }
-In the second form, the line defines several parts at location X, Y. Each part is either defined by its PARTID string, or can be an object of the form
- { "part": PARTID, ... }
-with any of the optional keys "ammo", "ammo_types", "ammo_qty", or "fuel" as above.
+In the first form, the line defines a single part at location X,Y of vehicle part type PARTID. It
+can have the optional "ammo", "ammo_types", "ammo_qty", or "fuel" keys with an appropriate value
+following.
-Several different lines can have the same X, Y co-ordinates and each one adds additional parts to that location. Parts must be added in the correct order ie: A wheel hub must be added prior to the wheel, but after the frame.
+In the second form, the line defines several parts at location X, Y. Each part is either defined by
+its PARTID string, or can be an object of the form { "part": PARTID, ... } with any of the optional
+keys "ammo", "ammo_types", "ammo_qty", or "fuel" as above.
+
+Several different lines can have the same X, Y co-ordinates and each one adds additional parts to
+that location. Parts must be added in the correct order ie: A wheel hub must be added prior to the
+wheel, but after the frame.
### Items list
-The items list contains an arbitrary number of lines. Each line is of the form:
- { "x": X, "y": Y, TYPE: DATA },
-and describes the items that may spawn at that location.
-TYPE and DATA may be one of:
+
+The items list contains an arbitrary number of lines. Each line is of the form: { "x": X, "y": Y,
+TYPE: DATA }, and describes the items that may spawn at that location. TYPE and DATA may be one of:
+
```C++
"items": "itemid" // single item of that type
"items": [ "itemid1", "itemid2", ... ] // all the items in the array
@@ -63,6 +69,8 @@ TYPE and DATA may be one of:
// whether the group is a collection or distribution
"item_groups": [ "groupid1", "groupid2" ... ] // one or more items for each group
```
-the optional keyword "chance" provides an X in 100 chance that a particular item definition will spawn.
+
+the optional keyword "chance" provides an X in 100 chance that a particular item definition will
+spawn.
Multiple lines of items may share the same X and Y values.
diff --git a/doc/VITAMIN.md b/doc/VITAMIN.md
index 2abcb854a270..363522a34b34 100644
--- a/doc/VITAMIN.md
+++ b/doc/VITAMIN.md
@@ -18,59 +18,79 @@
"disease_excess": [ [ 10, 19 ], [ 20, 29 ], [ 30, 40 ] ]
},
```
+
### `id`
+
Mandatory. The id of the vitamin.
### `type`
+
Mandatory. Must be `vitamin`.
### `vit_type`
+
The type of the vitamin. Valid values are:
#### `vitamin`
-When simplified nutrition is enabled, this vitamin will not be added to any items and any time the game attempts to retrieve it from the player it will give 0.
-Only nutritional vitamins should have this type.
+
+When simplified nutrition is enabled, this vitamin will not be added to any items and any time the
+game attempts to retrieve it from the player it will give 0. Only nutritional vitamins should have
+this type.
#### `toxin`
+
This is some toxic chemical or component. This currently has no effect.
#### `drug`
+
This is a drug. This currently has no effect.
#### `counter`
-This is a counter for something, that is neither a toxin, vitamin, or drug. This currently has no effect.
+
+This is a counter for something, that is neither a toxin, vitamin, or drug. This currently has no
+effect.
### `name`
-What the vitamin shows up as where vitamins are displayed, such as the vitamins display in the item menu.
+
+What the vitamin shows up as where vitamins are displayed, such as the vitamins display in the item
+menu.
### `deficiency`
+
The id of an effect that is triggered by a deficiency of this vitamin.
### `excess`
+
The id of an effect that is triggered by a excess of this vitamin.
### `min`
+
The smallest amount of this vitamin that the player can have.
### `max`
+
The highest amount of this vitamin that the avatar can have.
### `rate`
+
How long it takes to lose one unit of this vitamin.
### `flags`
+
An array of string flags, see the flags section below for valid ones
### `disease`
-What the thresholds of deficiency of this vitamin are.
-Each pair in the list determines the start and end points of that tier of deficiency.
-Each tier of deficiency corresponds to the intensity level of the effect defined in `deficiency`.
+
+What the thresholds of deficiency of this vitamin are. Each pair in the list determines the start
+and end points of that tier of deficiency. Each tier of deficiency corresponds to the intensity
+level of the effect defined in `deficiency`.
### `disease_excess`
-What the thresholds of excess of this vitamin are.
-Each pair in the list determines the start and end points of that tier of excess.
-Each tier of excess corresponds to the intensity level of the effect defined in `excess`.
+
+What the thresholds of excess of this vitamin are. Each pair in the list determines the start and
+end points of that tier of excess. Each tier of excess corresponds to the intensity level of the
+effect defined in `excess`.
## flags
-- ```NO_DISPLAY``` - This vitamin will not be shown when examining a food
+- `NO_DISPLAY` - This vitamin will not be shown when examining a food
diff --git a/doc/WEATHER_TYPE.md b/doc/WEATHER_TYPE.md
index b5e699b1126a..14cb84205926 100644
--- a/doc/WEATHER_TYPE.md
+++ b/doc/WEATHER_TYPE.md
@@ -1,98 +1,100 @@
## Weather_type
-Weather type specifies conditions under which it can occur
-(temperature, humidity, pressure, windpower, time of day, etc.)
-and what effects it causes on the game world and reality bubble.
+Weather type specifies conditions under which it can occur (temperature, humidity, pressure,
+windpower, time of day, etc.) and what effects it causes on the game world and reality bubble.
-When selecting weather type, the game goes over the list defined in region settings
-and selects the last entry that is considered eligible under current conditions.
-If none of the entries are eligible, invalid weather type `"none"` will be used.
+When selecting weather type, the game goes over the list defined in region settings and selects the
+last entry that is considered eligible under current conditions. If none of the entries are
+eligible, invalid weather type `"none"` will be used.
### Fields
-| Identifier | Description
-|--- |---
-| id | (_mandatory_) Unique ID. Must be one continuous word, use underscores if necessary.
-| name | (_mandatory_) In-game name displayed.
-| color | (_mandatory_) Color of in-game name.
-| glyph | (_mandatory_) Glyph used on overmap.
-| map_color | (_mandatory_) Color of overmap glyph.
-| ranged_penalty | (_mandatory_) Penalty to ranged attacks.
-| sight_penalty | (_mandatory_) Sight penalty, aka multiplier to tile transparency.
-| light_modifier | (_mandatory_) Flat bonus to ambient light.
-| sound_attn | (_mandatory_) Sound attenuation (flat reduction to volume).
-| dangerous | (_mandatory_) If true, prompts for activity interrupt.
-| precip | (_mandatory_) Amount of associated precipitation. Valid values are: `none`, `very_light`, `light` and `heavy`.
-| rains | (_mandatory_) Whether said precipitation falls as rain.
-| acidic | (_optional_) Whether said precipitation is acidic.
-| sound_category | (_optional_) Sound effects to play. Valid values are: `silent`, `drizzle`, `rainy`, `thunder`, `flurries`, `snowstorm` and `snow`.
-| sun_intensity | (_mandatory_) Sunlight intensity. Valid values are: `none`, `light`, `normal`, and `high`. Normal and high are considered "direct sunlight".
-| weather_animation | (_optional_) Weather animation in reality bubble. [Details](#weather_animation)
-| effects | (_optional_) `[string, int]` pair array for the effects the weather causes. [Details](#effects)
-| requirements | (_optional_) Conditions under which this weather type will be eligible to be selected. [Details](#requirements)
+| Identifier | Description |
+| ----------------- | -------------------------------------------------------------------------------------------------------------------------------------------- |
+| id | (_mandatory_) Unique ID. Must be one continuous word, use underscores if necessary. |
+| name | (_mandatory_) In-game name displayed. |
+| color | (_mandatory_) Color of in-game name. |
+| glyph | (_mandatory_) Glyph used on overmap. |
+| map_color | (_mandatory_) Color of overmap glyph. |
+| ranged_penalty | (_mandatory_) Penalty to ranged attacks. |
+| sight_penalty | (_mandatory_) Sight penalty, aka multiplier to tile transparency. |
+| light_modifier | (_mandatory_) Flat bonus to ambient light. |
+| sound_attn | (_mandatory_) Sound attenuation (flat reduction to volume). |
+| dangerous | (_mandatory_) If true, prompts for activity interrupt. |
+| precip | (_mandatory_) Amount of associated precipitation. Valid values are: `none`, `very_light`, `light` and `heavy`. |
+| rains | (_mandatory_) Whether said precipitation falls as rain. |
+| acidic | (_optional_) Whether said precipitation is acidic. |
+| sound_category | (_optional_) Sound effects to play. Valid values are: `silent`, `drizzle`, `rainy`, `thunder`, `flurries`, `snowstorm` and `snow`. |
+| sun_intensity | (_mandatory_) Sunlight intensity. Valid values are: `none`, `light`, `normal`, and `high`. Normal and high are considered "direct sunlight". |
+| weather_animation | (_optional_) Weather animation in reality bubble. [Details](#weather_animation) |
+| effects | (_optional_) `[string, int]` pair array for the effects the weather causes. [Details](#effects) |
+| requirements | (_optional_) Conditions under which this weather type will be eligible to be selected. [Details](#requirements) |
### weather_animation
+
All members are mandatory.
-| Identifier | Description
-|--- |---
-| factor | Display density: 0 is none, 1 will blot out the screen.
-| glyph | Glyph to use in ASCII mode.
-| color | Glyph color.
-| tile | Graphical tile to use in TILES mode.
+| Identifier | Description |
+| ---------- | ------------------------------------------------------- |
+| factor | Display density: 0 is none, 1 will blot out the screen. |
+| glyph | Glyph to use in ASCII mode. |
+| color | Glyph color. |
+| tile | Graphical tile to use in TILES mode. |
### effects
+
`int` here is effect intensity.
-| Identifier | Description
-|--- |---
-| wet | wets player by `int` amount
-| thunder | thunder sound with chance 1 in `int`
-| lightening | 1 in `int` chance of sound plus message and possible super charging electric fields
-| light_acid | causes pain unless waterproof
-| acid_rain | causes more pain unless waterproof
+| Identifier | Description |
+| ---------- | ----------------------------------------------------------------------------------- |
+| wet | wets player by `int` amount |
+| thunder | thunder sound with chance 1 in `int` |
+| lightening | 1 in `int` chance of sound plus message and possible super charging electric fields |
+| light_acid | causes pain unless waterproof |
+| acid_rain | causes more pain unless waterproof |
### requirements
+
All members are optional.
-| Identifier | Description
-|--- |---
-| pressure_min | Min pressure
-| pressure_max | Max pressure
-| humidity_min | Min humidity
-| humidity_max | Max humidity
-| temperature_min | Min temperature
-| temperature_max | Max temperature
-| windpower_min | Min windpower
-| windpower_max | Max windpower
-| humidity_and_pressure | If there are pressure and humidity requirements are they both required or just one
-| acidic | Does this require acidic precipitation
-| time | Time of day. Valid values are: day, night, and both.
-| required_weathers | Will only be selected if conditions match for any of the specified types, i.e. rain can only happen if the conditions for clouds, light drizzle or drizzle are present. Required weathers should be "before" this one in the region weather list.
+| Identifier | Description |
+| --------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| pressure_min | Min pressure |
+| pressure_max | Max pressure |
+| humidity_min | Min humidity |
+| humidity_max | Max humidity |
+| temperature_min | Min temperature |
+| temperature_max | Max temperature |
+| windpower_min | Min windpower |
+| windpower_max | Max windpower |
+| humidity_and_pressure | If there are pressure and humidity requirements are they both required or just one |
+| acidic | Does this require acidic precipitation |
+| time | Time of day. Valid values are: day, night, and both. |
+| required_weathers | Will only be selected if conditions match for any of the specified types, i.e. rain can only happen if the conditions for clouds, light drizzle or drizzle are present. Required weathers should be "before" this one in the region weather list. |
### Example
```json
{
- "id": "lightning",
- "type": "weather_type",
- "name": "Lightning Storm",
- "color": "c_yellow",
- "map_color": "h_yellow",
- "glyph": "%",
- "ranged_penalty": 4,
- "sight_penalty": 1.25,
- "light_modifier": -45,
- "sound_attn": 8,
- "dangerous": false,
- "precip": "heavy",
- "rains": true,
- "acidic": false,
- "effects": [ { "name": "thunder", "intensity": 50 }, { "name": "lightning", "intensity": 600 } ],
- "tiles_animation": "weather_rain_drop",
- "weather_animation": { "factor": 0.04, "color": "c_light_blue", "glyph": "," },
- "sound_category": "thunder",
- "sun_intensity": "none",
- "requirements": { "pressure_max": 990, "required_weathers": [ "thunder" ] }
+ "id": "lightning",
+ "type": "weather_type",
+ "name": "Lightning Storm",
+ "color": "c_yellow",
+ "map_color": "h_yellow",
+ "glyph": "%",
+ "ranged_penalty": 4,
+ "sight_penalty": 1.25,
+ "light_modifier": -45,
+ "sound_attn": 8,
+ "dangerous": false,
+ "precip": "heavy",
+ "rains": true,
+ "acidic": false,
+ "effects": [{ "name": "thunder", "intensity": 50 }, { "name": "lightning", "intensity": 600 }],
+ "tiles_animation": "weather_rain_drop",
+ "weather_animation": { "factor": 0.04, "color": "c_light_blue", "glyph": "," },
+ "sound_category": "thunder",
+ "sun_intensity": "none",
+ "requirements": { "pressure_max": 990, "required_weathers": ["thunder"] }
}
```
diff --git a/doc/melee_weapons/MELEE_BALANCE_SPREADSHEET.md b/doc/melee_weapons/MELEE_BALANCE_SPREADSHEET.md
index 7b73ac3cd3e4..aa866ada87a9 100644
--- a/doc/melee_weapons/MELEE_BALANCE_SPREADSHEET.md
+++ b/doc/melee_weapons/MELEE_BALANCE_SPREADSHEET.md
@@ -1,69 +1,138 @@
## Overview
-This is documentation for the 'Melee Weapons Evaluation.md' spreadsheet. This is the spreadsheet that drove the 0.E experimental melee weapon rebalance project. The original spreadsheet is available as a [google doc](https://docs.google.com/spreadsheets/d/14eQIe4AO_f6OxCt1XcB4NLAs6-5R1wQW-ydZG0orYdY/edit#gid=1787713396) but the static copy is preserved in case stuff moves.
-This is a complicated spreadsheet, with a lot of moving parts. The following are the tabs and the data flow:
+This is documentation for the 'Melee Weapons Evaluation.md' spreadsheet. This is the spreadsheet
+that drove the 0.E experimental melee weapon rebalance project. The original spreadsheet is
+available as a
+[google doc](https://docs.google.com/spreadsheets/d/14eQIe4AO_f6OxCt1XcB4NLAs6-5R1wQW-ydZG0orYdY/edit#gid=1787713396)
+but the static copy is preserved in case stuff moves.
+
+This is a complicated spreadsheet, with a lot of moving parts. The following are the tabs and the
+data flow:
### Raw
-Raw was the initial data dump, taken from some October 2019 version of the 0.D experimental using mlangsdorf's usual mods (hence all the Blazemod stuff). Raw was slightly annoyed with some extra categories:
-* Weapon Class - this is a simple numeric rating. 0 is not a weapon, 1 is an explosive device, 2 is a ranged weapon or gun, 3 is a tool that makes a poor weapon like a cooking pot, 4 is a helmet, 5 is a tool that makes a good weapon like a sledgehammer, 6 is an improvised or fake weapon, and 7 is an actual dedicated weapon
-* Category - this is a categorization of items as weapons, to make it easier to compare similar weapons. Melee weapon categories are Ax, Ax2, Club, Club2, Flail, Flail2, Knife, Polearm, Spear, Staff, Shortsword, Sword, and Sword2. Categories that end with 2 are two-handed weapons.
-* Acc - this is a recaluclation of weapon accuracy based on the accuracy factors below, because review of the data showed that a lot of weapons have accuracy values that are not supported by the criteria in GAME_BALANCE.md.
-* Grip - the item's grip, as described in GAME_BALANCE.md
-* Leng - the item's leng, as described in GAME_BALANCE.md
-* Surf - the item's striking surface, as described in GAME_BALANCE.md.
-* Bal - the item's balance, as described in GAME_BALANCE.md. Additional categories beyond Clumsy were added and assigned by eye.
-### Filter
-Filter takes the initial data from raw and a weapon class (in cell B1) and filters out items with a lower weapon class to make further analysis easier.
+Raw was the initial data dump, taken from some October 2019 version of the 0.D experimental using
+mlangsdorf's usual mods (hence all the Blazemod stuff). Raw was slightly annoyed with some extra
+categories:
+
+- Weapon Class - this is a simple numeric rating. 0 is not a weapon, 1 is an explosive device, 2 is
+ a ranged weapon or gun, 3 is a tool that makes a poor weapon like a cooking pot, 4 is a helmet, 5
+ is a tool that makes a good weapon like a sledgehammer, 6 is an improvised or fake weapon, and 7
+ is an actual dedicated weapon
+- Category - this is a categorization of items as weapons, to make it easier to compare similar
+ weapons. Melee weapon categories are Ax, Ax2, Club, Club2, Flail, Flail2, Knife, Polearm, Spear,
+ Staff, Shortsword, Sword, and Sword2. Categories that end with 2 are two-handed weapons.
+- Acc - this is a recaluclation of weapon accuracy based on the accuracy factors below, because
+ review of the data showed that a lot of weapons have accuracy values that are not supported by the
+ criteria in GAME_BALANCE.md.
+- Grip - the item's grip, as described in GAME_BALANCE.md
+- Leng - the item's leng, as described in GAME_BALANCE.md
+- Surf - the item's striking surface, as described in GAME_BALANCE.md.
+- Bal - the item's balance, as described in GAME_BALANCE.md. Additional categories beyond Clumsy
+ were added and assigned by eye.
-### New Formula
-This calculates the new weapon evaluation formaula on the inital stats. This is complicated.
-
-The first 7 rows have some header data. The bulk of the calculation starts on row 8.
+### Filter
-The weapons were evaluated using a Strength 10, Dexterity 10, Perception 10 survivor with skill 4 in all weapons. The base hit, stat crit, skill crit, bash mult, cut mult, and stab mult values are derived from the relevant bits of src/melee.cpp.
+Filter takes the initial data from raw and a weapon class (in cell B1) and filters out items with a
+lower weapon class to make further analysis easier.
-* Columnn A "Average" is the new weapon evaluation value for the weapon, on the edge of the sheet for easy reference.
-* Columns B-J are the weapon's original stats, taken from the Raw tab via the Filter tab.
-* Column K "roll_hit" is sum of the evaluator's base_hit and the weapon's acc.
-* Column L "Wpn Crit" is weapon's contribution to critical hits.
-* Columns M-N "3 Crit" and "2 Crit" are the chances of a triple and double critical hit occuring, based on skill, accuracy, and stats.
-* Columns O-Q "Average Non-Crit" damage are the weapon's calculated average damage before armor for non-critical hits in each of the 3 categories.
+### New Formula
-All that is pretty straightforward. The next three blocks are where it gets complicated. Columns S-Z are repeated as AB-AI and AK-AR with different monster stats.
-* Row S "Hits" is the expected number of hits per 1000 attacks, calculated against a normal distribution with a mean value of 5 * ( roll_hit - monster Dodge ) and a standard deviation of 25.
-* Row T "Crit %s" is the percentage of those hits that should be critical hits.
-* Row U "Dmg" is the average expected damage past armor. This is simply the sum of max( 0, damage type - armor amount ) for each of the 3 damage types.
-* Row V "Crit" is the average expected critical damage past armor. This is the sum of the critical damage - armor amount for each of the three damage amounts, but the formula is substantially more complicated because the 3 damage types have different critical damage multipliers and reduce effective armor by different amounts on a critical.
-* Rows W-Y repeat damage and critical damage, but for rapid strikes where the base damage is multiplied by 2/3rds.
-* Row Z "Dmg/Turn" is either:
-** 100 * ( Dmg * ( Hits - num crits ) + Crit Dmg * ( num hits ) ) / ( 1000 * Moves ) - ie, sum of damage per hit for each normal hit + crit damage for each critical hit divided by the number of moves in 1000 attacks, multiplied by 100 moves/second, OR
-** 100 * ( Dmg * ( Hits - num crits ) / 2 + Crit Dmg * ( num hits ) / 2 + rapid strike Dmg * ( Hits - num crits ) / 2 + rapid strike Crit Dmg * ( num hits ) / 2 ) / ( 1000 - ( hits / 2 ) * Moves + Hits * 0.33 * Moves ) - same as above, but accounting for rapid strike reduced damage and movement cost.
+This calculates the new weapon evaluation formaula on the inital stats. This is complicated.
+
+The first 7 rows have some header data. The bulk of the calculation starts on row 8.
+
+The weapons were evaluated using a Strength 10, Dexterity 10, Perception 10 survivor with skill 4 in
+all weapons. The base hit, stat crit, skill crit, bash mult, cut mult, and stab mult values are
+derived from the relevant bits of src/melee.cpp.
+
+- Columnn A "Average" is the new weapon evaluation value for the weapon, on the edge of the sheet
+ for easy reference.
+- Columns B-J are the weapon's original stats, taken from the Raw tab via the Filter tab.
+- Column K "roll_hit" is sum of the evaluator's base_hit and the weapon's acc.
+- Column L "Wpn Crit" is weapon's contribution to critical hits.
+- Columns M-N "3 Crit" and "2 Crit" are the chances of a triple and double critical hit occuring,
+ based on skill, accuracy, and stats.
+- Columns O-Q "Average Non-Crit" damage are the weapon's calculated average damage before armor for
+ non-critical hits in each of the 3 categories.
+
+All that is pretty straightforward. The next three blocks are where it gets complicated. Columns S-Z
+are repeated as AB-AI and AK-AR with different monster stats.
+
+- Row S "Hits" is the expected number of hits per 1000 attacks, calculated against a normal
+ distribution with a mean value of 5 * ( roll_hit - monster Dodge ) and a standard deviation of 25.
+- Row T "Crit %s" is the percentage of those hits that should be critical hits.
+- Row U "Dmg" is the average expected damage past armor. This is simply the sum of max( 0, damage
+ type - armor amount ) for each of the 3 damage types.
+- Row V "Crit" is the average expected critical damage past armor. This is the sum of the critical
+ damage - armor amount for each of the three damage amounts, but the formula is substantially more
+ complicated because the 3 damage types have different critical damage multipliers and reduce
+ effective armor by different amounts on a critical.
+- Rows W-Y repeat damage and critical damage, but for rapid strikes where the base damage is
+ multiplied by 2/3rds.
+- Row Z "Dmg/Turn" is either: ** 100 * ( Dmg * ( Hits - num crits ) + Crit Dmg * ( num hits ) ) / (
+ 1000 * Moves ) - ie, sum of damage per hit for each normal hit + crit damage for each critical hit
+ divided by the number of moves in 1000 attacks, multiplied by 100 moves/second, OR ** 100 * (
+ Dmg * ( Hits - num crits ) / 2 + Crit Dmg * ( num hits ) / 2 + rapid strike Dmg * ( Hits - num
+ crits ) / 2 + rapid strike Crit Dmg * ( num hits ) / 2 ) / ( 1000 - ( hits / 2 ) * Moves + Hits *
+ 0.33 * Moves ) - same as above, but accounting for rapid strike reduced damage and movement cost.
Finally,
-* Column AT "Weapon" is a repeat of the weapon name for refernce
-* Column Au "Value" is the average of columns Z, AI, and AR, multiplied by 1.5 for Reach 2 weapons and 1.75 for Reach 3 weapons.
+
+- Column AT "Weapon" is a repeat of the weapon name for refernce
+- Column Au "Value" is the average of columns Z, AI, and AR, multiplied by 1.5 for Reach 2 weapons
+ and 1.75 for Reach 3 weapons.
### New Formula Sorted
-This compares the old weapon values versus the values from the new formula, and sorts weapons by decreasing new value by weapon category to make it easier to spot weapons that are unsually good (I'm looking at you, broadsword) or bad for their category.
+
+This compares the old weapon values versus the values from the new formula, and sorts weapons by
+decreasing new value by weapon category to make it easier to spot weapons that are unsually good
+(I'm looking at you, broadsword) or bad for their category.
### Proposed Values
-This repeats the New Formula tab, except that Columns C-J were copied over and pasted as values, and then adjusted to make the numbers nice.
-* Accuracy was adjusted to the new values based on grip, length, striking surface, and balance from the Raw tab. That means a bunch of previously accurate weapons like bionic claws got a nerf, and inappropriately inaccurate weapons like katanas got a buff.
-* Weapons in the same category were generally adjusted to have roughly the same evaluation, though with different values. In general, European weapons get more of their damage from bash and less from cut, and are often heavier and slower, than Asian weapons. This is a little arbitrary and not entirely realistic, but everyone seems to expect it. As a case in point, the Japanese two-handed sword "nodachi" has 164 attack speed, bash 6, and cut 47, while the European two-handed sword zweihander has 169 attack speed, base 18, cut 39. They both have values of about 26.
-* Inferior weapons got a 1 step penalty in balance compared to the real weapon, and half of the cut/pierce damage is moved to bash, and the cut/pierce damage is 1/4th's the real weapons. Ie, an inferior pike does Bash 30, Stab 11 to a pike's Bash 8, Stab 44.
-* Fake weapons are as badly balanced as the inferior version of a weapon, but are basically blunt so the striking surface usually got upgraded to "every", ironically making them as accurate or more accurate than the real version of the weapon. Bash damage was halved from the inferior version, and cut/pierce damage was again reduced to 1/4th of the inferior version, which is 1/16th the real version. A fake pike does Bash 15, Stab 3.
-* A lot of weapons got damage boosts to bring up their evaluated value. A lot of low damage weapons with rapid strike were being drastically overvalued, but the damage past armor tests showed that rapid strike often just lets you bounce off armor twice as fast.
-* Spears got some rough formula for pierce damage based on weight that I can't recover anymore, but in general the differences between spears are more minor than they used to be.
-* Polearms got the same rough damage as Ax2 at range 2, but got a separate balance line at range 1 with the raw damage reduced by 0.7. This makes polearms very impressive at range, but slightly worse than quarterstaffs against adjacent targets.
+This repeats the New Formula tab, except that Columns C-J were copied over and pasted as values, and
+then adjusted to make the numbers nice.
+
+- Accuracy was adjusted to the new values based on grip, length, striking surface, and balance from
+ the Raw tab. That means a bunch of previously accurate weapons like bionic claws got a nerf, and
+ inappropriately inaccurate weapons like katanas got a buff.
+- Weapons in the same category were generally adjusted to have roughly the same evaluation, though
+ with different values. In general, European weapons get more of their damage from bash and less
+ from cut, and are often heavier and slower, than Asian weapons. This is a little arbitrary and not
+ entirely realistic, but everyone seems to expect it. As a case in point, the Japanese two-handed
+ sword "nodachi" has 164 attack speed, bash 6, and cut 47, while the European two-handed sword
+ zweihander has 169 attack speed, base 18, cut 39. They both have values of about 26.
+- Inferior weapons got a 1 step penalty in balance compared to the real weapon, and half of the
+ cut/pierce damage is moved to bash, and the cut/pierce damage is 1/4th's the real weapons. Ie, an
+ inferior pike does Bash 30, Stab 11 to a pike's Bash 8, Stab 44.
+- Fake weapons are as badly balanced as the inferior version of a weapon, but are basically blunt so
+ the striking surface usually got upgraded to "every", ironically making them as accurate or more
+ accurate than the real version of the weapon. Bash damage was halved from the inferior version,
+ and cut/pierce damage was again reduced to 1/4th of the inferior version, which is 1/16th the real
+ version. A fake pike does Bash 15, Stab 3.
+- A lot of weapons got damage boosts to bring up their evaluated value. A lot of low damage weapons
+ with rapid strike were being drastically overvalued, but the damage past armor tests showed that
+ rapid strike often just lets you bounce off armor twice as fast.
+- Spears got some rough formula for pierce damage based on weight that I can't recover anymore, but
+ in general the differences between spears are more minor than they used to be.
+- Polearms got the same rough damage as Ax2 at range 2, but got a separate balance line at range 1
+ with the raw damage reduced by 0.7. This makes polearms very impressive at range, but slightly
+ worse than quarterstaffs against adjacent targets.
### Poposed Values Sorted
+
This is another comparison tab like New Formula Sorted, but used the data from Proposed Values.
### Comparison
+
This is a summary tab.
-In columns A-P, each weapon from the filter tab gets it's current and proposed stats, it's original value, value under the new formula, and value under the new formula using the proposed values. For proposed values, values that improved are highlighted in green and those that got worse are in red. Ideally, this will make it easy to compare changes and catch mistakes.
+In columns A-P, each weapon from the filter tab gets it's current and proposed stats, it's original
+value, value under the new formula, and value under the new formula using the proposed values. For
+proposed values, values that improved are highlighted in green and those that got worse are in red.
+Ideally, this will make it easy to compare changes and catch mistakes.
-Columns S-AH repeat the process, but cells S1, T1, U1, and V1 can be used to specify category names, and then the subtotal shows a selected subset of weapons that match those categories. This simplifies comparing things.
+Columns S-AH repeat the process, but cells S1, T1, U1, and V1 can be used to specify category names,
+and then the subtotal shows a selected subset of weapons that match those categories. This
+simplifies comparing things.