mirror of
https://github.com/vcmi/vcmi.git
synced 2025-01-26 03:52:01 +02:00
Merge pull request #5011 from IvanSavenko/markdown_lint
Added markdownlint to Github CI
This commit is contained in:
commit
38e0b08b34
6
.github/ISSUE_TEMPLATE/bug_report.md
vendored
6
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@ -15,6 +15,7 @@ Please attach game logs: `VCMI_client.txt`, `VCMI_server.txt` etc.
|
|||||||
|
|
||||||
**To Reproduce**
|
**To Reproduce**
|
||||||
Steps to reproduce the behavior:
|
Steps to reproduce the behavior:
|
||||||
|
|
||||||
1. Go to '...'
|
1. Go to '...'
|
||||||
2. Click on '....'
|
2. Click on '....'
|
||||||
3. Scroll down to '....'
|
3. Scroll down to '....'
|
||||||
@ -33,8 +34,9 @@ If this something which worked well some time ago, please let us know about vers
|
|||||||
If applicable, add screenshots to help explain your problem.
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
**Version**
|
**Version**
|
||||||
- OS: [e.g. Windows, macOS Intel, macOS ARM, Android, Linux, iOS]
|
|
||||||
- Version: [VCMI version]
|
- OS: [e.g. Windows, macOS Intel, macOS ARM, Android, Linux, iOS]
|
||||||
|
- Version: [VCMI version]
|
||||||
|
|
||||||
**Additional context**
|
**Additional context**
|
||||||
Add any other context about the problem here.
|
Add any other context about the problem here.
|
||||||
|
6
.github/workflows/github.yml
vendored
6
.github/workflows/github.yml
vendored
@ -402,3 +402,9 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
sudo apt install python3-jstyleson
|
sudo apt install python3-jstyleson
|
||||||
python3 CI/validate_json.py
|
python3 CI/validate_json.py
|
||||||
|
|
||||||
|
- name: Validate Markdown
|
||||||
|
uses: DavidAnson/markdownlint-cli2-action@v18
|
||||||
|
with:
|
||||||
|
config: 'CI/example.markdownlint-cli2.jsonc'
|
||||||
|
globs: '**/*.md'
|
||||||
|
280
CI/example.markdownlint-cli2.jsonc
Normal file
280
CI/example.markdownlint-cli2.jsonc
Normal file
@ -0,0 +1,280 @@
|
|||||||
|
{
|
||||||
|
"config" : {
|
||||||
|
"default" : true,
|
||||||
|
|
||||||
|
// MD001/heading-increment : Heading levels should only increment by one level at a time : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md001.md
|
||||||
|
"MD001": false,
|
||||||
|
|
||||||
|
// MD003/heading-style : Heading style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md003.md
|
||||||
|
"MD003": {
|
||||||
|
"style": "atx"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD004/ul-style : Unordered list style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md004.md
|
||||||
|
"MD004": false,
|
||||||
|
// FIXME: enable and consider fixing
|
||||||
|
//{
|
||||||
|
// "style": "consistent"
|
||||||
|
//},
|
||||||
|
|
||||||
|
// MD005/list-indent : Inconsistent indentation for list items at the same level : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md005.md
|
||||||
|
"MD005": true,
|
||||||
|
|
||||||
|
// MD007/ul-indent : Unordered list indentation : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md007.md
|
||||||
|
"MD007": {
|
||||||
|
// Spaces for indent
|
||||||
|
"indent": 2,
|
||||||
|
// Whether to indent the first level of the list
|
||||||
|
"start_indented": false,
|
||||||
|
// Spaces for first level indent (when start_indented is set)
|
||||||
|
"start_indent": 0
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD009/no-trailing-spaces : Trailing spaces : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md009.md
|
||||||
|
"MD009": {
|
||||||
|
// Spaces for line break
|
||||||
|
"br_spaces": 2,
|
||||||
|
// Allow spaces for empty lines in list items
|
||||||
|
"list_item_empty_lines": false,
|
||||||
|
// Include unnecessary breaks
|
||||||
|
"strict": false
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD010/no-hard-tabs : Hard tabs : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md010.md
|
||||||
|
"MD010": {
|
||||||
|
// Include code blocks
|
||||||
|
"code_blocks": false,
|
||||||
|
// Fenced code languages to ignore
|
||||||
|
"ignore_code_languages": [],
|
||||||
|
// Number of spaces for each hard tab
|
||||||
|
"spaces_per_tab": 4
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD011/no-reversed-links : Reversed link syntax : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md011.md
|
||||||
|
"MD011": true,
|
||||||
|
|
||||||
|
// MD012/no-multiple-blanks : Multiple consecutive blank lines : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md012.md
|
||||||
|
"MD012": {
|
||||||
|
// Consecutive blank lines
|
||||||
|
"maximum": 1
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD013/line-length : Line length : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md013.md
|
||||||
|
"MD013": false,
|
||||||
|
|
||||||
|
// MD014/commands-show-output : Dollar signs used before commands without showing output : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md014.md
|
||||||
|
"MD014": true,
|
||||||
|
|
||||||
|
// MD018/no-missing-space-atx : No space after hash on atx style heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md018.md
|
||||||
|
"MD018": true,
|
||||||
|
|
||||||
|
// MD019/no-multiple-space-atx : Multiple spaces after hash on atx style heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md019.md
|
||||||
|
"MD019": true,
|
||||||
|
|
||||||
|
// MD020/no-missing-space-closed-atx : No space inside hashes on closed atx style heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md020.md
|
||||||
|
"MD020": true,
|
||||||
|
|
||||||
|
// MD021/no-multiple-space-closed-atx : Multiple spaces inside hashes on closed atx style heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md021.md
|
||||||
|
"MD021": true,
|
||||||
|
|
||||||
|
// MD022/blanks-around-headings : Headings should be surrounded by blank lines : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md022.md
|
||||||
|
"MD022": {
|
||||||
|
// Blank lines above heading
|
||||||
|
"lines_above": 1,
|
||||||
|
// Blank lines below heading
|
||||||
|
"lines_below": 1
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD023/heading-start-left : Headings must start at the beginning of the line : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md023.md
|
||||||
|
"MD023": true,
|
||||||
|
|
||||||
|
// MD024/no-duplicate-heading : Multiple headings with the same content : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md024.md
|
||||||
|
"MD024": false,
|
||||||
|
// FIXME: false positives?
|
||||||
|
//{
|
||||||
|
// // Only check sibling headings
|
||||||
|
// "allow_different_nesting": true,
|
||||||
|
// // Only check sibling headings
|
||||||
|
// "siblings_only": true
|
||||||
|
//},
|
||||||
|
|
||||||
|
// MD025/single-title/single-h1 : Multiple top-level headings in the same document : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md025.md
|
||||||
|
"MD025": {
|
||||||
|
// Heading level
|
||||||
|
"level": 1,
|
||||||
|
// RegExp for matching title in front matter
|
||||||
|
"front_matter_title": "^\\s*title\\s*[:=]"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD026/no-trailing-punctuation : Trailing punctuation in heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md026.md
|
||||||
|
"MD026": {
|
||||||
|
// Punctuation characters
|
||||||
|
"punctuation": ".,;:!。,;:!"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD027/no-multiple-space-blockquote : Multiple spaces after blockquote symbol : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md027.md
|
||||||
|
"MD027": true,
|
||||||
|
|
||||||
|
// MD028/no-blanks-blockquote : Blank line inside blockquote : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md028.md
|
||||||
|
"MD028": true,
|
||||||
|
|
||||||
|
// MD029/ol-prefix : Ordered list item prefix : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md029.md
|
||||||
|
"MD029": false,
|
||||||
|
// FIXME: false positives or broken formatting
|
||||||
|
//{
|
||||||
|
// // List style
|
||||||
|
// "style": "ordered"
|
||||||
|
//},
|
||||||
|
|
||||||
|
// MD030/list-marker-space : Spaces after list markers : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md030.md
|
||||||
|
"MD030": {
|
||||||
|
// Spaces for single-line unordered list items
|
||||||
|
"ul_single": 1,
|
||||||
|
// Spaces for single-line ordered list items
|
||||||
|
"ol_single": 1,
|
||||||
|
// Spaces for multi-line unordered list items
|
||||||
|
"ul_multi": 1,
|
||||||
|
// Spaces for multi-line ordered list items
|
||||||
|
"ol_multi": 1
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD031/blanks-around-fences : Fenced code blocks should be surrounded by blank lines : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md031.md
|
||||||
|
"MD031": {
|
||||||
|
// Include list items
|
||||||
|
"list_items": false
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD032/blanks-around-lists : Lists should be surrounded by blank lines : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md032.md
|
||||||
|
"MD032": true,
|
||||||
|
|
||||||
|
// MD033/no-inline-html : Inline HTML : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md033.md
|
||||||
|
"MD033": false,
|
||||||
|
// FIXME: enable and consider fixing
|
||||||
|
//{
|
||||||
|
// // Allowed elements
|
||||||
|
// "allowed_elements": []
|
||||||
|
//},
|
||||||
|
|
||||||
|
// MD034/no-bare-urls : Bare URL used : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md034.md
|
||||||
|
"MD034": true,
|
||||||
|
|
||||||
|
// MD035/hr-style : Horizontal rule style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md035.md
|
||||||
|
"MD035": {
|
||||||
|
// Horizontal rule style
|
||||||
|
"style": "consistent"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD036/no-emphasis-as-heading : Emphasis used instead of a heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md036.md
|
||||||
|
"MD036": false,
|
||||||
|
// FIXME: enable and consider fixing
|
||||||
|
// {
|
||||||
|
// // Punctuation characters
|
||||||
|
// "punctuation": ".,;:!?。,;:!?"
|
||||||
|
// },
|
||||||
|
|
||||||
|
// MD037/no-space-in-emphasis : Spaces inside emphasis markers : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md037.md
|
||||||
|
"MD037": true,
|
||||||
|
|
||||||
|
// MD038/no-space-in-code : Spaces inside code span elements : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md038.md
|
||||||
|
"MD038": true,
|
||||||
|
|
||||||
|
// MD039/no-space-in-links : Spaces inside link text : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md039.md
|
||||||
|
"MD039": true,
|
||||||
|
|
||||||
|
// MD040/fenced-code-language : Fenced code blocks should have a language specified : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md040.md
|
||||||
|
"MD040": false,
|
||||||
|
// FIXME: enable and consider fixing
|
||||||
|
//{
|
||||||
|
//// List of languages
|
||||||
|
// "allowed_languages": [ "cpp", "json5", "sh" ],
|
||||||
|
//// Require language only
|
||||||
|
// "language_only": true
|
||||||
|
//},
|
||||||
|
|
||||||
|
// MD041/first-line-heading/first-line-h1 : First line in a file should be a top-level heading : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md041.md
|
||||||
|
"MD041": {
|
||||||
|
// Heading level
|
||||||
|
"level": 1,
|
||||||
|
// RegExp for matching title in front matter
|
||||||
|
"front_matter_title": "^\\s*title\\s*[:=]"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD042/no-empty-links : No empty links : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md042.md
|
||||||
|
"MD042": true,
|
||||||
|
|
||||||
|
// MD043/required-headings : Required heading structure : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md043.md
|
||||||
|
"MD043": false,
|
||||||
|
|
||||||
|
// MD044/proper-names : Proper names should have the correct capitalization : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md044.md
|
||||||
|
"MD044": false,
|
||||||
|
|
||||||
|
// MD045/no-alt-text : Images should have alternate text (alt text) : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md045.md
|
||||||
|
"MD045": false,
|
||||||
|
|
||||||
|
// MD046/code-block-style : Code block style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md046.md
|
||||||
|
"MD046": {
|
||||||
|
// Block style
|
||||||
|
"style": "fenced"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD047/single-trailing-newline : Files should end with a single newline character : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md047.md
|
||||||
|
"MD047": true,
|
||||||
|
|
||||||
|
// MD048/code-fence-style : Code fence style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md048.md
|
||||||
|
"MD048": {
|
||||||
|
// Code fence style
|
||||||
|
"style": "backtick"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD049/emphasis-style : Emphasis style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md049.md
|
||||||
|
"MD049": {
|
||||||
|
// Emphasis style
|
||||||
|
"style": "asterisk"
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD050/strong-style : Strong style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md050.md
|
||||||
|
"MD050": {
|
||||||
|
// Strong style
|
||||||
|
"style": "asterisk"
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// MD051/link-fragments : Link fragments should be valid : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md051.md
|
||||||
|
"MD051": true,
|
||||||
|
|
||||||
|
// MD052/reference-links-images : Reference links and images should use a label that is defined : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md052.md
|
||||||
|
"MD052": {
|
||||||
|
// Include shortcut syntax
|
||||||
|
"shortcut_syntax": false
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD053/link-image-reference-definitions : Link and image reference definitions should be needed : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md053.md
|
||||||
|
"MD053": {
|
||||||
|
// Ignored definitions
|
||||||
|
"ignored_definitions": [
|
||||||
|
"//"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD054/link-image-style : Link and image style : https://github.com/DavidAnson/markdownlint/blob/v0.32.1/doc/md054.md
|
||||||
|
"MD054": {
|
||||||
|
// Allow autolinks
|
||||||
|
"autolink": true,
|
||||||
|
// Allow inline links and images
|
||||||
|
"inline": true,
|
||||||
|
// Allow full reference links and images
|
||||||
|
"full": true,
|
||||||
|
// Allow collapsed reference links and images
|
||||||
|
"collapsed": true,
|
||||||
|
// Allow shortcut reference links and images
|
||||||
|
"shortcut": true,
|
||||||
|
// Allow URLs as inline links
|
||||||
|
"url_inline": true
|
||||||
|
},
|
||||||
|
|
||||||
|
// MD058 - Tables should be surrounded by blank lines
|
||||||
|
"MD058" : true
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
844
ChangeLog.md
844
ChangeLog.md
File diff suppressed because it is too large
Load Diff
@ -1,11 +1,11 @@
|
|||||||
|
# VCMI Project
|
||||||
|
|
||||||
[![VCMI](https://github.com/vcmi/vcmi/actions/workflows/github.yml/badge.svg?branch=develop&event=push)](https://github.com/vcmi/vcmi/actions/workflows/github.yml?query=branch%3Adevelop+event%3Apush)
|
[![VCMI](https://github.com/vcmi/vcmi/actions/workflows/github.yml/badge.svg?branch=develop&event=push)](https://github.com/vcmi/vcmi/actions/workflows/github.yml?query=branch%3Adevelop+event%3Apush)
|
||||||
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/1.5.0/total)](https://github.com/vcmi/vcmi/releases/tag/1.5.0)
|
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/1.5.0/total)](https://github.com/vcmi/vcmi/releases/tag/1.5.0)
|
||||||
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/1.5.6/total)](https://github.com/vcmi/vcmi/releases/tag/1.5.6)
|
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/1.5.6/total)](https://github.com/vcmi/vcmi/releases/tag/1.5.6)
|
||||||
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/1.5.7/total)](https://github.com/vcmi/vcmi/releases/tag/1.5.7)
|
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/1.5.7/total)](https://github.com/vcmi/vcmi/releases/tag/1.5.7)
|
||||||
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/total)](https://github.com/vcmi/vcmi/releases)
|
[![Github Downloads](https://img.shields.io/github/downloads/vcmi/vcmi/total)](https://github.com/vcmi/vcmi/releases)
|
||||||
|
|
||||||
# VCMI Project
|
|
||||||
|
|
||||||
VCMI is an open-source recreation of Heroes of Might & Magic III engine, giving it new and extended possibilities.
|
VCMI is an open-source recreation of Heroes of Might & Magic III engine, giving it new and extended possibilities.
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
@ -15,14 +15,13 @@ VCMI is an open-source recreation of Heroes of Might & Magic III engine, giving
|
|||||||
<img src="https://github.com/vcmi/VCMI.eu/blob/master/static/img/screenshots/1.4.0/Quick%20Hero%20Select%20Bastion.jpg?raw=true" alt="New widget for Hero selection, featuring Pavillon Town" style="height:120px;"/>
|
<img src="https://github.com/vcmi/VCMI.eu/blob/master/static/img/screenshots/1.4.0/Quick%20Hero%20Select%20Bastion.jpg?raw=true" alt="New widget for Hero selection, featuring Pavillon Town" style="height:120px;"/>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
|
||||||
## Links
|
## Links
|
||||||
|
|
||||||
* Homepage: https://vcmi.eu/
|
* Homepage: <https://vcmi.eu/>
|
||||||
* Forums: https://forum.vcmi.eu/
|
* Forums: <https://forum.vcmi.eu/>
|
||||||
* Bugtracker: https://github.com/vcmi/vcmi/issues
|
* Bugtracker: <https://github.com/vcmi/vcmi/issues>
|
||||||
* Discord: https://discord.gg/chBT42V
|
* Discord: <https://discord.gg/chBT42V>
|
||||||
* GPT Store: https://chat.openai.com/g/g-1kNhX0mlO-vcmi-assistant
|
* GPT Store: <https://chat.openai.com/g/g-1kNhX0mlO-vcmi-assistant>
|
||||||
|
|
||||||
## Latest release
|
## Latest release
|
||||||
|
|
||||||
@ -31,6 +30,7 @@ Loading saves made with different major version of VCMI is usually **not** suppo
|
|||||||
Please see corresponding installation guide articles for details for your platform.
|
Please see corresponding installation guide articles for details for your platform.
|
||||||
|
|
||||||
## Installation guides
|
## Installation guides
|
||||||
|
|
||||||
- [Windows](players/Installation_Windows.md)
|
- [Windows](players/Installation_Windows.md)
|
||||||
- [macOS](players/Installation_macOS.md)
|
- [macOS](players/Installation_macOS.md)
|
||||||
- [Linux](players/Installation_Linux.md)
|
- [Linux](players/Installation_Linux.md)
|
||||||
@ -70,6 +70,7 @@ See also installation guide for [Heroes Chronicles](players/Heroes_Chronicles.md
|
|||||||
## Documentation and guidelines for developers
|
## Documentation and guidelines for developers
|
||||||
|
|
||||||
Development environment setup instructions:
|
Development environment setup instructions:
|
||||||
|
|
||||||
- [Building VCMI for Android](developers/Building_Android.md)
|
- [Building VCMI for Android](developers/Building_Android.md)
|
||||||
- [Building VCMI for iOS](developers/Building_iOS.md)
|
- [Building VCMI for iOS](developers/Building_iOS.md)
|
||||||
- [Building VCMI for Linux](developers/Building_Linux.md)
|
- [Building VCMI for Linux](developers/Building_Linux.md)
|
||||||
@ -78,6 +79,7 @@ Development environment setup instructions:
|
|||||||
- [Conan](developers/Conan.md)
|
- [Conan](developers/Conan.md)
|
||||||
|
|
||||||
Engine documentation: (NOTE: may be outdated)
|
Engine documentation: (NOTE: may be outdated)
|
||||||
|
|
||||||
- [Development with Qt Creator](developers/Development_with_Qt_Creator.md)
|
- [Development with Qt Creator](developers/Development_with_Qt_Creator.md)
|
||||||
- [Coding Guidelines](developers/Coding_Guidelines.md)
|
- [Coding Guidelines](developers/Coding_Guidelines.md)
|
||||||
- [Bonus System](developers/Bonus_System.md)
|
- [Bonus System](developers/Bonus_System.md)
|
||||||
@ -95,6 +97,6 @@ Engine documentation: (NOTE: may be outdated)
|
|||||||
## Copyright and license
|
## Copyright and license
|
||||||
|
|
||||||
VCMI Project source code is licensed under GPL version 2 or later.
|
VCMI Project source code is licensed under GPL version 2 or later.
|
||||||
VCMI Project assets are licensed under CC-BY-SA 4.0. Assets sources and information about contributors are available under following link: https://github.com/vcmi/vcmi-assets
|
VCMI Project assets are licensed under CC-BY-SA 4.0. Assets sources and information about contributors are available under following link: <https://github.com/vcmi/vcmi-assets>
|
||||||
|
|
||||||
Copyright (C) 2007-2024 VCMI Team (check AUTHORS file for the contributors list)
|
Copyright (C) 2007-2024 VCMI Team (check AUTHORS file for the contributors list)
|
||||||
|
@ -6,16 +6,17 @@ There are two types of AI: adventure and battle.
|
|||||||
**Battle AIs** are responsible for fighting, i.e. moving stacks on the battlefield
|
**Battle AIs** are responsible for fighting, i.e. moving stacks on the battlefield
|
||||||
|
|
||||||
We have 3 battle AIs so far:
|
We have 3 battle AIs so far:
|
||||||
|
|
||||||
* BattleAI - strongest
|
* BattleAI - strongest
|
||||||
* StupidAI - for neutrals, should be simple so that experienced players can abuse it
|
* StupidAI - for neutrals, should be simple so that experienced players can abuse it
|
||||||
* Empty AI - should do nothing at all. If needed another battle AI can be introduced.
|
* Empty AI - should do nothing at all. If needed another battle AI can be introduced.
|
||||||
|
|
||||||
Each battle AI consist of a few classes, but the main class, kind of entry point usually has the same name as the package itself. In BattleAI it is the BattleAI class. It implements some battle specific interface, do not remember. Main method there is activeStack(battle::Unit* stack). It is invoked by the system when it's time to move your stack. The thing you use to interact with the game and receive the gamestate is usually referenced in the code as cb. CPlayerSpecificCallback it should be. It has a lot of methods and can do anything. For instance it has battleGetUnitsIf(), which returns all units on the battlefield matching some lambda condition.
|
Each battle AI consist of a few classes, but the main class, kind of entry point usually has the same name as the package itself. In BattleAI it is the BattleAI class. It implements some battle specific interface, do not remember. Main method there is `activeStack(battle::Unit * stack)`. It is invoked by the system when it's time to move your stack. The thing you use to interact with the game and receive the gamestate is usually referenced in the code as `cb`. `CPlayerSpecificCallback` it should be. It has a lot of methods and can do anything. For instance it has battleGetUnitsIf(), which returns all units on the battlefield matching some lambda condition.
|
||||||
Each side in a battle is represented by an CArmedInstance object. CHeroInstance and CGDwelling, CGMonster and more are subclasses of CArmedInstance. CArmedInstance contains a set of stacks. When the battle starts, these stacks are converted to battle stacks. Usually Battle AIs reference them using the interface battle::Unit *.
|
Each side in a battle is represented by an `CArmedInstance` object. `CHeroInstance` and `CGDwelling`, `CGMonster` and more are subclasses of `CArmedInstance`. `CArmedInstance` contains a set of stacks. When the battle starts, these stacks are converted to battle stacks. Usually Battle AIs reference them using the interface `battle::Unit *`.
|
||||||
Units have bonuses. Nearly everything aspect of a unit is configured in the form of bonuses. Attack, defense, health, retaliation, shooter or not, initial count of shots and so on.
|
Units have bonuses. Nearly everything aspect of a unit is configured in the form of bonuses. Attack, defense, health, retaliation, shooter or not, initial count of shots and so on.
|
||||||
When you call unit->getAttack() it summarizes all these bonuses and returns the resulting value.
|
When you call `unit->getAttack()` it summarizes all these bonuses and returns the resulting value.
|
||||||
|
|
||||||
One important class is HypotheticBattle. It is used to evaluate the effects of an action without changing the actual gamestate. It is a wrapper around CPlayerSpecificCallback or another HypotheticBattle so it can provide you data, Internally it has a set of modified unit states and intercepts some calls to underlying callback and returns these internal states instead. These states in turn are wrappers around original units and contain modified bonuses (CStackWithBonuses). So if you need to emulate an attack you can call hypotheticbattle.getforupdate() and it will return the CStackWithBonuses which you can safely change.
|
One important class is `HypotheticBattle`. It is used to evaluate the effects of an action without changing the actual gamestate. It is a wrapper around `CPlayerSpecificCallback` or another `HypotheticBattle` so it can provide you data, Internally it has a set of modified unit states and intercepts some calls to underlying callback and returns these internal states instead. These states in turn are wrappers around original units and contain modified bonuses (`CStackWithBonuses`). So if you need to emulate an attack you can call `hypotheticbattle.getforupdate()` and it will return the `CStackWithBonuses` which you can safely change.
|
||||||
|
|
||||||
## BattleAI
|
## BattleAI
|
||||||
|
|
||||||
@ -38,17 +39,20 @@ BattleAI itself handles all the rest and issues actual commands
|
|||||||
Adventure AI responsible for moving heroes on map, gathering things, developing town. Main idea is to gather all possible tasks on map, prioritize them and select the best one for each heroes. Initially was a fork of VCAI
|
Adventure AI responsible for moving heroes on map, gathering things, developing town. Main idea is to gather all possible tasks on map, prioritize them and select the best one for each heroes. Initially was a fork of VCAI
|
||||||
|
|
||||||
### Parts
|
### Parts
|
||||||
|
|
||||||
Gateway - a callback for server used to invoke AI actions when server thinks it is time to do something. Through this callback AI is informed about various events like hero level up, tile revialed, blocking dialogs and so on. In order to do this Gaateway implements specific interface. The interface is exactly the same for human and AI
|
Gateway - a callback for server used to invoke AI actions when server thinks it is time to do something. Through this callback AI is informed about various events like hero level up, tile revialed, blocking dialogs and so on. In order to do this Gaateway implements specific interface. The interface is exactly the same for human and AI
|
||||||
Another important actor for server interaction is CCallback * cb. This one is used to retrieve gamestate information and ask server to do things like hero moving, spell casting and so on. Each AI has own instance of Gateway and it is a root object which holds all AI state. Gateway has an event method yourTurn which invokes makeTurn in another thread. The last passes control to Nullkiller engine.
|
Another important actor for server interaction is CCallback * cb. This one is used to retrieve gamestate information and ask server to do things like hero moving, spell casting and so on. Each AI has own instance of Gateway and it is a root object which holds all AI state. Gateway has an event method yourTurn which invokes makeTurn in another thread. The last passes control to Nullkiller engine.
|
||||||
|
|
||||||
Nullkiller engine - place where actual AI logic is organized. It contains a main loop for gathering and prioritizing things. Its algorithm:
|
Nullkiller engine - place where actual AI logic is organized. It contains a main loop for gathering and prioritizing things. Its algorithm:
|
||||||
|
|
||||||
* reset AI state, it avoids keeping some memory about the game in general to reduce amount of things serialized into savefile state. The only serialized things are in nullkiller->memory. This helps reducing save incompatibility. It should be mostly enough for AI to analyze data avaialble in CCallback
|
* reset AI state, it avoids keeping some memory about the game in general to reduce amount of things serialized into savefile state. The only serialized things are in nullkiller->memory. This helps reducing save incompatibility. It should be mostly enough for AI to analyze data avaialble in CCallback
|
||||||
* main loop, loop iteration is called a pass
|
* main loop, loop iteration is called a pass
|
||||||
** update AI state, some state is lazy and updates once per day to avoid performance hit, some state is recalculated each loop iteration. At this stage analysers and pathfidner work
|
* update AI state, some state is lazy and updates once per day to avoid performance hit, some state is recalculated each loop iteration. At this stage analysers and pathfidner work
|
||||||
** gathering goals, prioritizing and decomposing them
|
* gathering goals, prioritizing and decomposing them
|
||||||
** execute selected best goals
|
* execute selected best goals
|
||||||
|
|
||||||
Analyzer - a module gathering data from CCallback *. Its goal to make some statistics and avoid making any significant decissions.
|
Analyzer - a module gathering data from CCallback *. Its goal to make some statistics and avoid making any significant decissions.
|
||||||
|
|
||||||
* HeroAnalyser - decides upong which hero suits better to be main (army carrier and fighter) and which is better to be a scout (gathering unguarded resources, exploring)
|
* HeroAnalyser - decides upong which hero suits better to be main (army carrier and fighter) and which is better to be a scout (gathering unguarded resources, exploring)
|
||||||
* BuildAnalyzer - prepares information on what we can build in our towns, and what resources we need to do this
|
* BuildAnalyzer - prepares information on what we can build in our towns, and what resources we need to do this
|
||||||
* DangerHitMapAnalyser - checks if enemy hero can rich each tile, how fast and what is their army strangth
|
* DangerHitMapAnalyser - checks if enemy hero can rich each tile, how fast and what is their army strangth
|
||||||
@ -61,9 +65,11 @@ Analyzer - a module gathering data from CCallback *. Its goal to make some stati
|
|||||||
* PriorityEvaluator - gathers information on task rewards, evaluates their priority using Fuzzy Light library (fuzzy logic)
|
* PriorityEvaluator - gathers information on task rewards, evaluates their priority using Fuzzy Light library (fuzzy logic)
|
||||||
|
|
||||||
### Goals
|
### Goals
|
||||||
|
|
||||||
Units of activity in AI. Can be AbstractGoal, Task, Marker and Behavior
|
Units of activity in AI. Can be AbstractGoal, Task, Marker and Behavior
|
||||||
|
|
||||||
Task - simple thing which can be done right away in order to gain some reward. Or a composition of simple things in case if more than one action is needed to gain the reward.
|
Task - simple thing which can be done right away in order to gain some reward. Or a composition of simple things in case if more than one action is needed to gain the reward.
|
||||||
|
|
||||||
* AdventureSpellCast - town portal, water walk, air walk, summon boat
|
* AdventureSpellCast - town portal, water walk, air walk, summon boat
|
||||||
* BuildBoat - builds a boat in a specific shipyard
|
* BuildBoat - builds a boat in a specific shipyard
|
||||||
* BuildThis - builds a building in a specified town
|
* BuildThis - builds a building in a specified town
|
||||||
@ -78,6 +84,7 @@ Task - simple thing which can be done right away in order to gain some reward. O
|
|||||||
* StayAtTown - stay at town for the rest of the day (to regain mana)
|
* StayAtTown - stay at town for the rest of the day (to regain mana)
|
||||||
|
|
||||||
Behavior - a core game activity
|
Behavior - a core game activity
|
||||||
|
|
||||||
* CaptureObjectsBehavior - generally it is about visiting map objects which give reward. It can capture any object, even those which are behind monsters and so on. But due to performance considerations it is not allowed to handle monsters and quests now.
|
* CaptureObjectsBehavior - generally it is about visiting map objects which give reward. It can capture any object, even those which are behind monsters and so on. But due to performance considerations it is not allowed to handle monsters and quests now.
|
||||||
* ClusterBehavior - uses information of ObjectClusterizer to unblock objects hidden behind various blockers. It kills guards, completes quests, captures garrisons.
|
* ClusterBehavior - uses information of ObjectClusterizer to unblock objects hidden behind various blockers. It kills guards, completes quests, captures garrisons.
|
||||||
* BuildingBehavior - develops our towns
|
* BuildingBehavior - develops our towns
|
||||||
@ -89,6 +96,7 @@ Behavior - a core game activity
|
|||||||
* DefenceBehavior - defend towns by eliminating treatening heroes or hiding in town garrison
|
* DefenceBehavior - defend towns by eliminating treatening heroes or hiding in town garrison
|
||||||
|
|
||||||
AbstractGoal - some goals can not be completed because it is not clear how to do this. They express desire to do something, not exact plan. DeepDecomposer is used to refine such goals until they are turned into such plan or discarded. Some examples:
|
AbstractGoal - some goals can not be completed because it is not clear how to do this. They express desire to do something, not exact plan. DeepDecomposer is used to refine such goals until they are turned into such plan or discarded. Some examples:
|
||||||
|
|
||||||
* CaptureObject - you need to visit some object (flag a shipyard for instance) but do not know how
|
* CaptureObject - you need to visit some object (flag a shipyard for instance) but do not know how
|
||||||
* CompleteQuest - you need to bypass bordergate or borderguard or questguard but do not know how
|
* CompleteQuest - you need to bypass bordergate or borderguard or questguard but do not know how
|
||||||
AbstractGoal usually comes in form of composition with some elementar task blocked by abstract objective. For instance CaptureObject(Shipyard), ExecuteHeroChain(visit x, build boat, visit enemy town). When such composition is decomposed it can turn into either a pair of herochains or into another abstract composition if path to shipyard is also blocked with something.
|
AbstractGoal usually comes in form of composition with some elementar task blocked by abstract objective. For instance CaptureObject(Shipyard), ExecuteHeroChain(visit x, build boat, visit enemy town). When such composition is decomposed it can turn into either a pair of herochains or into another abstract composition if path to shipyard is also blocked with something.
|
||||||
|
@ -26,6 +26,7 @@ There are two basic types of operations that can be performed on the graph:
|
|||||||
### Adding a new node
|
### Adding a new node
|
||||||
|
|
||||||
When node is attached to a new black parent (the only possibility - adding parent is the same as adding a child to it), the propagation system is triggered and works as follows:
|
When node is attached to a new black parent (the only possibility - adding parent is the same as adding a child to it), the propagation system is triggered and works as follows:
|
||||||
|
|
||||||
- For the attached node and its all red ancestors
|
- For the attached node and its all red ancestors
|
||||||
- For every bonus
|
- For every bonus
|
||||||
- Call propagator giving the new descendant - then attach appropriately bonuses to the red descendant of attached node (or the node itself).
|
- Call propagator giving the new descendant - then attach appropriately bonuses to the red descendant of attached node (or the node itself).
|
||||||
@ -54,7 +55,7 @@ Updaters are objects attached to bonuses. They can modify a bonus (typically by
|
|||||||
|
|
||||||
The following example shows an artifact providing a bonus based on the level of the hero that wears it:
|
The following example shows an artifact providing a bonus based on the level of the hero that wears it:
|
||||||
|
|
||||||
```javascript
|
```json5
|
||||||
"core:greaterGnollsFlail":
|
"core:greaterGnollsFlail":
|
||||||
{
|
{
|
||||||
"text" : { "description" : "This mighty flail increases the attack of all gnolls under the hero's command by twice the hero's level." },
|
"text" : { "description" : "This mighty flail increases the attack of all gnolls under the hero's command by twice the hero's level." },
|
||||||
|
@ -1,26 +1,26 @@
|
|||||||
# Building Android
|
# Building Android
|
||||||
|
|
||||||
The following instructions apply to **v1.2 and later**. For earlier versions the best documentation is https://github.com/vcmi/vcmi-android/blob/master/building.txt (and reading scripts in that repo), however very limited to no support will be provided from our side if you wish to go down that rabbit hole.
|
The following instructions apply to **v1.2 and later**. For earlier versions the best documentation is <https://github.com/vcmi/vcmi-android/blob/master/building.txt> (and reading scripts in that repo), however very limited to no support will be provided from our side if you wish to go down that rabbit hole.
|
||||||
|
|
||||||
*Note*: building has been tested only on Linux and macOS. It may or may not work on Windows out of the box.
|
*Note*: building has been tested only on Linux and macOS. It may or may not work on Windows out of the box.
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
1. CMake 3.20+: download from your package manager or from https://cmake.org/download/
|
1. CMake 3.20+: download from your package manager or from <https://cmake.org/download/>
|
||||||
2. JDK 11, not necessarily from Oracle
|
2. JDK 11, not necessarily from Oracle
|
||||||
3. Android command line tools or Android Studio for your OS: https://developer.android.com/studio/
|
3. Android command line tools or Android Studio for your OS: <https://developer.android.com/studio/>
|
||||||
4. Android NDK version **r25c (25.2.9519653)**, there're multiple ways to obtain it:
|
4. Android NDK version **r25c (25.2.9519653)**, there're multiple ways to obtain it:
|
||||||
- install with Android Studio
|
- install with Android Studio
|
||||||
- install with `sdkmanager` command line tool
|
- install with `sdkmanager` command line tool
|
||||||
- download from https://developer.android.com/ndk/downloads
|
- download from <https://developer.android.com/ndk/downloads>
|
||||||
- download with Conan, see [#NDK and Conan](#ndk-and-conan)
|
- download with Conan, see [#NDK and Conan](#ndk-and-conan)
|
||||||
5. Optional:
|
5. Optional:
|
||||||
- Ninja: download from your package manager or from https://github.com/ninja-build/ninja/releases
|
- Ninja: download from your package manager or from <https://github.com/ninja-build/ninja/releases>
|
||||||
- Ccache: download from your package manager or from https://github.com/ccache/ccache/releases
|
- Ccache: download from your package manager or from <https://github.com/ccache/ccache/releases>
|
||||||
|
|
||||||
## Obtaining source code
|
## Obtaining source code
|
||||||
|
|
||||||
Clone https://github.com/vcmi/vcmi with submodules. Example for command line:
|
Clone <https://github.com/vcmi/vcmi> with submodules. Example for command line:
|
||||||
|
|
||||||
```
|
```
|
||||||
git clone --recurse-submodules https://github.com/vcmi/vcmi.git
|
git clone --recurse-submodules https://github.com/vcmi/vcmi.git
|
||||||
@ -31,6 +31,7 @@ git clone --recurse-submodules https://github.com/vcmi/vcmi.git
|
|||||||
We use Conan package manager to build/consume dependencies, find detailed usage instructions [here](./Conan.md). Note that the link points to the state of the current branch, for the latest release check the same document in the [master branch](https://github.com/vcmi/vcmi/blob/master/docs/developers/Сonan.md).
|
We use Conan package manager to build/consume dependencies, find detailed usage instructions [here](./Conan.md). Note that the link points to the state of the current branch, for the latest release check the same document in the [master branch](https://github.com/vcmi/vcmi/blob/master/docs/developers/Сonan.md).
|
||||||
|
|
||||||
On the step where you need to replace **PROFILE**, choose:
|
On the step where you need to replace **PROFILE**, choose:
|
||||||
|
|
||||||
- `android-32` to build for 32-bit architecture (armeabi-v7a)
|
- `android-32` to build for 32-bit architecture (armeabi-v7a)
|
||||||
- `android-64` to build for 64-bit architecture (aarch64-v8a)
|
- `android-64` to build for 64-bit architecture (aarch64-v8a)
|
||||||
|
|
||||||
@ -38,7 +39,7 @@ On the step where you need to replace **PROFILE**, choose:
|
|||||||
|
|
||||||
Conan must be aware of the NDK location when you execute `conan install`. There're multiple ways to achieve that as written in the [Conan docs](https://docs.conan.io/1/integrations/cross_platform/android.html):
|
Conan must be aware of the NDK location when you execute `conan install`. There're multiple ways to achieve that as written in the [Conan docs](https://docs.conan.io/1/integrations/cross_platform/android.html):
|
||||||
|
|
||||||
- the easiest is to download NDK from Conan (option 1 in the docs), then all the magic happens automatically. On the step where you need to replace **PROFILE**, choose _android-**X**-ndk_ where _**X**_ is either `32` or `64`.
|
- the easiest is to download NDK from Conan (option 1 in the docs), then all the magic happens automatically. On the step where you need to replace **PROFILE**, choose *android-**X**-ndk* where ***X*** is either `32` or `64`.
|
||||||
- to use an already installed NDK, you can simply pass it on the command line to `conan install`: (note that this will work only when consuming the pre-built binaries)
|
- to use an already installed NDK, you can simply pass it on the command line to `conan install`: (note that this will work only when consuming the pre-built binaries)
|
||||||
|
|
||||||
```
|
```
|
||||||
|
@ -41,7 +41,7 @@ NOTE: `fuzzylite-devel` package is no longer available in recent version of Fedo
|
|||||||
|
|
||||||
On Arch-based distributions, there is a development package available for VCMI on the AUR.
|
On Arch-based distributions, there is a development package available for VCMI on the AUR.
|
||||||
|
|
||||||
It can be found at https://aur.archlinux.org/packages/vcmi-git/
|
It can be found at <https://aur.archlinux.org/packages/vcmi-git/>
|
||||||
|
|
||||||
Information about building packages from the Arch User Repository (AUR) can be found at the Arch wiki.
|
Information about building packages from the Arch User Repository (AUR) can be found at the Arch wiki.
|
||||||
|
|
||||||
@ -109,9 +109,9 @@ This will generate `vcmiclient`, `vcmiserver`, `vcmilauncher` as well as .so lib
|
|||||||
|
|
||||||
### RPM package
|
### RPM package
|
||||||
|
|
||||||
The first step is to prepare a RPM build environment. On Fedora systems you can follow this guide: http://fedoraproject.org/wiki/How_to_create_an_RPM_package#SPEC_file_overview
|
The first step is to prepare a RPM build environment. On Fedora systems you can follow this guide: <http://fedoraproject.org/wiki/How_to_create_an_RPM_package#SPEC_file_overview>
|
||||||
|
|
||||||
0. Enable RPMFusion free repo to access to ffmpeg libs:
|
1. Enable RPMFusion free repo to access to ffmpeg libs:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo dnf install https://download1.rpmfusion.org/free/fedora/rpmfusion-free-release-$(rpm -E %fedora).noarch.rpm
|
sudo dnf install https://download1.rpmfusion.org/free/fedora/rpmfusion-free-release-$(rpm -E %fedora).noarch.rpm
|
||||||
@ -120,30 +120,31 @@ sudo dnf install https://download1.rpmfusion.org/free/fedora/rpmfusion-free-rele
|
|||||||
> [!NOTE]
|
> [!NOTE]
|
||||||
> The stock ffmpeg from Fedora repo is no good as it lacks a lots of codecs
|
> The stock ffmpeg from Fedora repo is no good as it lacks a lots of codecs
|
||||||
|
|
||||||
1. Perform a git clone from a tagged branch for the right Fedora version from https://github.com/rpmfusion/vcmi; for example for Fedora 38: <pre>git clone -b f38 --single-branch https://github.com/rpmfusion/vcmi.git</pre>
|
2. Perform a git clone from a tagged branch for the right Fedora version from <https://github.com/rpmfusion/vcmi>; for example for Fedora 38: <pre>git clone -b f38 --single-branch https://github.com/rpmfusion/vcmi.git</pre>
|
||||||
|
|
||||||
2. Copy all files to ~/rpmbuild/SPECS with command: <pre>cp vcmi/* ~/rpmbuild/SPECS</pre>
|
3. Copy all files to ~/rpmbuild/SPECS with command: <pre>cp vcmi/* ~/rpmbuild/SPECS</pre>
|
||||||
|
|
||||||
3. Fetch all sources by using spectool:
|
4. Fetch all sources by using spectool:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo dnf install rpmdevtools
|
sudo dnf install rpmdevtools
|
||||||
spectool -g -R ~/rpmbuild/SPECS/vcmi.spec
|
spectool -g -R ~/rpmbuild/SPECS/vcmi.spec
|
||||||
```
|
```
|
||||||
|
|
||||||
4. Fetch all dependencies required to build the RPM:
|
5. Fetch all dependencies required to build the RPM:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo dnf install dnf-plugins-core
|
sudo dnf install dnf-plugins-core
|
||||||
sudo dnf builddep ~/rpmbuild/SPECS/vcmi.spec
|
sudo dnf builddep ~/rpmbuild/SPECS/vcmi.spec
|
||||||
```
|
```
|
||||||
|
|
||||||
4. Go to ~/rpmbuild/SPECS and open terminal in this folder and type:
|
6. Go to ~/rpmbuild/SPECS and open terminal in this folder and type:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
rpmbuild -ba ~/rpmbuild/SPECS/vcmi.spec
|
rpmbuild -ba ~/rpmbuild/SPECS/vcmi.spec
|
||||||
```
|
```
|
||||||
|
|
||||||
5. Generated RPM is in folder ~/rpmbuild/RPMS
|
7. Generated RPM is in folder ~/rpmbuild/RPMS
|
||||||
|
|
||||||
If you want to package the generated RPM above for different processor architectures and operating systems you can use the tool mock.
|
If you want to package the generated RPM above for different processor architectures and operating systems you can use the tool mock.
|
||||||
Moreover, it is necessary to install mock-rpmfusion_free due to the packages ffmpeg-devel and ffmpeg-libs which aren't available in the standard RPM repositories(at least for Fedora). Go to ~/rpmbuild/SRPMS in terminal and type:
|
Moreover, it is necessary to install mock-rpmfusion_free due to the packages ffmpeg-devel and ffmpeg-libs which aren't available in the standard RPM repositories(at least for Fedora). Go to ~/rpmbuild/SRPMS in terminal and type:
|
||||||
|
@ -24,9 +24,11 @@ Warning! Replace `%VCMI_DIR%` with path you've chosen for VCMI installation in t
|
|||||||
It is recommended to avoid non-ascii characters in the path to your working folders. The folder should not be write-protected by system.
|
It is recommended to avoid non-ascii characters in the path to your working folders. The folder should not be write-protected by system.
|
||||||
|
|
||||||
Good locations:
|
Good locations:
|
||||||
|
|
||||||
- `C:\VCMI`
|
- `C:\VCMI`
|
||||||
|
|
||||||
Bad locations:
|
Bad locations:
|
||||||
|
|
||||||
- `C:\Users\Michał\VCMI (non-ascii character)`
|
- `C:\Users\Michał\VCMI (non-ascii character)`
|
||||||
- `C:\Program Files (x86)\VCMI (write protection)`
|
- `C:\Program Files (x86)\VCMI (write protection)`
|
||||||
|
|
||||||
@ -38,13 +40,14 @@ You have two options: to use pre-built libraries or build your own. We strongly
|
|||||||
|
|
||||||
#### Download and unpack archive
|
#### Download and unpack archive
|
||||||
|
|
||||||
Vcpkg Archives are available at our GitHub: https://github.com/vcmi/vcmi-deps-windows/releases
|
Vcpkg Archives are available at our GitHub: <https://github.com/vcmi/vcmi-deps-windows/releases>
|
||||||
|
|
||||||
- Download latest version available.
|
- Download latest version available.
|
||||||
EG: v1.6 assets - [vcpkg-export-x64-windows-v143.7z](https://github.com/vcmi/vcmi-deps-windows/releases/download/v1.6/vcpkg-export-x64-windows-v143.7z)
|
EG: v1.6 assets - [vcpkg-export-x64-windows-v143.7z](https://github.com/vcmi/vcmi-deps-windows/releases/download/v1.6/vcpkg-export-x64-windows-v143.7z)
|
||||||
- Extract archive by right clicking on it and choosing "7-zip -> Extract Here".
|
- Extract archive by right clicking on it and choosing "7-zip -> Extract Here".
|
||||||
|
|
||||||
#### Move dependencies to target directory
|
#### Move dependencies to target directory
|
||||||
|
|
||||||
Once extracted, a `vcpkg` directory will appear with `installed` and `scripts` subfolders inside.
|
Once extracted, a `vcpkg` directory will appear with `installed` and `scripts` subfolders inside.
|
||||||
Move extracted `vcpkg` directory into your `%VCMI_DIR%`
|
Move extracted `vcpkg` directory into your `%VCMI_DIR%`
|
||||||
|
|
||||||
@ -65,7 +68,9 @@ Be aware that building Vcpkg might take a lot of time depend on your CPU model a
|
|||||||
|
|
||||||
From command line use:
|
From command line use:
|
||||||
|
|
||||||
git clone https://github.com/microsoft/vcpkg.git %VCMI_DIR%/vcpkg
|
```sh
|
||||||
|
git clone https://github.com/microsoft/vcpkg.git %VCMI_DIR%/vcpkg
|
||||||
|
```
|
||||||
|
|
||||||
#### Build vcpkg and dependencies
|
#### Build vcpkg and dependencies
|
||||||
|
|
||||||
@ -85,6 +90,7 @@ Extract `ccache` to a folder of your choosing, add the folder to the `PATH` envi
|
|||||||
## Build VCMI
|
## Build VCMI
|
||||||
|
|
||||||
#### From GIT GUI
|
#### From GIT GUI
|
||||||
|
|
||||||
- Open SourceTree
|
- Open SourceTree
|
||||||
- File -> Clone
|
- File -> Clone
|
||||||
- select `https://github.com/vcmi/vcmi/` as source
|
- select `https://github.com/vcmi/vcmi/` as source
|
||||||
@ -94,15 +100,18 @@ Extract `ccache` to a folder of your choosing, add the folder to the `PATH` envi
|
|||||||
- click Clone
|
- click Clone
|
||||||
|
|
||||||
#### From command line
|
#### From command line
|
||||||
|
|
||||||
- `git clone --recursive https://github.com/vcmi/vcmi.git %VCMI_DIR%/source`
|
- `git clone --recursive https://github.com/vcmi/vcmi.git %VCMI_DIR%/source`
|
||||||
|
|
||||||
### Generate solution for VCMI
|
### Generate solution for VCMI
|
||||||
|
|
||||||
- Create `%VCMI_DIR%/build` folder
|
- Create `%VCMI_DIR%/build` folder
|
||||||
- Open a command line prompt at `%VCMI_DIR%/build`
|
- Open a command line prompt at `%VCMI_DIR%/build`
|
||||||
- Execute `cd %VCMI_DIR%/build`
|
- Execute `cd %VCMI_DIR%/build`
|
||||||
- Create solution (Visual Studio 2022 64-bit) `cmake %VCMI_DIR%/source -DCMAKE_TOOLCHAIN_FILE=%VCMI_DIR%/vcpkg/scripts/buildsystems/vcpkg.cmake -G "Visual Studio 17 2022" -A x64`
|
- Create solution (Visual Studio 2022 64-bit) `cmake %VCMI_DIR%/source -DCMAKE_TOOLCHAIN_FILE=%VCMI_DIR%/vcpkg/scripts/buildsystems/vcpkg.cmake -G "Visual Studio 17 2022" -A x64`
|
||||||
|
|
||||||
### Compile VCMI with Visual Studio
|
### Compile VCMI with Visual Studio
|
||||||
|
|
||||||
- Open `%VCMI_DIR%/build/VCMI.sln` in Visual Studio
|
- Open `%VCMI_DIR%/build/VCMI.sln` in Visual Studio
|
||||||
- Select `Release` build type in the combobox
|
- Select `Release` build type in the combobox
|
||||||
- If you want to use ccache:
|
- If you want to use ccache:
|
||||||
@ -113,7 +122,8 @@ Extract `ccache` to a folder of your choosing, add the folder to the `PATH` envi
|
|||||||
- VCMI will be built in `%VCMI_DIR%/build/bin` folder!
|
- VCMI will be built in `%VCMI_DIR%/build/bin` folder!
|
||||||
|
|
||||||
### Compile VCMI with MinGW via MSYS2
|
### Compile VCMI with MinGW via MSYS2
|
||||||
- Install MSYS2 from https://www.msys2.org/
|
|
||||||
|
- Install MSYS2 from <https://www.msys2.org/>
|
||||||
- Start the `MSYS MinGW x64`-shell
|
- Start the `MSYS MinGW x64`-shell
|
||||||
- Install dependencies: `pacman -S mingw-w64-x86_64-SDL2 mingw-w64-x86_64-SDL2_image mingw-w64-x86_64-SDL2_mixer mingw-w64-x86_64-SDL2_ttf mingw-w64-x86_64-boost mingw-w64-x86_64-gcc mingw-w64-x86_64-ninja mingw-w64-x86_64-qt5-static mingw-w64-x86_64-qt5-tools mingw-w64-x86_64-tbb`
|
- Install dependencies: `pacman -S mingw-w64-x86_64-SDL2 mingw-w64-x86_64-SDL2_image mingw-w64-x86_64-SDL2_mixer mingw-w64-x86_64-SDL2_ttf mingw-w64-x86_64-boost mingw-w64-x86_64-gcc mingw-w64-x86_64-ninja mingw-w64-x86_64-qt5-static mingw-w64-x86_64-qt5-tools mingw-w64-x86_64-tbb`
|
||||||
- Generate and build solution from VCMI-root dir: `cmake --preset windows-mingw-release && cmake --build --preset windows-mingw-release`
|
- Generate and build solution from VCMI-root dir: `cmake --preset windows-mingw-release && cmake --build --preset windows-mingw-release`
|
||||||
@ -134,8 +144,10 @@ Vcpkg might be very unstable due to limited popularity and fact of using bleedin
|
|||||||
|
|
||||||
Pre-built version we provide is always manually tested with all supported versions of MSVC for both Release and Debug builds and all known quirks are listed below.
|
Pre-built version we provide is always manually tested with all supported versions of MSVC for both Release and Debug builds and all known quirks are listed below.
|
||||||
|
|
||||||
#$# Build is successful but can not start new game
|
### Build is successful but can not start new game
|
||||||
|
|
||||||
Make sure you have:
|
Make sure you have:
|
||||||
|
|
||||||
* Installed Heroes III from disk or using GOG installer
|
* Installed Heroes III from disk or using GOG installer
|
||||||
* Copied `Data`, `Maps` and `Mp3` folders from Heroes III to: `%USERPROFILE%\Documents\My Games\vcmi\`
|
* Copied `Data`, `Maps` and `Mp3` folders from Heroes III to: `%USERPROFILE%\Documents\My Games\vcmi\`
|
||||||
|
|
||||||
|
@ -91,7 +91,7 @@ Open `VCMI.xcodeproj` from the build directory, select `vcmiclient` scheme and h
|
|||||||
|
|
||||||
## Packaging project into DMG file
|
## Packaging project into DMG file
|
||||||
|
|
||||||
After building, run `cpack` from the build directory. If using Xcode generator, also pass `-C `<configuration name> with the same configuration that you used to build the project.
|
After building, run `cpack` from the build directory. If using Xcode generator, also pass `-C <configuration name>` with the same configuration that you used to build the project.
|
||||||
|
|
||||||
If you use Conan, it's expected that you use **conan-generated** directory at step 4 of [Conan package manager](Conan.md).
|
If you use Conan, it's expected that you use **conan-generated** directory at step 4 of [Conan package manager](Conan.md).
|
||||||
|
|
||||||
|
@ -3,9 +3,7 @@
|
|||||||
* `-D CMAKE_BUILD_TYPE=Debug`
|
* `-D CMAKE_BUILD_TYPE=Debug`
|
||||||
* Enables debug info and disables optimizations
|
* Enables debug info and disables optimizations
|
||||||
* `-D CMAKE_EXPORT_COMPILE_COMMANDS=ON`
|
* `-D CMAKE_EXPORT_COMPILE_COMMANDS=ON`
|
||||||
* Creates `compile_commands.json` for [clangd](https://clangd.llvm.org/) language server.
|
* Creates `compile_commands.json` for [clangd](https://clangd.llvm.org/) language server. For clangd to find the JSON, create a file named `.clangd` with this content
|
||||||
|
|
||||||
For clangd to find the JSON, create a file named `.clangd` with this content
|
|
||||||
```
|
```
|
||||||
CompileFlags:
|
CompileFlags:
|
||||||
CompilationDatabase: build
|
CompilationDatabase: build
|
||||||
|
@ -29,6 +29,7 @@ Most of VCMI configuration files uses Json format and located in "config" direct
|
|||||||
### Main purposes of client
|
### Main purposes of client
|
||||||
|
|
||||||
Client is responsible for:
|
Client is responsible for:
|
||||||
|
|
||||||
- displaying state of game to human player
|
- displaying state of game to human player
|
||||||
- capturing player's actions and sending requests to server
|
- capturing player's actions and sending requests to server
|
||||||
- displaying changes in state of game indicated by server
|
- displaying changes in state of game indicated by server
|
||||||
@ -94,7 +95,6 @@ Forward declarations of the lib in headers of other parts of the project need to
|
|||||||
`<other forward declarations>`
|
`<other forward declarations>`
|
||||||
`<classes>`
|
`<classes>`
|
||||||
|
|
||||||
|
|
||||||
##### New project part
|
##### New project part
|
||||||
|
|
||||||
If you're creating new project part, place `VCMI_LIB_USING_NAMESPACE` in its `StdInc.h` to be able to use lib classes without explicit namespace in implementation files. Example: <https://github.com/vcmi/vcmi/blob/develop/launcher/StdInc.h>
|
If you're creating new project part, place `VCMI_LIB_USING_NAMESPACE` in its `StdInc.h` to be able to use lib classes without explicit namespace in implementation files. Example: <https://github.com/vcmi/vcmi/blob/develop/launcher/StdInc.h>
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
VCMI implementation bases on C++17 standard. Any feature is acceptable as long as it's will pass build on our CI, but there is list below on what is already being used.
|
VCMI implementation bases on C++17 standard. Any feature is acceptable as long as it's will pass build on our CI, but there is list below on what is already being used.
|
||||||
|
|
||||||
Any compiler supporting C++17 should work, but this has not been thoroughly tested. You can find information about extensions and compiler support at http://en.cppreference.com/w/cpp/compiler_support
|
Any compiler supporting C++17 should work, but this has not been thoroughly tested. You can find information about extensions and compiler support at <http://en.cppreference.com/w/cpp/compiler_support>
|
||||||
|
|
||||||
## Style Guidelines
|
## Style Guidelines
|
||||||
|
|
||||||
@ -20,7 +20,7 @@ Inside a code block put the opening brace on the next line after the current sta
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
code();
|
code();
|
||||||
@ -30,7 +30,7 @@ if(a)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a) {
|
if(a) {
|
||||||
code();
|
code();
|
||||||
code();
|
code();
|
||||||
@ -41,14 +41,14 @@ Avoid using unnecessary open/close braces, vertical space is usually limited:
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
code();
|
code();
|
||||||
```
|
```
|
||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a) {
|
if(a) {
|
||||||
code();
|
code();
|
||||||
}
|
}
|
||||||
@ -58,7 +58,7 @@ Unless there are either multiple hierarchical conditions being used or that the
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
if(b)
|
if(b)
|
||||||
@ -68,7 +68,7 @@ if(a)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
if(b)
|
if(b)
|
||||||
code();
|
code();
|
||||||
@ -78,7 +78,7 @@ If there are brackets inside the body, outside brackets are required.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
for(auto elem : list)
|
for(auto elem : list)
|
||||||
@ -90,7 +90,7 @@ if(a)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
for(auto elem : list)
|
for(auto elem : list)
|
||||||
{
|
{
|
||||||
@ -102,7 +102,7 @@ If "else" branch has brackets then "if" should also have brackets even if it is
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
code();
|
code();
|
||||||
@ -118,7 +118,7 @@ else
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
code();
|
code();
|
||||||
else
|
else
|
||||||
@ -134,7 +134,7 @@ If you intentionally want to avoid usage of "else if" and keep if body indent ma
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
code();
|
code();
|
||||||
@ -148,7 +148,7 @@ else
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
code();
|
code();
|
||||||
else
|
else
|
||||||
@ -160,7 +160,7 @@ When defining a method, use a new line for the brace, like this:
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
void method()
|
void method()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
@ -168,7 +168,7 @@ void method()
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
void Method() {
|
void Method() {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
@ -179,14 +179,14 @@ Use white space in expressions liberally, except in the presence of parenthesis.
|
|||||||
|
|
||||||
**Good:**
|
**Good:**
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a + 5 > method(blah('a') + 4))
|
if(a + 5 > method(blah('a') + 4))
|
||||||
foo += 24;
|
foo += 24;
|
||||||
```
|
```
|
||||||
|
|
||||||
**Bad:**
|
**Bad:**
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a+5>method(blah('a')+4))
|
if(a+5>method(blah('a')+4))
|
||||||
foo+=24;
|
foo+=24;
|
||||||
```
|
```
|
||||||
@ -199,13 +199,13 @@ Use a space before and after the address or pointer character in a pointer decla
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
CIntObject * images[100];
|
CIntObject * images[100];
|
||||||
```
|
```
|
||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
CIntObject* images[100]; or
|
CIntObject* images[100]; or
|
||||||
CIntObject *images[100];
|
CIntObject *images[100];
|
||||||
```
|
```
|
||||||
@ -214,14 +214,14 @@ Do not use spaces before parentheses.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
code();
|
code();
|
||||||
```
|
```
|
||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if (a)
|
if (a)
|
||||||
code();
|
code();
|
||||||
```
|
```
|
||||||
@ -230,7 +230,7 @@ Do not use extra spaces around conditions inside parentheses.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a && b)
|
if(a && b)
|
||||||
code();
|
code();
|
||||||
|
|
||||||
@ -240,7 +240,7 @@ if(a && (b || c))
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if( a && b )
|
if( a && b )
|
||||||
code();
|
code();
|
||||||
|
|
||||||
@ -252,14 +252,14 @@ Do not use more than one space between operators.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if((a && b) || (c + 1 == d))
|
if((a && b) || (c + 1 == d))
|
||||||
code();
|
code();
|
||||||
```
|
```
|
||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if((a && b) || (c + 1 == d))
|
if((a && b) || (c + 1 == d))
|
||||||
code();
|
code();
|
||||||
|
|
||||||
@ -273,14 +273,14 @@ When allocating objects, don't use parentheses for creating stack-based objects
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
std::vector<int> v;
|
std::vector<int> v;
|
||||||
CGBoat btn = new CGBoat();
|
CGBoat btn = new CGBoat();
|
||||||
```
|
```
|
||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
std::vector<int> v(); // shouldn't compile anyway
|
std::vector<int> v(); // shouldn't compile anyway
|
||||||
CGBoat btn = new CGBoat;
|
CGBoat btn = new CGBoat;
|
||||||
```
|
```
|
||||||
@ -289,14 +289,14 @@ Avoid overuse of parentheses:
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a && (b + 1))
|
if(a && (b + 1))
|
||||||
return c == d;
|
return c == d;
|
||||||
```
|
```
|
||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if((a && (b + 1)))
|
if((a && (b + 1)))
|
||||||
return (c == d);
|
return (c == d);
|
||||||
```
|
```
|
||||||
@ -305,7 +305,7 @@ if((a && (b + 1)))
|
|||||||
|
|
||||||
Base class list must be on same line with class name.
|
Base class list must be on same line with class name.
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
class CClass : public CClassBaseOne, public CClassBaseOne
|
class CClass : public CClassBaseOne, public CClassBaseOne
|
||||||
{
|
{
|
||||||
int id;
|
int id;
|
||||||
@ -321,7 +321,7 @@ When 'private:', 'public:' and other labels are not on the line after opening br
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
class CClass
|
class CClass
|
||||||
{
|
{
|
||||||
int id;
|
int id;
|
||||||
@ -333,7 +333,7 @@ public:
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
class CClass
|
class CClass
|
||||||
{
|
{
|
||||||
int id;
|
int id;
|
||||||
@ -344,7 +344,7 @@ public:
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
class CClass
|
class CClass
|
||||||
{
|
{
|
||||||
protected:
|
protected:
|
||||||
@ -357,7 +357,7 @@ public:
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
class CClass
|
class CClass
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -373,7 +373,7 @@ public:
|
|||||||
|
|
||||||
Constructor member and base class initialization must be on new line, indented with tab with leading colon.
|
Constructor member and base class initialization must be on new line, indented with tab with leading colon.
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
CClass::CClass()
|
CClass::CClass()
|
||||||
: CClassBaseOne(true, nullptr), id(0), bool parameters(false)
|
: CClassBaseOne(true, nullptr), id(0), bool parameters(false)
|
||||||
{
|
{
|
||||||
@ -387,7 +387,7 @@ Switch statements have the case at the same indentation as the switch.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
switch(alignment)
|
switch(alignment)
|
||||||
{
|
{
|
||||||
case EAlignment::EVIL:
|
case EAlignment::EVIL:
|
||||||
@ -407,7 +407,7 @@ default:
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
switch(alignment)
|
switch(alignment)
|
||||||
{
|
{
|
||||||
case EAlignment::EVIL:
|
case EAlignment::EVIL:
|
||||||
@ -447,7 +447,7 @@ break;
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
auto lambda = [this, a, &b](int3 & tile, int index) -> bool
|
auto lambda = [this, a, &b](int3 & tile, int index) -> bool
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -456,7 +456,7 @@ auto lambda = [this, a, &b](int3 & tile, int index) -> bool
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
auto lambda = [this,a,&b](int3 & tile, int index)->bool{do_that();};
|
auto lambda = [this,a,&b](int3 & tile, int index)->bool{do_that();};
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -464,7 +464,7 @@ Empty parameter list is required even if function takes no arguments.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
auto lambda = []()
|
auto lambda = []()
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -473,7 +473,7 @@ auto lambda = []()
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
auto lambda = []
|
auto lambda = []
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -484,7 +484,7 @@ Do not use inline lambda expressions inside if-else, for and other conditions.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
auto lambda = []()
|
auto lambda = []()
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -497,7 +497,7 @@ if(lambda)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if([]()
|
if([]()
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -511,7 +511,7 @@ Do not pass inline lambda expressions as parameter unless it's the last paramete
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
auto lambda = []()
|
auto lambda = []()
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -521,7 +521,7 @@ obj->someMethod(lambda, true);
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
obj->someMethod([]()
|
obj->someMethod([]()
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -530,7 +530,7 @@ obj->someMethod([]()
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
obj->someMethod(true, []()
|
obj->someMethod(true, []()
|
||||||
{
|
{
|
||||||
do_that();
|
do_that();
|
||||||
@ -543,7 +543,7 @@ Serialization of each element must be on it's own line since this make debugging
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
template <typename Handler> void serialize(Handler & h, const int version)
|
template <typename Handler> void serialize(Handler & h, const int version)
|
||||||
{
|
{
|
||||||
h & identifier;
|
h & identifier;
|
||||||
@ -555,7 +555,7 @@ template <typename Handler> void serialize(Handler & h, const int version)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
template <typename Handler> void serialize(Handler & h, const int version)
|
template <typename Handler> void serialize(Handler & h, const int version)
|
||||||
{
|
{
|
||||||
h & identifier & description & name & dependencies;
|
h & identifier & description & name & dependencies;
|
||||||
@ -566,7 +566,7 @@ Save backward compatibility code is exception when extra brackets are always use
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
template <typename Handler> void serialize(Handler & h, const int version)
|
template <typename Handler> void serialize(Handler & h, const int version)
|
||||||
{
|
{
|
||||||
h & identifier;
|
h & identifier;
|
||||||
@ -586,7 +586,7 @@ template <typename Handler> void serialize(Handler & h, const int version)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
template <typename Handler> void serialize(Handler & h, const int version)
|
template <typename Handler> void serialize(Handler & h, const int version)
|
||||||
{
|
{
|
||||||
h & identifier;
|
h & identifier;
|
||||||
@ -604,7 +604,7 @@ template <typename Handler> void serialize(Handler & h, const int version)
|
|||||||
|
|
||||||
For any new files, please paste the following info block at the very top of the source file:
|
For any new files, please paste the following info block at the very top of the source file:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
/*
|
/*
|
||||||
* Name_of_File.h, part of VCMI engine
|
* Name_of_File.h, part of VCMI engine
|
||||||
*
|
*
|
||||||
@ -628,7 +628,7 @@ For any header or source file code must be in following order:
|
|||||||
4. Forward declarations
|
4. Forward declarations
|
||||||
5. All other code
|
5. All other code
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
/*
|
/*
|
||||||
* Name_of_File.h, part of VCMI engine
|
* Name_of_File.h, part of VCMI engine
|
||||||
*
|
*
|
||||||
@ -652,7 +652,7 @@ If you comment on the same line with code there must be one single space between
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
code(); //Do something
|
code(); //Do something
|
||||||
@ -665,7 +665,7 @@ else // Do something.
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
code();//Do something
|
code();//Do something
|
||||||
@ -680,7 +680,7 @@ If you add single-line comment on own line slashes must have same indent as code
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
// Do something
|
// Do something
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
@ -692,7 +692,7 @@ if(a)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
// Do something
|
// Do something
|
||||||
if(a)
|
if(a)
|
||||||
{
|
{
|
||||||
@ -706,7 +706,7 @@ Avoid comments inside multi-line if-else conditions. If your conditions are too
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
bool isMyHeroAlive = a && b || (c + 1 > 15);
|
bool isMyHeroAlive = a && b || (c + 1 > 15);
|
||||||
bool canMyHeroMove = myTurn && hero.movePoints > 0;
|
bool canMyHeroMove = myTurn && hero.movePoints > 0;
|
||||||
if(isMyHeroAlive && canMyHeroMove)
|
if(isMyHeroAlive && canMyHeroMove)
|
||||||
@ -717,7 +717,7 @@ if(isMyHeroAlive && canMyHeroMove)
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
if((a && b || (c + 1 > 15)) //Check if hero still alive
|
if((a && b || (c + 1 > 15)) //Check if hero still alive
|
||||||
&& myTurn && hero.movePoints > 0) //Check if hero can move
|
&& myTurn && hero.movePoints > 0) //Check if hero can move
|
||||||
{
|
{
|
||||||
@ -727,7 +727,7 @@ if((a && b || (c + 1 > 15)) //Check if hero still alive
|
|||||||
|
|
||||||
You should write a comment before the class definition which describes shortly the class. 1-2 sentences are enough. Methods and class data members should be commented if they aren't self-describing only. Getters/Setters, simple methods where the purpose is clear or similar methods shouldn't be commented, because vertical space is usually limited. The style of documentation comments should be the three slashes-style: ///.
|
You should write a comment before the class definition which describes shortly the class. 1-2 sentences are enough. Methods and class data members should be commented if they aren't self-describing only. Getters/Setters, simple methods where the purpose is clear or similar methods shouldn't be commented, because vertical space is usually limited. The style of documentation comments should be the three slashes-style: ///.
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
/// Returns true if a debug/trace log message will be logged, false if not.
|
/// Returns true if a debug/trace log message will be logged, false if not.
|
||||||
/// Useful if performance is important and concatenating the log message is a expensive task.
|
/// Useful if performance is important and concatenating the log message is a expensive task.
|
||||||
bool isDebugEnabled() const;
|
bool isDebugEnabled() const;
|
||||||
@ -738,7 +738,7 @@ The above example doesn't follow a strict scheme on how to comment a method. It
|
|||||||
|
|
||||||
If you need a more detailed description for a method you can use such style:
|
If you need a more detailed description for a method you can use such style:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
/// <A short one line description>
|
/// <A short one line description>
|
||||||
///
|
///
|
||||||
/// <Longer description>
|
/// <Longer description>
|
||||||
@ -749,7 +749,7 @@ If you need a more detailed description for a method you can use such style:
|
|||||||
/// @return Description of the return value
|
/// @return Description of the return value
|
||||||
```
|
```
|
||||||
|
|
||||||
A good essay about writing comments: http://ardalis.com/when-to-comment-your-code
|
A good essay about writing comments: <http://ardalis.com/when-to-comment-your-code>
|
||||||
|
|
||||||
### Casing
|
### Casing
|
||||||
|
|
||||||
@ -775,7 +775,7 @@ Outdated. There is separate entry for [Logging API](Logging_API.md)
|
|||||||
|
|
||||||
If you want to trace the control flow of VCMI, then you should use the macro LOG_TRACE or LOG_TRACE_PARAMS. The first one prints a message when the function is entered or leaved. The name of the function will also be logged. In addition to this the second macro, let's you specify parameters which you want to print. You should print traces with parameters like this:
|
If you want to trace the control flow of VCMI, then you should use the macro LOG_TRACE or LOG_TRACE_PARAMS. The first one prints a message when the function is entered or leaved. The name of the function will also be logged. In addition to this the second macro, let's you specify parameters which you want to print. You should print traces with parameters like this:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
LOG_TRACE_PARAMS(logGlobal, "hero '%s', spellId '%d', pos '%s'.", hero, spellId, pos);
|
LOG_TRACE_PARAMS(logGlobal, "hero '%s', spellId '%d', pos '%s'.", hero, spellId, pos);
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -797,14 +797,14 @@ Do not use uncommon abbreviations for class, method, parameter and global object
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
CArt * getRandomArt(...)
|
CArt * getRandomArt(...)
|
||||||
class CIntObject
|
class CIntObject
|
||||||
```
|
```
|
||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
CArtifact * getRandomArtifact(...)
|
CArtifact * getRandomArtifact(...)
|
||||||
class CInterfaceObject
|
class CInterfaceObject
|
||||||
```
|
```
|
||||||
@ -827,7 +827,7 @@ The header StdInc.h should be included in every compilation unit. It has to be i
|
|||||||
|
|
||||||
Do not declare enumerations in global namespace. It is better to use strongly typed enum or to wrap them in class or namespace to avoid polluting global namespace:
|
Do not declare enumerations in global namespace. It is better to use strongly typed enum or to wrap them in class or namespace to avoid polluting global namespace:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
enum class EAlignment
|
enum class EAlignment
|
||||||
{
|
{
|
||||||
GOOD,
|
GOOD,
|
||||||
@ -848,7 +848,7 @@ namespace EAlignment
|
|||||||
|
|
||||||
If the comment duplicates the name of commented member, it's better if it wouldn't exist at all. It just increases maintenance cost. Bad:
|
If the comment duplicates the name of commented member, it's better if it wouldn't exist at all. It just increases maintenance cost. Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
size_t getHeroesCount(); //gets count of heroes (surprise?)
|
size_t getHeroesCount(); //gets count of heroes (surprise?)
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -862,13 +862,13 @@ Don't return const objects or primitive types from functions -- it's pointless.
|
|||||||
|
|
||||||
Bad:
|
Bad:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
const std::vector<CGObjectInstance *> guardingCreatures(int3 pos) const;
|
const std::vector<CGObjectInstance *> guardingCreatures(int3 pos) const;
|
||||||
```
|
```
|
||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
``` cpp
|
```cpp
|
||||||
std::vector<const CGObjectInstance *> guardingCreatures(int3 pos) const;
|
std::vector<const CGObjectInstance *> guardingCreatures(int3 pos) const;
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -27,7 +27,7 @@ The following platforms are supported and known to work, others might require ch
|
|||||||
- **Windows**: libraries are built with x86_64-mingw-w64-gcc version 10 (which is available in repositories of Ubuntu 22.04)
|
- **Windows**: libraries are built with x86_64-mingw-w64-gcc version 10 (which is available in repositories of Ubuntu 22.04)
|
||||||
- **Android**: libraries are built with NDK r25c (25.2.9519653)
|
- **Android**: libraries are built with NDK r25c (25.2.9519653)
|
||||||
|
|
||||||
2. Download the binaries archive and unpack it to `~/.conan` directory from https://github.com/vcmi/vcmi-dependencies/releases/latest
|
2. Download the binaries archive and unpack it to `~/.conan` directory from <https://github.com/vcmi/vcmi-dependencies/releases/latest>
|
||||||
|
|
||||||
- macOS: pick **dependencies-mac-intel.txz** if you have Intel Mac, otherwise - **dependencies-mac-arm.txz**
|
- macOS: pick **dependencies-mac-intel.txz** if you have Intel Mac, otherwise - **dependencies-mac-arm.txz**
|
||||||
- iOS: pick ***dependencies-ios.txz***
|
- iOS: pick ***dependencies-ios.txz***
|
||||||
@ -65,7 +65,7 @@ If you use `--build=never` and this command fails, then it means that you can't
|
|||||||
|
|
||||||
VCMI "recipe" also has some options that you can specify. For example, if you don't care about game videos, you can disable FFmpeg dependency by passing `-o with_ffmpeg=False`. If you only want to make release build, you can use `GENERATE_ONLY_BUILT_CONFIG=1` environment variable to skip generating files for other configurations (our CI does this).
|
VCMI "recipe" also has some options that you can specify. For example, if you don't care about game videos, you can disable FFmpeg dependency by passing `-o with_ffmpeg=False`. If you only want to make release build, you can use `GENERATE_ONLY_BUILT_CONFIG=1` environment variable to skip generating files for other configurations (our CI does this).
|
||||||
|
|
||||||
_Note_: you can find full reference of this command [in the official documentation](https://docs.conan.io/1/reference/commands/consumer/install.html) or by executing `conan help install`.
|
*Note*: you can find full reference of this command [in the official documentation](https://docs.conan.io/1/reference/commands/consumer/install.html) or by executing `conan help install`.
|
||||||
|
|
||||||
### Using our prebuilt binaries for macOS/iOS
|
### Using our prebuilt binaries for macOS/iOS
|
||||||
|
|
||||||
@ -86,7 +86,7 @@ This subsection describes platform specifics to build libraries from source prop
|
|||||||
|
|
||||||
#### Building for macOS/iOS
|
#### Building for macOS/iOS
|
||||||
|
|
||||||
- To build Locale module of Boost in versions >= 1.81, you must use `compiler.cppstd=11` Conan setting (our profiles already contain it). To use it with another profile, either add this setting to your _host_ profile or pass `-s compiler.cppstd=11` on the command line.
|
- To build Locale module of Boost in versions >= 1.81, you must use `compiler.cppstd=11` Conan setting (our profiles already contain it). To use it with another profile, either add this setting to your *host* profile or pass `-s compiler.cppstd=11` on the command line.
|
||||||
- If you wish to build dependencies against system libraries (like our prebuilt ones do), follow [below instructions](#using-recipes-for-system-libraries) executing `conan create` for all directories. Don't forget to pass `-o with_apple_system_libs=True` to `conan install` afterwards.
|
- If you wish to build dependencies against system libraries (like our prebuilt ones do), follow [below instructions](#using-recipes-for-system-libraries) executing `conan create` for all directories. Don't forget to pass `-o with_apple_system_libs=True` to `conan install` afterwards.
|
||||||
|
|
||||||
#### Building for Android
|
#### Building for Android
|
||||||
@ -105,11 +105,11 @@ After applying patch(es):
|
|||||||
2. Run `make`
|
2. Run `make`
|
||||||
3. Copy file `qtbase/jar/QtAndroid.jar` from the build directory to the **package directory**, e.g. `~/.conan/data/qt/5.15.14/_/_/package/SOME_HASH/jar`.
|
3. Copy file `qtbase/jar/QtAndroid.jar` from the build directory to the **package directory**, e.g. `~/.conan/data/qt/5.15.14/_/_/package/SOME_HASH/jar`.
|
||||||
|
|
||||||
_Note_: if you plan to build Qt from source again, then you don't need to perform the above _After applying patch(es)_ steps after building.
|
*Note*: if you plan to build Qt from source again, then you don't need to perform the above *After applying patch(es)* steps after building.
|
||||||
|
|
||||||
##### Using recipes for system libraries
|
##### Using recipes for system libraries
|
||||||
|
|
||||||
1. Clone/download https://github.com/kambala-decapitator/conan-system-libs
|
1. Clone/download <https://github.com/kambala-decapitator/conan-system-libs>
|
||||||
2. Execute `conan create PACKAGE vcmi/CHANNEL`, where `PACKAGE` is a directory path in that repository and `CHANNEL` is **apple** for macOS/iOS and **android** for Android. Do it for each library you need.
|
2. Execute `conan create PACKAGE vcmi/CHANNEL`, where `PACKAGE` is a directory path in that repository and `CHANNEL` is **apple** for macOS/iOS and **android** for Android. Do it for each library you need.
|
||||||
3. Now you can execute `conan install` to build all dependencies.
|
3. Now you can execute `conan install` to build all dependencies.
|
||||||
|
|
||||||
@ -172,7 +172,7 @@ cmake --preset ios-conan
|
|||||||
|
|
||||||
`CMakeUserPresets.json` file:
|
`CMakeUserPresets.json` file:
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
{
|
{
|
||||||
"version": 3,
|
"version": 3,
|
||||||
"cmakeMinimumRequired": {
|
"cmakeMinimumRequired": {
|
||||||
|
@ -6,7 +6,7 @@ Qt Creator is the recommended IDE for VCMI development on Linux distributions, b
|
|||||||
- Almost no manual configuration when used with CMake. Project configuration is read from CMake text files,
|
- Almost no manual configuration when used with CMake. Project configuration is read from CMake text files,
|
||||||
- Easy to setup and use with multiple different compiler toolchains: GCC, Visual Studio, Clang
|
- Easy to setup and use with multiple different compiler toolchains: GCC, Visual Studio, Clang
|
||||||
|
|
||||||
You can install Qt Creator from repository, but better to stick to latest version from Qt website: https://www.qt.io/download-qt-installer-oss
|
You can install Qt Creator from repository, but better to stick to latest version from Qt website: <https://www.qt.io/download-qt-installer-oss>
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
@ -21,4 +21,4 @@ The build dir should be set to something like /trunk/build for the debug build a
|
|||||||
There is a problem with QtCreator when debugging both vcmiclient and vcmiserver. If you debug the vcmiclient, start a game, attach the vcmiserver process to the gdb debugger(Debug \> Start Debugging \> Attach to Running External Application...) then breakpoints which are set for vcmiserver will be ignored. This looks like a bug, in any case it's not intuitively. Two workarounds are available luckily:
|
There is a problem with QtCreator when debugging both vcmiclient and vcmiserver. If you debug the vcmiclient, start a game, attach the vcmiserver process to the gdb debugger(Debug \> Start Debugging \> Attach to Running External Application...) then breakpoints which are set for vcmiserver will be ignored. This looks like a bug, in any case it's not intuitively. Two workarounds are available luckily:
|
||||||
|
|
||||||
1. Run vcmiclient (no debug mode), then attach server process to the debugger
|
1. Run vcmiclient (no debug mode), then attach server process to the debugger
|
||||||
2. Open two instances of QtCreator and debug vcmiserver and vcmiclient separately(it works!)
|
2. Open two instances of QtCreator and debug vcmiserver and vcmiclient separately (it works!)
|
||||||
|
@ -24,7 +24,7 @@ Some notes:
|
|||||||
|
|
||||||
### Setup settings.json
|
### Setup settings.json
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"logging" : {
|
"logging" : {
|
||||||
"console" : {
|
"console" : {
|
||||||
@ -68,7 +68,7 @@ The following code shows how the logging system can be configured:
|
|||||||
|
|
||||||
If `configureDefault` or `configure` won't be called, then logs aren't written either to the console or to the file. The default logging setups a system like this:
|
If `configureDefault` or `configure` won't be called, then logs aren't written either to the console or to the file. The default logging setups a system like this:
|
||||||
|
|
||||||
**Console**
|
#### Console
|
||||||
|
|
||||||
Format: %m
|
Format: %m
|
||||||
Threshold: info
|
Threshold: info
|
||||||
@ -76,17 +76,18 @@ coloredOutputEnabled: true
|
|||||||
|
|
||||||
colorMapping: trace -\> gray, debug -\> white, info -\> green, warn -\> yellow, error -\> red
|
colorMapping: trace -\> gray, debug -\> white, info -\> green, warn -\> yellow, error -\> red
|
||||||
|
|
||||||
**File**
|
#### File
|
||||||
|
|
||||||
Format: %d %l %n \[%t\] - %m
|
Format: %d %l %n \[%t\] - %m
|
||||||
|
|
||||||
**Loggers**
|
#### Loggers
|
||||||
|
|
||||||
global -\> info
|
global -\> info
|
||||||
|
|
||||||
### How to get a logger
|
### How to get a logger
|
||||||
|
|
||||||
There exist only one logger object per domain. A logger object cannot be copied. You can get access to a logger object by using the globally defined ones like `logGlobal` or `logAi`, etc... or by getting one manually:
|
There exist only one logger object per domain. A logger object cannot be copied. You can get access to a logger object by using the globally defined ones like `logGlobal` or `logAi`, etc... or by getting one manually:
|
||||||
|
|
||||||
```cpp
|
```cpp
|
||||||
Logger * logger = CLogger::getLogger(CLoggerDomain("rmg"));
|
Logger * logger = CLogger::getLogger(CLoggerDomain("rmg"));
|
||||||
```
|
```
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
//general purpose script, Lua or ERM, runs on server
|
//general purpose script, Lua or ERM, runs on server
|
||||||
"myScript":
|
"myScript":
|
||||||
@ -155,7 +155,7 @@ Following libraries are supported
|
|||||||
|
|
||||||
- TODO !!AR Артефакт (ресурс) в определенной позиции
|
- TODO !!AR Артефакт (ресурс) в определенной позиции
|
||||||
- TODO !!BA Битва
|
- TODO !!BA Битва
|
||||||
- !!BA:A$ return 1 for battle evaluation
|
- !!BA:A$ return 1 for battle evaluation
|
||||||
- TODO !!BF Препятствия на поле боя
|
- TODO !!BF Препятствия на поле боя
|
||||||
- TODO !!BG Действий монстров в бою
|
- TODO !!BG Действий монстров в бою
|
||||||
- TODO !!BH Действия героя в бою
|
- TODO !!BH Действия героя в бою
|
||||||
|
@ -5,12 +5,14 @@
|
|||||||
For implementation details see files located at `lib/network` directory.
|
For implementation details see files located at `lib/network` directory.
|
||||||
|
|
||||||
VCMI uses connection using TCP to communicate with server, even in single-player games. However, even though TCP is stream-based protocol, VCMI uses atomic messages for communication. Each message is a serialized stream of bytes, preceded by 4-byte message size:
|
VCMI uses connection using TCP to communicate with server, even in single-player games. However, even though TCP is stream-based protocol, VCMI uses atomic messages for communication. Each message is a serialized stream of bytes, preceded by 4-byte message size:
|
||||||
|
|
||||||
```
|
```
|
||||||
int32_t messageSize;
|
int32_t messageSize;
|
||||||
byte messagePayload[messageSize];
|
byte messagePayload[messageSize];
|
||||||
```
|
```
|
||||||
|
|
||||||
Networking can be used by:
|
Networking can be used by:
|
||||||
|
|
||||||
- game client (vcmiclient / VCMI_Client.exe). Actual application that player interacts with directly using UI.
|
- game client (vcmiclient / VCMI_Client.exe). Actual application that player interacts with directly using UI.
|
||||||
- match server (vcmiserver / VCMI_Server.exe / part of game client). This app controls game logic and coordinates multiplayer games.
|
- match server (vcmiserver / VCMI_Server.exe / part of game client). This app controls game logic and coordinates multiplayer games.
|
||||||
- lobby server (vcmilobby). This app provides access to global lobby through which players can play game over Internet.
|
- lobby server (vcmilobby). This app provides access to global lobby through which players can play game over Internet.
|
||||||
@ -28,11 +30,13 @@ For gameplay, VCMI serializes data into a binary stream. See [Serialization](Ser
|
|||||||
## Global lobby communication
|
## Global lobby communication
|
||||||
|
|
||||||
For implementation details see:
|
For implementation details see:
|
||||||
|
|
||||||
- game client: `client/globalLobby/GlobalLobbyClient.h
|
- game client: `client/globalLobby/GlobalLobbyClient.h
|
||||||
- match server: `server/GlobalLobbyProcessor.h
|
- match server: `server/GlobalLobbyProcessor.h
|
||||||
- lobby server: `client/globalLobby/GlobalLobbyClient.h
|
- lobby server: `client/globalLobby/GlobalLobbyClient.h
|
||||||
|
|
||||||
In case of global lobby, message payload uses plaintext json format - utf-8 encoded string:
|
In case of global lobby, message payload uses plaintext json format - utf-8 encoded string:
|
||||||
|
|
||||||
```
|
```
|
||||||
int32_t messageSize;
|
int32_t messageSize;
|
||||||
char jsonString[messageSize];
|
char jsonString[messageSize];
|
||||||
@ -43,6 +47,7 @@ Every message must be a struct (json object) that contains "type" field. Unlike
|
|||||||
### Communication flow
|
### Communication flow
|
||||||
|
|
||||||
Notes:
|
Notes:
|
||||||
|
|
||||||
- invalid message, such as corrupted json format or failure to validate message will result in no reply from server
|
- invalid message, such as corrupted json format or failure to validate message will result in no reply from server
|
||||||
- in addition to specified messages, match server will send `operationFailed` message on failure to apply player request
|
- in addition to specified messages, match server will send `operationFailed` message on failure to apply player request
|
||||||
|
|
||||||
@ -52,6 +57,7 @@ Notes:
|
|||||||
- lobby -> client: `accountCreated`
|
- lobby -> client: `accountCreated`
|
||||||
|
|
||||||
#### Login
|
#### Login
|
||||||
|
|
||||||
- client -> lobby: `clientLogin`
|
- client -> lobby: `clientLogin`
|
||||||
- lobby -> client: `loginSuccess`
|
- lobby -> client: `loginSuccess`
|
||||||
- lobby -> client: `chatHistory`
|
- lobby -> client: `chatHistory`
|
||||||
@ -59,10 +65,12 @@ Notes:
|
|||||||
- lobby -> client: `activeGameRooms`
|
- lobby -> client: `activeGameRooms`
|
||||||
|
|
||||||
#### Chat Message
|
#### Chat Message
|
||||||
|
|
||||||
- client -> lobby: `sendChatMessage`
|
- client -> lobby: `sendChatMessage`
|
||||||
- lobby -> every client: `chatMessage`
|
- lobby -> every client: `chatMessage`
|
||||||
|
|
||||||
#### New Game Room
|
#### New Game Room
|
||||||
|
|
||||||
- client starts match server instance
|
- client starts match server instance
|
||||||
- match -> lobby: `serverLogin`
|
- match -> lobby: `serverLogin`
|
||||||
- lobby -> match: `loginSuccess`
|
- lobby -> match: `loginSuccess`
|
||||||
@ -73,19 +81,23 @@ Notes:
|
|||||||
- lobby -> every client: `activeGameRooms`
|
- lobby -> every client: `activeGameRooms`
|
||||||
|
|
||||||
#### Joining a game room
|
#### Joining a game room
|
||||||
|
|
||||||
See [#Proxy mode](proxy-mode)
|
See [#Proxy mode](proxy-mode)
|
||||||
|
|
||||||
#### Leaving a game room
|
#### Leaving a game room
|
||||||
|
|
||||||
- client closes connection to match server
|
- client closes connection to match server
|
||||||
- match -> lobby: `leaveGameRoom`
|
- match -> lobby: `leaveGameRoom`
|
||||||
|
|
||||||
#### Sending an invite:
|
#### Sending an invite
|
||||||
|
|
||||||
- client -> lobby: `sendInvite`
|
- client -> lobby: `sendInvite`
|
||||||
- lobby -> target client: `inviteReceived`
|
- lobby -> target client: `inviteReceived`
|
||||||
|
|
||||||
Note: there is no dedicated procedure to accept an invite. Instead, invited player will use same flow as when joining public game room
|
Note: there is no dedicated procedure to accept an invite. Instead, invited player will use same flow as when joining public game room
|
||||||
|
|
||||||
#### Logout
|
#### Logout
|
||||||
|
|
||||||
- client closes connection
|
- client closes connection
|
||||||
- lobby -> every client: `activeAccounts`
|
- lobby -> every client: `activeAccounts`
|
||||||
|
|
||||||
@ -94,6 +106,7 @@ Note: there is no dedicated procedure to accept an invite. Instead, invited play
|
|||||||
In order to connect players located behind NAT, VCMI lobby can operate in "proxy" mode. In this mode, connection will be act as proxy and will transmit gameplay data from client to a match server, without any data processing on lobby server.
|
In order to connect players located behind NAT, VCMI lobby can operate in "proxy" mode. In this mode, connection will be act as proxy and will transmit gameplay data from client to a match server, without any data processing on lobby server.
|
||||||
|
|
||||||
Currently, process to establish connection using proxy mode is:
|
Currently, process to establish connection using proxy mode is:
|
||||||
|
|
||||||
- Player attempt to join open game room using `joinGameRoom` message
|
- Player attempt to join open game room using `joinGameRoom` message
|
||||||
- Lobby server validates requests and on success - notifies match server about new player in lobby using control connection
|
- Lobby server validates requests and on success - notifies match server about new player in lobby using control connection
|
||||||
- Match server receives request, establishes new connection to game lobby, sends `serverProxyLogin` message to lobby server and immediately transfers this connection to VCMIServer class to use as connection for gameplay communication
|
- Match server receives request, establishes new connection to game lobby, sends `serverProxyLogin` message to lobby server and immediately transfers this connection to VCMIServer class to use as connection for gameplay communication
|
||||||
|
@ -44,17 +44,17 @@ We want to notify players about updates on as many social services as possible.
|
|||||||
- Administrator access: SXX, Warmonger
|
- Administrator access: SXX, Warmonger
|
||||||
- Twitter account: <https://twitter.com/VCMIOfficial>
|
- Twitter account: <https://twitter.com/VCMIOfficial>
|
||||||
- Administrator access: SXX.
|
- Administrator access: SXX.
|
||||||
- User access via TweetDeck:
|
- User access via TweetDeck:
|
||||||
- VK / VKontakte page: <https://vk.com/VCMIOfficial>
|
- VK / VKontakte page: <https://vk.com/VCMIOfficial>
|
||||||
- Owner: SXX
|
- Owner: SXX
|
||||||
- Administrator access: AVS
|
- Administrator access: AVS
|
||||||
- Steam group: <https://steamcommunity.com/groups/VCMI>
|
- Steam group: <https://steamcommunity.com/groups/VCMI>
|
||||||
- Administrator access: SXX
|
- Administrator access: SXX
|
||||||
- Moderator access: Dydzio
|
- Moderator access: Dydzio
|
||||||
- Reddit: <https://reddit.com/r/vcmi/>
|
- Reddit: <https://reddit.com/r/vcmi/>
|
||||||
- Administrator access: SXX
|
- Administrator access: SXX
|
||||||
- ModDB entry: <http://www.moddb.com/engines/vcmi>
|
- ModDB entry: <http://www.moddb.com/engines/vcmi>
|
||||||
- Administrator access: SXX
|
- Administrator access: SXX
|
||||||
|
|
||||||
### Communication channels
|
### Communication channels
|
||||||
|
|
||||||
@ -94,13 +94,11 @@ Reserve accounts for other code hosting services:
|
|||||||
2. Use 2FA on CloudFlare and just ask everyone to get FreeOTP and then use shared secret.
|
2. Use 2FA on CloudFlare and just ask everyone to get FreeOTP and then use shared secret.
|
||||||
3. Centralized way to post news about game updates to all social media.
|
3. Centralized way to post news about game updates to all social media.
|
||||||
|
|
||||||
# Project Servers Configuration
|
## Project Servers Configuration
|
||||||
|
|
||||||
This section dedicated to explain specific configurations of our servers for anyone who might need to improve it in future.
|
This section dedicated to explain specific configurations of our servers for anyone who might need to improve it in future.
|
||||||
|
|
||||||
## Droplet configuration
|
### Droplet configuration
|
||||||
|
|
||||||
### Droplet and hosted services
|
|
||||||
|
|
||||||
Currently we using two droplets:
|
Currently we using two droplets:
|
||||||
|
|
||||||
|
@ -1,12 +1,16 @@
|
|||||||
# Release Process
|
# Release Process
|
||||||
|
|
||||||
## Versioning
|
## Versioning
|
||||||
|
|
||||||
For releases VCMI uses version numbering in form "1.X.Y", where:
|
For releases VCMI uses version numbering in form "1.X.Y", where:
|
||||||
|
|
||||||
- 'X' indicates major release. Different major versions are generally not compatible with each other. Save format is different, network protocol is different, mod format likely different.
|
- 'X' indicates major release. Different major versions are generally not compatible with each other. Save format is different, network protocol is different, mod format likely different.
|
||||||
- 'Y' indicates hotfix release. Despite its name this is usually not urgent, but planned release. Different hotfixes for same major version are fully compatible with each other.
|
- 'Y' indicates hotfix release. Despite its name this is usually not urgent, but planned release. Different hotfixes for same major version are fully compatible with each other.
|
||||||
|
|
||||||
## Branches
|
## Branches
|
||||||
|
|
||||||
Our branching strategy is very similar to GitFlow:
|
Our branching strategy is very similar to GitFlow:
|
||||||
|
|
||||||
- `master` branch has release commits. One commit - one release. Each release commit should be tagged with version `1.X.Y` when corresponding version is released. State of master branch represents state of latest public release.
|
- `master` branch has release commits. One commit - one release. Each release commit should be tagged with version `1.X.Y` when corresponding version is released. State of master branch represents state of latest public release.
|
||||||
- `beta` branch is for stabilization of ongoing release. Beta branch is created when new major release enters stabilization stage and is used for both major release itself as well as for subsequent hotfixes. Only changes that are safe, have minimal chance of regressions and improve player experience should be targeted into this branch. Breaking changes (e.g. save format changes) are forbidden in beta.
|
- `beta` branch is for stabilization of ongoing release. Beta branch is created when new major release enters stabilization stage and is used for both major release itself as well as for subsequent hotfixes. Only changes that are safe, have minimal chance of regressions and improve player experience should be targeted into this branch. Breaking changes (e.g. save format changes) are forbidden in beta.
|
||||||
- `develop` branch is a main branch for ongoing development. Pull requests with new features should be targeted to this branch, `develop` version is one major release ahead of `beta`.
|
- `develop` branch is a main branch for ongoing development. Pull requests with new features should be targeted to this branch, `develop` version is one major release ahead of `beta`.
|
||||||
@ -14,12 +18,14 @@ Our branching strategy is very similar to GitFlow:
|
|||||||
## Release process step-by-step
|
## Release process step-by-step
|
||||||
|
|
||||||
### Initial release setup (major releases only)
|
### Initial release setup (major releases only)
|
||||||
|
|
||||||
Should be done immediately after start of stabilization stage for previous release
|
Should be done immediately after start of stabilization stage for previous release
|
||||||
|
|
||||||
- Create project named `Release 1.X`
|
- Create project named `Release 1.X`
|
||||||
- Add all features and bugs that should be fixed as part of this release into this project
|
- Add all features and bugs that should be fixed as part of this release into this project
|
||||||
|
|
||||||
### Start of stabilization stage (major releases only)
|
### Start of stabilization stage (major releases only)
|
||||||
|
|
||||||
Should be done 2 weeks before planned release date. All major features should be finished at this point.
|
Should be done 2 weeks before planned release date. All major features should be finished at this point.
|
||||||
|
|
||||||
- Create `beta` branch from `develop`
|
- Create `beta` branch from `develop`
|
||||||
@ -34,6 +40,7 @@ Should be done 2 weeks before planned release date. All major features should be
|
|||||||
- Bump version and build ID for Android on `beta` branch
|
- Bump version and build ID for Android on `beta` branch
|
||||||
|
|
||||||
### Release preparation stage
|
### Release preparation stage
|
||||||
|
|
||||||
Should be done 1 week before release. Release date should be decided at this point.
|
Should be done 1 week before release. Release date should be decided at this point.
|
||||||
|
|
||||||
- Make sure to announce codebase freeze deadline (1 day before release) to all developers
|
- Make sure to announce codebase freeze deadline (1 day before release) to all developers
|
||||||
@ -45,21 +52,23 @@ Should be done 1 week before release. Release date should be decided at this poi
|
|||||||
- - Update downloads counter in `docs/readme.md`
|
- - Update downloads counter in `docs/readme.md`
|
||||||
|
|
||||||
### Release preparation stage
|
### Release preparation stage
|
||||||
|
|
||||||
Should be done 1 day before release. At this point beta branch is in full freeze.
|
Should be done 1 day before release. At this point beta branch is in full freeze.
|
||||||
|
|
||||||
- Merge release preparation PR into `beta`
|
- Merge release preparation PR into `beta`
|
||||||
- Merge `beta` into `master`. This will trigger CI pipeline that will generate release packages
|
- Merge `beta` into `master`. This will trigger CI pipeline that will generate release packages
|
||||||
- Create draft release page, specify `1.x.y` as tag for `master` after publishing
|
- Create draft release page, specify `1.x.y` as tag for `master` after publishing
|
||||||
- Check that artifacts for all platforms have been built by CI on `master` branch
|
- Check that artifacts for all platforms have been built by CI on `master` branch
|
||||||
- Download and rename all build artifacts to use form "VCMI-1.X.Y-Platform.xxx"
|
- Download and rename all build artifacts to use form `VCMI-1.X.Y-Platform.xxx`
|
||||||
- Attach build artifacts for all platforms to release page
|
- Attach build artifacts for all platforms to release page
|
||||||
- Manually extract Windows installer, remove `$PLUGINSDIR` directory which contains installer files and repackage data as .zip archive
|
- Manually extract Windows installer, remove `$PLUGINSDIR` directory which contains installer files and repackage data as .zip archive
|
||||||
- Attach produced zip archive to release page as an alternative Windows installer
|
- Attach produced zip archive to release page as an alternative Windows installer
|
||||||
- Upload built AAB to Google Play and send created release draft for review (usually takes several hours)
|
- Upload built AAB to Google Play and send created release draft for review (usually takes several hours)
|
||||||
- Prepare pull request for [vcmi-updates](https://github.com/vcmi/vcmi-updates)
|
- Prepare pull request for [vcmi-updates](https://github.com/vcmi/vcmi-updates)
|
||||||
- (major releases only) Prepare pull request with release update for web site https://github.com/vcmi/VCMI.eu
|
- (major releases only) Prepare pull request with release update for web site <https://github.com/vcmi/VCMI.eu>
|
||||||
|
|
||||||
### Release publishing phase
|
### Release publishing phase
|
||||||
|
|
||||||
Should be done on release date
|
Should be done on release date
|
||||||
|
|
||||||
- Trigger builds for new release on Ubuntu PPA
|
- Trigger builds for new release on Ubuntu PPA
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
# Ubuntu PPA
|
# Ubuntu PPA
|
||||||
|
|
||||||
## Main links
|
## Main links
|
||||||
|
|
||||||
- [Team](https://launchpad.net/~vcmi)
|
- [Team](https://launchpad.net/~vcmi)
|
||||||
- [Project](https://launchpad.net/vcmi)
|
- [Project](https://launchpad.net/vcmi)
|
||||||
- [Sources](https://code.launchpad.net/~vcmi/vcmi/+git/vcmi)
|
- [Sources](https://code.launchpad.net/~vcmi/vcmi/+git/vcmi)
|
||||||
@ -14,22 +15,31 @@
|
|||||||
## Automatic daily builds process
|
## Automatic daily builds process
|
||||||
|
|
||||||
### Code import
|
### Code import
|
||||||
|
|
||||||
- Launchpad performs regular (once per few hours) clone of our git repository.
|
- Launchpad performs regular (once per few hours) clone of our git repository.
|
||||||
- This process can be observed on [Sources](https://code.launchpad.net/~vcmi/vcmi/+git/vcmi) page.
|
- This process can be observed on [Sources](https://code.launchpad.net/~vcmi/vcmi/+git/vcmi) page.
|
||||||
- If necessary, it is possible to trigger fresh clone immediately (Import Now button)
|
- If necessary, it is possible to trigger fresh clone immediately (Import Now button)
|
||||||
|
|
||||||
### Build dependencies
|
### Build dependencies
|
||||||
|
|
||||||
- All packages required for building of vcmi are defined in [debian/control](https://github.com/vcmi/vcmi/blob/develop/debian/control) file
|
- All packages required for building of vcmi are defined in [debian/control](https://github.com/vcmi/vcmi/blob/develop/debian/control) file
|
||||||
- Launchpad will automatically install build dependencies during build
|
- Launchpad will automatically install build dependencies during build
|
||||||
- Dependencies of output .deb package are defined implicitly as dependencies of packages required for build
|
- Dependencies of output .deb package are defined implicitly as dependencies of packages required for build
|
||||||
|
|
||||||
### Recipe building
|
### Recipe building
|
||||||
|
|
||||||
- Every 24 hours Launchpad triggers daily builds on all recipes that have build schedule enable. For vcmi this is [Daily recipe](https://code.launchpad.net/~vcmi/+recipe/vcmi-daily)
|
- Every 24 hours Launchpad triggers daily builds on all recipes that have build schedule enable. For vcmi this is [Daily recipe](https://code.launchpad.net/~vcmi/+recipe/vcmi-daily)
|
||||||
- Alternatively, builds can be triggered manually using "request build(s) link on recipe page. VCMI uses this for [Stable recipe](https://code.launchpad.net/~vcmi/+recipe/vcmi-stable)
|
- Alternatively, builds can be triggered manually using "request build(s) link on recipe page. VCMI uses this for [Stable recipe](https://code.launchpad.net/~vcmi/+recipe/vcmi-stable)
|
||||||
|
|
||||||
### Recipe content (build settings)
|
### Recipe content (build settings)
|
||||||
|
|
||||||
- Version of resulting .deb package is set in recipe content, e.g `{debupstream}+git{revtime}` for daily builds
|
- Version of resulting .deb package is set in recipe content, e.g `{debupstream}+git{revtime}` for daily builds
|
||||||
- Base version (referred as `debupstream` on Launchpad is taken from source code, [debian/changelog](https://github.com/vcmi/vcmi/blob/develop/debian/changelog) file
|
- Base version (referred as `debupstream` on Launchpad is taken from source code, [debian/changelog](https://github.com/vcmi/vcmi/blob/develop/debian/changelog) file
|
||||||
- CMake configuration settings are taken from source code, [debian/rules](https://github.com/vcmi/vcmi/blob/develop/debian/rules) file
|
- CMake configuration settings are taken from source code, [debian/rules](https://github.com/vcmi/vcmi/blob/develop/debian/rules) file
|
||||||
- Branch which is used for build is specified in recipe content, e.g. `lp:vcmi master`
|
- Branch which is used for build is specified in recipe content, e.g. `lp:vcmi master`
|
||||||
|
|
||||||
## Workflow for creating a release build
|
## Workflow for creating a release build
|
||||||
|
|
||||||
- if necessary, push all required changes including `debian/changelog` update to `vcmi/master` branch
|
- if necessary, push all required changes including `debian/changelog` update to `vcmi/master` branch
|
||||||
- Go to [Sources](https://code.launchpad.net/~vcmi/vcmi/+git/vcmi) and run repository import.
|
- Go to [Sources](https://code.launchpad.net/~vcmi/vcmi/+git/vcmi) and run repository import.
|
||||||
- Wait for import to finish, which usually happens within a minute. Press F5 to actually see changes.
|
- Wait for import to finish, which usually happens within a minute. Press F5 to actually see changes.
|
||||||
@ -37,8 +47,10 @@
|
|||||||
- Wait for builds to finish. This takes quite a while, usually - over a hour, even more for arm builds
|
- Wait for builds to finish. This takes quite a while, usually - over a hour, even more for arm builds
|
||||||
- Once built, all successfully built packages are automatically copied to PPA linked to the recipe
|
- Once built, all successfully built packages are automatically copied to PPA linked to the recipe
|
||||||
- If any of builds have failed, open page with build info and check logs.
|
- If any of builds have failed, open page with build info and check logs.
|
||||||
|
|
||||||
## People with access
|
## People with access
|
||||||
- [alexvins](https://github.com/alexvins) (https://launchpad.net/~alexvins)
|
|
||||||
- [ArseniyShestakov](https://github.com/ArseniyShestakov) (https://launchpad.net/~sxx)
|
- [alexvins](https://github.com/alexvins) (<https://launchpad.net/~alexvins>)
|
||||||
- [IvanSavenko](https://github.com/IvanSavenko) (https://launchpad.net/~saven-ivan)
|
- [ArseniyShestakov](https://github.com/ArseniyShestakov) (<https://launchpad.net/~sxx>)
|
||||||
- (Not member of VCMI, creator of PPA) (https://launchpad.net/~mantas)
|
- [IvanSavenko](https://github.com/IvanSavenko) (<https://launchpad.net/~saven-ivan>)
|
||||||
|
- (Not member of VCMI, creator of PPA) (<https://launchpad.net/~mantas>)
|
||||||
|
@ -8,7 +8,7 @@ VCMI allows overriding HoMM3 .def files with .json replacement. Compared to .def
|
|||||||
|
|
||||||
## Format description
|
## Format description
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
// Base path of all images in animation. Optional.
|
// Base path of all images in animation. Optional.
|
||||||
// Can be used to avoid using long path to images
|
// Can be used to avoid using long path to images
|
||||||
@ -58,12 +58,14 @@ VCMI allows overriding HoMM3 .def files with .json replacement. Compared to .def
|
|||||||
### Replacing a button
|
### Replacing a button
|
||||||
|
|
||||||
This json file will allow replacing .def file for a button with png images. Buttons require following images:
|
This json file will allow replacing .def file for a button with png images. Buttons require following images:
|
||||||
|
|
||||||
1. Active state. Button is active and can be pressed by player
|
1. Active state. Button is active and can be pressed by player
|
||||||
2. Pressed state. Player pressed button but have not released it yet
|
2. Pressed state. Player pressed button but have not released it yet
|
||||||
3. Blocked state. Button is blocked and can not be interacted with. Note that some buttons are never blocked and can be used without this image
|
3. Blocked state. Button is blocked and can not be interacted with. Note that some buttons are never blocked and can be used without this image
|
||||||
4. Highlighted state. This state is used by only some buttons and only in some cases. For example, in main menu buttons will appear highlighted when mouse cursor is on top of the image. Another example is buttons that can be selected, such as settings that can be toggled on or off
|
4. Highlighted state. This state is used by only some buttons and only in some cases. For example, in main menu buttons will appear highlighted when mouse cursor is on top of the image. Another example is buttons that can be selected, such as settings that can be toggled on or off
|
||||||
|
|
||||||
```javascript
|
```json5
|
||||||
|
{
|
||||||
"basepath" : "interface/MyButton", // all images are located in this directory
|
"basepath" : "interface/MyButton", // all images are located in this directory
|
||||||
|
|
||||||
"images" :
|
"images" :
|
||||||
@ -80,7 +82,8 @@ This json file will allow replacing .def file for a button with png images. Butt
|
|||||||
|
|
||||||
This json file allows defining one animation sequence, for example for adventure map objects or for town buildings.
|
This json file allows defining one animation sequence, for example for adventure map objects or for town buildings.
|
||||||
|
|
||||||
```javascript
|
```json5
|
||||||
|
{
|
||||||
"basepath" : "myTown/myBuilding", // all images are located in this directory
|
"basepath" : "myTown/myBuilding", // all images are located in this directory
|
||||||
|
|
||||||
"sequences" :
|
"sequences" :
|
||||||
|
@ -15,7 +15,7 @@ The limiters take no parameters:
|
|||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"limiters" : [ "SHOOTER_ONLY" ]
|
"limiters" : [ "SHOOTER_ONLY" ]
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -30,7 +30,7 @@ Parameters:
|
|||||||
- (optional) bonus sourceType and sourceId in struct
|
- (optional) bonus sourceType and sourceId in struct
|
||||||
- example: (from Adele's bless):
|
- example: (from Adele's bless):
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"limiters" : [
|
"limiters" : [
|
||||||
{
|
{
|
||||||
"type" : "HAS_ANOTHER_BONUS_LIMITER",
|
"type" : "HAS_ANOTHER_BONUS_LIMITER",
|
||||||
@ -64,6 +64,7 @@ Parameters:
|
|||||||
If parameters is empty, level limiter works as CREATURES_ONLY limiter
|
If parameters is empty, level limiter works as CREATURES_ONLY limiter
|
||||||
|
|
||||||
Parameters:
|
Parameters:
|
||||||
|
|
||||||
- Minimal level
|
- Minimal level
|
||||||
- Maximal level
|
- Maximal level
|
||||||
|
|
||||||
@ -81,14 +82,14 @@ Parameters:
|
|||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"limiters": [ {
|
"limiters": [ {
|
||||||
"type":"CREATURE_TYPE_LIMITER",
|
"type":"CREATURE_TYPE_LIMITER",
|
||||||
"parameters": [ "angel", true ]
|
"parameters": [ "angel", true ]
|
||||||
} ],
|
} ],
|
||||||
```
|
```
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"limiters" : [ {
|
"limiters" : [ {
|
||||||
"type" : "CREATURE_TERRAIN_LIMITER",
|
"type" : "CREATURE_TERRAIN_LIMITER",
|
||||||
"parameters" : ["sand"]
|
"parameters" : ["sand"]
|
||||||
@ -112,7 +113,7 @@ and operate on the remaining limiters in that list:
|
|||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"limiters" : [
|
"limiters" : [
|
||||||
"noneOf",
|
"noneOf",
|
||||||
"IS_UNDEAD",
|
"IS_UNDEAD",
|
||||||
|
@ -128,7 +128,7 @@ Allows to raise different creatures than Skeletons after battle.
|
|||||||
- addInfo: Level of Necromancy secondary skill (1 - Basic, 3 - Expert)
|
- addInfo: Level of Necromancy secondary skill (1 - Basic, 3 - Expert)
|
||||||
- Example (from Cloak Of The Undead King):
|
- Example (from Cloak Of The Undead King):
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
"type" : "IMPROVED_NECROMANCY",
|
"type" : "IMPROVED_NECROMANCY",
|
||||||
"subtype" : "creature.walkingDead",
|
"subtype" : "creature.walkingDead",
|
||||||
@ -256,7 +256,7 @@ Gives creature under effect of this spell additional bonus, which is hardcoded a
|
|||||||
|
|
||||||
Modifies 'val' parameter of spell effects that give bonuses by specified value. For example, Aenain makes Disrupting Ray decrease target's defense by additional 2 points:
|
Modifies 'val' parameter of spell effects that give bonuses by specified value. For example, Aenain makes Disrupting Ray decrease target's defense by additional 2 points:
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"disruptingRay" : {
|
"disruptingRay" : {
|
||||||
"addInfo" : -2,
|
"addInfo" : -2,
|
||||||
"subtype" : "spell.disruptingRay",
|
"subtype" : "spell.disruptingRay",
|
||||||
@ -271,7 +271,7 @@ Modifies 'val' parameter of spell effects that give bonuses by specified value.
|
|||||||
|
|
||||||
Changes 'val' parameter of spell effects that give bonuses to a specified value. For example, Fortune cast by Melody always modifies luck by +3:
|
Changes 'val' parameter of spell effects that give bonuses to a specified value. For example, Fortune cast by Melody always modifies luck by +3:
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"fortune" : {
|
"fortune" : {
|
||||||
"addInfo" : 3,
|
"addInfo" : 3,
|
||||||
"subtype" : "spell.fortune",
|
"subtype" : "spell.fortune",
|
||||||
@ -669,6 +669,7 @@ Affected unit can attack walls during siege battles (Cyclops)
|
|||||||
### CATAPULT_EXTRA_SHOTS
|
### CATAPULT_EXTRA_SHOTS
|
||||||
|
|
||||||
Defines spell mastery level for spell used by CATAPULT bonus
|
Defines spell mastery level for spell used by CATAPULT bonus
|
||||||
|
|
||||||
- subtype: affected spell
|
- subtype: affected spell
|
||||||
- val: spell mastery level to use
|
- val: spell mastery level to use
|
||||||
|
|
||||||
@ -842,7 +843,7 @@ Determines how many times per combat affected creature can cast its targeted spe
|
|||||||
|
|
||||||
### CREATURE_ENCHANT_POWER
|
### CREATURE_ENCHANT_POWER
|
||||||
|
|
||||||
- val: Total duration of spells cast by creature, in turns
|
- val: Total duration of spells cast by creature, in turns
|
||||||
|
|
||||||
### REBIRTH
|
### REBIRTH
|
||||||
|
|
||||||
|
@ -12,25 +12,27 @@ Check the files in *config/heroes/* for additional usage examples.
|
|||||||
|
|
||||||
- Type: Complex
|
- Type: Complex
|
||||||
- Parameters: valPer20, stepSize=1
|
- Parameters: valPer20, stepSize=1
|
||||||
- Effect: Updates val to
|
- Effect: Updates val to `ceil(valPer20 * floor(heroLevel / stepSize) / 20)`
|
||||||
|
|
||||||
` ceil(valPer20 * floor(heroLevel / stepSize) / 20)`
|
|
||||||
|
|
||||||
Example: The following updater will cause a bonus to grow by 6 for every
|
Example: The following updater will cause a bonus to grow by 6 for every
|
||||||
40 levels. At first level, rounding will cause the bonus to be 0.
|
40 levels. At first level, rounding will cause the bonus to be 0.
|
||||||
|
|
||||||
` "updater" : {`
|
```json5
|
||||||
` "parameters" : [ 6, 2 ],`
|
"updater" : {
|
||||||
` "type" : "GROWS_WITH_LEVEL"`
|
"parameters" : [ 6, 2 ],
|
||||||
` }`
|
"type" : "GROWS_WITH_LEVEL"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
Example: The following updater will cause a bonus to grow by 3 for every
|
Example: The following updater will cause a bonus to grow by 3 for every
|
||||||
20 levels. At first level, rounding will cause the bonus to be 1.
|
20 levels. At first level, rounding will cause the bonus to be 1.
|
||||||
|
|
||||||
` "updater" : {`
|
```json5
|
||||||
` "parameters" : [ 3 ],`
|
"updater" : {
|
||||||
` "type" : "GROWS_WITH_LEVEL"`
|
"parameters" : [ 3 ],
|
||||||
` }`
|
"type" : "GROWS_WITH_LEVEL"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
Remarks:
|
Remarks:
|
||||||
|
|
||||||
@ -42,13 +44,9 @@ Remarks:
|
|||||||
## TIMES_HERO_LEVEL
|
## TIMES_HERO_LEVEL
|
||||||
|
|
||||||
- Type: Simple
|
- Type: Simple
|
||||||
- Effect: Updates val to
|
- Effect: Updates val to `val * heroLevel`
|
||||||
|
|
||||||
` val * heroLevel`
|
Usage: `"updater" : "TIMES_HERO_LEVEL"`
|
||||||
|
|
||||||
Usage:
|
|
||||||
|
|
||||||
` "updater" : "TIMES_HERO_LEVEL"`
|
|
||||||
|
|
||||||
Remark: This updater is redundant, in the sense that GROWS_WITH_LEVEL
|
Remark: This updater is redundant, in the sense that GROWS_WITH_LEVEL
|
||||||
can also express the desired scaling by setting valPer20 to 20\*val. It
|
can also express the desired scaling by setting valPer20 to 20\*val. It
|
||||||
@ -57,32 +55,28 @@ has been added for convenience.
|
|||||||
## TIMES_STACK_LEVEL
|
## TIMES_STACK_LEVEL
|
||||||
|
|
||||||
- Type: Simple
|
- Type: Simple
|
||||||
- Effect: Updates val to
|
- Effect: Updates val to `val * stackLevel`
|
||||||
|
|
||||||
` val * stackLevel`
|
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
` "updater" : "TIMES_STACK_LEVEL"`
|
`"updater" : "TIMES_STACK_LEVEL"`
|
||||||
|
|
||||||
Remark: The stack level for war machines is 0.
|
Remark: The stack level for war machines is 0.
|
||||||
|
|
||||||
## ARMY_MOVEMENT
|
## ARMY_MOVEMENT
|
||||||
|
|
||||||
- Type: Complex
|
- Type: Complex
|
||||||
- Parameters: basePerSpeed, dividePerSpeed, additionalMultiplier,
|
- Parameters: basePerSpeed, dividePerSpeed, additionalMultiplier, maxValue
|
||||||
maxValue
|
- Effect: Updates val to `val+= max((floor(basePerSpeed / dividePerSpeed) * additionalMultiplier), maxValue)`
|
||||||
- Effect: Updates val to val+= max((floor(basePerSpeed /
|
- Remark: this updater is designed for MOVEMENT bonus to match H3 army movement rules (in the example - actual movement updater, which produces values same as in default movement.txt).
|
||||||
dividePerSpeed)\* additionalMultiplier), maxValue)
|
|
||||||
- Remark: this updater is designed for MOVEMENT bonus to match H3 army
|
|
||||||
movement rules (in the example - actual movement updater, which
|
|
||||||
produces values same as in default movement.txt).
|
|
||||||
- Example:
|
- Example:
|
||||||
|
|
||||||
` "updater" : {`
|
```json5
|
||||||
` "parameters" : [ 20, 3, 10, 700 ],`
|
"updater" : {
|
||||||
` "type" : "ARMY_MOVEMENT"`
|
"parameters" : [ 20, 3, 10, 700 ],
|
||||||
` }`
|
"type" : "ARMY_MOVEMENT"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
## BONUS_OWNER_UPDATER
|
## BONUS_OWNER_UPDATER
|
||||||
|
|
||||||
|
@ -3,13 +3,18 @@
|
|||||||
Total value of Bonus is calculated using the following:
|
Total value of Bonus is calculated using the following:
|
||||||
|
|
||||||
- For each bonus source type we calculate new source value (for all bonus value types except PERCENT_TO_SOURCE and PERCENT_TO_TARGET_TYPE) using the following:
|
- For each bonus source type we calculate new source value (for all bonus value types except PERCENT_TO_SOURCE and PERCENT_TO_TARGET_TYPE) using the following:
|
||||||
` newVal = (val * (100 + PERCENT_TO_SOURCE) / 100))`
|
|
||||||
|
```
|
||||||
|
newVal = (val * (100 + PERCENT_TO_SOURCE) / 100))
|
||||||
|
```
|
||||||
|
|
||||||
- PERCENT_TO_TARGET_TYPE applies as PERCENT_TO_SOURCE to targetSourceType of bonus.
|
- PERCENT_TO_TARGET_TYPE applies as PERCENT_TO_SOURCE to targetSourceType of bonus.
|
||||||
|
|
||||||
- All bonus value types summarized and then used as subject of the following formula:
|
- All bonus value types summarized and then used as subject of the following formula:
|
||||||
|
|
||||||
` clamp(((BASE_NUMBER * (100 + PERCENT_TO_BASE) / 100) + ADDITIVE_VALUE) * (100 + PERCENT_TO_ALL) / 100), INDEPENDENT_MAX, INDEPENDENT_MIN)`
|
```
|
||||||
|
clamp(((BASE_NUMBER * (100 + PERCENT_TO_BASE) / 100) + ADDITIVE_VALUE) * (100 + PERCENT_TO_ALL) / 100), INDEPENDENT_MAX, INDEPENDENT_MIN)
|
||||||
|
```
|
||||||
|
|
||||||
Semantics of INDEPENDENT_MAX and INDEPENDENT_MIN are wrapped, and first means than bonus total value will be at least INDEPENDENT_MAX, and second means than bonus value will be at most INDEPENDENT_MIN.
|
Semantics of INDEPENDENT_MAX and INDEPENDENT_MIN are wrapped, and first means than bonus total value will be at least INDEPENDENT_MAX, and second means than bonus value will be at most INDEPENDENT_MIN.
|
||||||
|
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
All parameters but type are optional.
|
All parameters but type are optional.
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
// Type of the bonus. See Bonus Types for full list
|
// Type of the bonus. See Bonus Types for full list
|
||||||
"type": "BONUS_TYPE",
|
"type": "BONUS_TYPE",
|
||||||
@ -81,7 +81,7 @@ See [Game Identifiers](Game_Identifiers.md) for full list of available identifie
|
|||||||
|
|
||||||
### Example
|
### Example
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"bonus" :
|
"bonus" :
|
||||||
{
|
{
|
||||||
"type" : "HATE",
|
"type" : "HATE",
|
||||||
|
@ -22,7 +22,7 @@ Includes:
|
|||||||
|
|
||||||
Function of all of these objects can be enabled by this:
|
Function of all of these objects can be enabled by this:
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"function" : "castleGates"
|
"function" : "castleGates"
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -58,31 +58,31 @@ CBuilding class.
|
|||||||
|
|
||||||
#### unlock guild level
|
#### unlock guild level
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"guildLevels" : 1
|
"guildLevels" : 1
|
||||||
```
|
```
|
||||||
|
|
||||||
#### unlock hero recruitment
|
#### unlock hero recruitment
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"allowsHeroPurchase" : true
|
"allowsHeroPurchase" : true
|
||||||
```
|
```
|
||||||
|
|
||||||
#### unlock ship purchase
|
#### unlock ship purchase
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"allowsShipPurchase" : true
|
"allowsShipPurchase" : true
|
||||||
```
|
```
|
||||||
|
|
||||||
#### unlock building purchase
|
#### unlock building purchase
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"allowsBuildingPurchase" : true
|
"allowsBuildingPurchase" : true
|
||||||
```
|
```
|
||||||
|
|
||||||
#### unlocks creatures
|
#### unlocks creatures
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"dwelling" : { "level" : 1, "creature" : "archer" }
|
"dwelling" : { "level" : 1, "creature" : "archer" }
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -92,31 +92,31 @@ Turn into town bonus? What about creature-specific bonuses from hordes?
|
|||||||
|
|
||||||
#### gives resources
|
#### gives resources
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"provides" : { "gold" : 500 }
|
"provides" : { "gold" : 500 }
|
||||||
```
|
```
|
||||||
|
|
||||||
#### gives guild spells
|
#### gives guild spells
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"guildSpells" : [5, 0, 0, 0, 0]
|
"guildSpells" : [5, 0, 0, 0, 0]
|
||||||
```
|
```
|
||||||
|
|
||||||
#### gives thieves guild
|
#### gives thieves guild
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"thievesGuildLevels" : 1
|
"thievesGuildLevels" : 1
|
||||||
```
|
```
|
||||||
|
|
||||||
#### gives fortifications
|
#### gives fortifications
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"fortificationLevels" : 1
|
"fortificationLevels" : 1
|
||||||
```
|
```
|
||||||
|
|
||||||
#### gives war machine
|
#### gives war machine
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"warMachine" : "ballista"
|
"warMachine" : "ballista"
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -134,7 +134,7 @@ Includes:
|
|||||||
- bonus to scouting range
|
- bonus to scouting range
|
||||||
- bonus to player
|
- bonus to player
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"bonuses" :
|
"bonuses" :
|
||||||
{
|
{
|
||||||
"moraleToDefenders" :
|
"moraleToDefenders" :
|
||||||
@ -162,12 +162,12 @@ Possible issue - with removing of fixed ID's buildings in different town
|
|||||||
may no longer share same ID. However Capitol must be unique across all
|
may no longer share same ID. However Capitol must be unique across all
|
||||||
town. Should be fixed somehow.
|
town. Should be fixed somehow.
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"onePerPlayer" : true
|
"onePerPlayer" : true
|
||||||
```
|
```
|
||||||
|
|
||||||
#### chance to be built on start
|
#### chance to be built on start
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"prebuiltChance" : 75
|
"prebuiltChance" : 75
|
||||||
```
|
```
|
@ -3,12 +3,13 @@
|
|||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Starting from version 1.3, VCMI supports its own campaign format.
|
Starting from version 1.3, VCMI supports its own campaign format.
|
||||||
Campaigns have *.vcmp file format and it consists from campaign json and set of scenarios (can be both *.vmap and *.h3m)
|
Campaigns have `*.vcmp` file format and it consists from campaign json and set of scenarios (can be both `*.vmap` and `*.h3m`)
|
||||||
|
|
||||||
To start making campaign, create file named `header.json`. See also [Packing campaign](#packing-campaign)
|
To start making campaign, create file named `header.json`. See also [Packing campaign](#packing-campaign)
|
||||||
|
|
||||||
Basic structure of this file is here, each section is described in details below
|
Basic structure of this file is here, each section is described in details below
|
||||||
```js
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"version" : 1,
|
"version" : 1,
|
||||||
|
|
||||||
@ -32,7 +33,8 @@ Basic structure of this file is here, each section is described in details below
|
|||||||
## Header properties
|
## Header properties
|
||||||
|
|
||||||
In header are parameters describing campaign properties
|
In header are parameters describing campaign properties
|
||||||
```js
|
|
||||||
|
```json5
|
||||||
...
|
...
|
||||||
"regions": {...},
|
"regions": {...},
|
||||||
"name": "Campaign name",
|
"name": "Campaign name",
|
||||||
@ -60,7 +62,8 @@ In header are parameters describing campaign properties
|
|||||||
## Scenario description
|
## Scenario description
|
||||||
|
|
||||||
Scenario description looks like follow:
|
Scenario description looks like follow:
|
||||||
```js
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"map": "maps/SomeMap",
|
"map": "maps/SomeMap",
|
||||||
"preconditions": [],
|
"preconditions": [],
|
||||||
@ -77,7 +80,7 @@ Scenario description looks like follow:
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
- `"map"` map name without extension but with relative path. Both *.h3m and *.vmap maps are supported. If you will pack scenarios inside campaign, numerical map name should be used, see details in [packing campaign](#packing-campaign)
|
- `"map"` map name without extension but with relative path. Both `*.h3m` and `*.vmap` maps are supported. If you will pack scenarios inside campaign, numerical map name should be used, see details in [packing campaign](#packing-campaign)
|
||||||
- `"preconditions"` enumerate scenarios indexes which must be completed to unlock this scenario. For example, if you want to make sequential missions, you should specify `"preconditions": []` for first scenario, but for second scenario it should be `"preconditions": [0]` and for third `"preconditions": [0, 1]`. But you can allow non-linear conquering using this parameter
|
- `"preconditions"` enumerate scenarios indexes which must be completed to unlock this scenario. For example, if you want to make sequential missions, you should specify `"preconditions": []` for first scenario, but for second scenario it should be `"preconditions": [0]` and for third `"preconditions": [0, 1]`. But you can allow non-linear conquering using this parameter
|
||||||
- `"color"` defines color id for the region. Possible values are `0: red, 1: blue, tan: 2, green: 3, orange: 4, purple: 5, teal: 6, pink: 7`
|
- `"color"` defines color id for the region. Possible values are `0: red, 1: blue, tan: 2, green: 3, orange: 4, purple: 5, teal: 6, pink: 7`
|
||||||
- `"difficulty"` sets initial difficulty for this scenario. If `"allowDifficultySelection"`is defined for campaign, difficulty may be changed by player. Possible values are `0: pawn, 1: knight, 2: rook, 3: queen, 4: king`
|
- `"difficulty"` sets initial difficulty for this scenario. If `"allowDifficultySelection"`is defined for campaign, difficulty may be changed by player. Possible values are `0: pawn, 1: knight, 2: rook, 3: queen, 4: king`
|
||||||
@ -96,7 +99,8 @@ Scenario description looks like follow:
|
|||||||
### Prolog/Epilog
|
### Prolog/Epilog
|
||||||
|
|
||||||
Prolog and epilog properties are optional
|
Prolog and epilog properties are optional
|
||||||
```js
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"video": "NEUTRALA.smk", //video to show
|
"video": "NEUTRALA.smk", //video to show
|
||||||
"music": "musicFile.ogg", //music to play, should be located in music directory
|
"music": "musicFile.ogg", //music to play, should be located in music directory
|
||||||
@ -115,7 +119,7 @@ If `startOptions` is `none`, `bonuses` field will be ignored
|
|||||||
|
|
||||||
If `startOptions` is `bonus`, bonus format may vary depending on its type.
|
If `startOptions` is `bonus`, bonus format may vary depending on its type.
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
{
|
{
|
||||||
"what": "",
|
"what": "",
|
||||||
|
|
||||||
@ -158,19 +162,21 @@ If `startOptions` is `bonus`, bonus format may vary depending on its type.
|
|||||||
|
|
||||||
If `startOptions` is `crossover`, heroes from specific scenario will be moved to this scenario. Bonus format is following
|
If `startOptions` is `crossover`, heroes from specific scenario will be moved to this scenario. Bonus format is following
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
{
|
{
|
||||||
"playerColor": 0,
|
"playerColor": 0,
|
||||||
"scenario": 0
|
"scenario": 0
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
- `"playerColor"` from what player color heroes shall be taken. Possible values are `0: red, 1: blue, tan: 2, green: 3, orange: 4, purple: 5, teal: 6, pink: 7`
|
- `"playerColor"` from what player color heroes shall be taken. Possible values are `0: red, 1: blue, tan: 2, green: 3, orange: 4, purple: 5, teal: 6, pink: 7`
|
||||||
- `"scenario"` from which scenario heroes shall be taken. 0 means first scenario
|
- `"scenario"` from which scenario heroes shall be taken. 0 means first scenario
|
||||||
|
|
||||||
#### Hero start option
|
#### Hero start option
|
||||||
|
|
||||||
If `startOptions` is `hero`, hero can be chosen as a starting bonus. Bonus format is following
|
If `startOptions` is `hero`, hero can be chosen as a starting bonus. Bonus format is following
|
||||||
```js
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"playerColor": 0,
|
"playerColor": 0,
|
||||||
"hero": "random"
|
"hero": "random"
|
||||||
@ -184,7 +190,7 @@ If `startOptions` is `hero`, hero can be chosen as a starting bonus. Bonus forma
|
|||||||
|
|
||||||
Predefined campaign regions are located in file `campaign_regions.json`
|
Predefined campaign regions are located in file `campaign_regions.json`
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
{
|
{
|
||||||
"background": "ownRegionBackground.png",
|
"background": "ownRegionBackground.png",
|
||||||
"suffix": ["Enabled", "Selected", "Conquered"],
|
"suffix": ["Enabled", "Selected", "Conquered"],
|
||||||
@ -213,6 +219,7 @@ This file is a zip archive.
|
|||||||
The scenarios should be named as in `"map"` field from header. Subfolders are allowed.
|
The scenarios should be named as in `"map"` field from header. Subfolders are allowed.
|
||||||
|
|
||||||
## Compatibility table
|
## Compatibility table
|
||||||
|
|
||||||
| Version | Min VCMI | Max VCMI | Description |
|
| Version | Min VCMI | Max VCMI | Description |
|
||||||
|---------|----------|----------|-------------|
|
|---------|----------|----------|-------------|
|
||||||
| 1 | 1.3 | | Initial release |
|
| 1 | 1.3 | | Initial release |
|
@ -21,6 +21,7 @@ In this tutorial we will recreate options tab to support chess timers UI.
|
|||||||
### Creating mod structure
|
### Creating mod structure
|
||||||
|
|
||||||
To start making mod, create following folders structure;
|
To start making mod, create following folders structure;
|
||||||
|
|
||||||
```
|
```
|
||||||
extendedLobby/
|
extendedLobby/
|
||||||
|- content/
|
|- content/
|
||||||
@ -31,7 +32,8 @@ extendedLobby/
|
|||||||
```
|
```
|
||||||
|
|
||||||
File `mod.json` is generic and could look like this:
|
File `mod.json` is generic and could look like this:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"name" : "Configurable UI tutorial mod",
|
"name" : "Configurable UI tutorial mod",
|
||||||
"description" : "See tutorial here https://github.com/vcmi/vcmi/wiki/Configurable-UI-widgets",
|
"description" : "See tutorial here https://github.com/vcmi/vcmi/wiki/Configurable-UI-widgets",
|
||||||
@ -44,7 +46,7 @@ File `mod.json` is generic and could look like this:
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
After that you can copy `extendedLobby/ folder to `mods/` folder and your mod will immediately appear in launcher but it does nothing for now.
|
After that you can copy `extendedLobby/` folder to `mods/` folder and your mod will immediately appear in launcher but it does nothing for now.
|
||||||
|
|
||||||
### Making layout for timer
|
### Making layout for timer
|
||||||
|
|
||||||
@ -64,7 +66,8 @@ So we need to modify turn duration label and add combo box with timer types
|
|||||||
Open `optionsTab.json` and scroll it until you see comment `timer`. Three elements after this comment are related to timer.
|
Open `optionsTab.json` and scroll it until you see comment `timer`. Three elements after this comment are related to timer.
|
||||||
|
|
||||||
Let's find first element, which is label
|
Let's find first element, which is label
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"items"
|
"items"
|
||||||
[
|
[
|
||||||
@ -85,7 +88,8 @@ Let's find first element, which is label
|
|||||||
```
|
```
|
||||||
|
|
||||||
And modify it a bit
|
And modify it a bit
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"name": "labelTimer", //add name, only for convenience
|
"name": "labelTimer", //add name, only for convenience
|
||||||
"type": "label",
|
"type": "label",
|
||||||
@ -98,7 +102,8 @@ And modify it a bit
|
|||||||
```
|
```
|
||||||
|
|
||||||
But we also need proper background image for this label. Add image widget BEFORE labelTimer widget:
|
But we also need proper background image for this label. Add image widget BEFORE labelTimer widget:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"type": "picture",
|
"type": "picture",
|
||||||
"image": "RmgTTBk",
|
"image": "RmgTTBk",
|
||||||
@ -109,6 +114,7 @@ But we also need proper background image for this label. Add image widget BEFORE
|
|||||||
...
|
...
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
In order to make it work, add file `RmgTTBk.bmp` to `content/sprites/`
|
In order to make it work, add file `RmgTTBk.bmp` to `content/sprites/`
|
||||||
|
|
||||||
Elements named `labelTurnDurationValue` and `sliderTurnDuration` we will keep without change - they are needed to configure classic timer.
|
Elements named `labelTurnDurationValue` and `sliderTurnDuration` we will keep without change - they are needed to configure classic timer.
|
||||||
@ -121,7 +127,7 @@ Copy image `DrDoCoBk.bmp` to `content/sprites/`. Button objects use animated ima
|
|||||||
For normal, pressed, blocked and highlighted. Our combo box inherits this behavior, so let's convert image to animation.
|
For normal, pressed, blocked and highlighted. Our combo box inherits this behavior, so let's convert image to animation.
|
||||||
In order to do it, we need to create file `DrDoCoBk.json` in same folder `content/sprites/` with following content:
|
In order to do it, we need to create file `DrDoCoBk.json` in same folder `content/sprites/` with following content:
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
{
|
{
|
||||||
"sequences" :
|
"sequences" :
|
||||||
[
|
[
|
||||||
@ -140,7 +146,7 @@ Thus we created file with animation, containing single frame which can be used f
|
|||||||
|
|
||||||
Let's add one more element after `//timer` comment:
|
Let's add one more element after `//timer` comment:
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
...
|
...
|
||||||
//timer
|
//timer
|
||||||
{
|
{
|
||||||
@ -157,7 +163,7 @@ Let's add one more element after `//timer` comment:
|
|||||||
|
|
||||||
We also want to have label on the top of this combo box showing which element is selected. You need to add `items` array, where additional elements can be specified, label in our case:
|
We also want to have label on the top of this combo box showing which element is selected. You need to add `items` array, where additional elements can be specified, label in our case:
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
...
|
...
|
||||||
//timer
|
//timer
|
||||||
{
|
{
|
||||||
@ -189,7 +195,7 @@ First of all, add images to `content/sprites/` folder: `List2Bk.bmp` for drop-do
|
|||||||
|
|
||||||
Now specify items inside `dropDown` field
|
Now specify items inside `dropDown` field
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"dropDown":
|
"dropDown":
|
||||||
{
|
{
|
||||||
"items":
|
"items":
|
||||||
@ -272,7 +278,8 @@ After view part is done, let's make behavioural part.
|
|||||||
Let's hide elements, related to classic timer when chess timer is selected and show them back if classic selected.
|
Let's hide elements, related to classic timer when chess timer is selected and show them back if classic selected.
|
||||||
|
|
||||||
To do that, find `"variables"` part inside `optionsTab.json` and add there `"timers"` array, containing 2 elements:
|
To do that, find `"variables"` part inside `optionsTab.json` and add there `"timers"` array, containing 2 elements:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
"variables":
|
"variables":
|
||||||
{
|
{
|
||||||
"timers":
|
"timers":
|
||||||
@ -303,7 +310,7 @@ Now we show and hide elements, but visually you still can some "artifacts":
|
|||||||
It's because options tab background image we use has those elements drawn. Let's hide them with overlay image `timchebk.bmp`.
|
It's because options tab background image we use has those elements drawn. Let's hide them with overlay image `timchebk.bmp`.
|
||||||
It should be drawn before all other timer elements:
|
It should be drawn before all other timer elements:
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
...
|
...
|
||||||
// timer
|
// timer
|
||||||
{
|
{
|
||||||
@ -328,9 +335,10 @@ It works and can switch elements, the only missing part is chess timer configura
|
|||||||
|
|
||||||
We should add text input fields, to specify different timers. We will use background for them `timerField.bmp`, copy it to `content/sprites/` folder of your mod.
|
We should add text input fields, to specify different timers. We will use background for them `timerField.bmp`, copy it to `content/sprites/` folder of your mod.
|
||||||
|
|
||||||
There are 4 different timers: base, turn, battle and creature. Read about them here: https://github.com/vcmi/vcmi/issues/1364
|
There are 4 different timers: base, turn, battle and creature. Read about them here: <https://github.com/vcmi/vcmi/issues/1364>
|
||||||
We can add editors for them into items list, their format will be following:
|
We can add editors for them into items list, their format will be following:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"name": "chessFieldBase",
|
"name": "chessFieldBase",
|
||||||
"type": "textInput",
|
"type": "textInput",
|
||||||
@ -345,6 +353,7 @@ We can add editors for them into items list, their format will be following:
|
|||||||
```
|
```
|
||||||
|
|
||||||
Add three remaining elements for different timers by yourself. You can play with all settings, except callback. There are 4 predefined callbacks to setup timers:
|
Add three remaining elements for different timers by yourself. You can play with all settings, except callback. There are 4 predefined callbacks to setup timers:
|
||||||
|
|
||||||
- `parseAndSetTimer_base`
|
- `parseAndSetTimer_base`
|
||||||
- `parseAndSetTimer_turn`
|
- `parseAndSetTimer_turn`
|
||||||
- `parseAndSetTimer_battle`
|
- `parseAndSetTimer_battle`
|
||||||
@ -363,7 +372,7 @@ There are different basic types, which can be used as value.
|
|||||||
|
|
||||||
#### Primitive types
|
#### Primitive types
|
||||||
|
|
||||||
Read JSON documentation for primitive types description: https://www.json.org/json-en.html
|
Read JSON documentation for primitive types description: <https://www.json.org/json-en.html>
|
||||||
|
|
||||||
#### Text
|
#### Text
|
||||||
|
|
||||||
@ -430,7 +439,8 @@ One of predefined values:
|
|||||||
### Configurable objects
|
### Configurable objects
|
||||||
|
|
||||||
Configurable object has following structure:
|
Configurable object has following structure:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"items": [],
|
"items": [],
|
||||||
"variables": {}, //optional
|
"variables": {}, //optional
|
||||||
@ -748,11 +758,13 @@ Used only as special object for [combo box](#combo-box)
|
|||||||
`"position"`: [position](#position)
|
`"position"`: [position](#position)
|
||||||
|
|
||||||
`"items": []` array of overlay widgets with certain types and names:
|
`"items": []` array of overlay widgets with certain types and names:
|
||||||
- `"name": "hoverImage"`, `"type": ` [picture](#picture) - image to be shown when cursor hovers elements
|
|
||||||
- `"name": "labelName"`, `"type": ` [label](#label) - element caption
|
- `"name": "hoverImage"`, `"type":` [picture](#picture) - image to be shown when cursor hovers elements
|
||||||
|
- `"name": "labelName"`, `"type":` [label](#label) - element caption
|
||||||
|
|
||||||
**Callbacks**
|
**Callbacks**
|
||||||
- `sliderMove` connect to slider callback to correctly navigate over elements
|
|
||||||
|
- `sliderMove` connect to slider callback to correctly navigate over elements
|
||||||
|
|
||||||
#### Layout
|
#### Layout
|
||||||
|
|
||||||
@ -769,7 +781,8 @@ Used only as special object for [combo box](#combo-box)
|
|||||||
While designing a new element, you can make it configurable to reuse all functionality described above. It will provide flexibility to further changes as well as modding capabilities.
|
While designing a new element, you can make it configurable to reuse all functionality described above. It will provide flexibility to further changes as well as modding capabilities.
|
||||||
|
|
||||||
Class should inherit `InterfaceObjectConfigurable`.
|
Class should inherit `InterfaceObjectConfigurable`.
|
||||||
```C++
|
|
||||||
|
```cpp
|
||||||
#include "gui/InterfaceObjectConfigurable.h" //assuming we are in client folder
|
#include "gui/InterfaceObjectConfigurable.h" //assuming we are in client folder
|
||||||
|
|
||||||
class MyYesNoDialog: public InterfaceObjectConfigurable
|
class MyYesNoDialog: public InterfaceObjectConfigurable
|
||||||
@ -781,7 +794,7 @@ class MyYesNoDialog: public InterfaceObjectConfigurable
|
|||||||
|
|
||||||
To make new object work, it's sufficient to define constructor, which receives const reference to `JsonNode`.
|
To make new object work, it's sufficient to define constructor, which receives const reference to `JsonNode`.
|
||||||
|
|
||||||
```C++
|
```cpp
|
||||||
MyYesNoDialog::MyYesNoDialog(const JsonNode & config):
|
MyYesNoDialog::MyYesNoDialog(const JsonNode & config):
|
||||||
InterfaceObjectConfigurable(), //you can pass arguments same as for CIntObject
|
InterfaceObjectConfigurable(), //you can pass arguments same as for CIntObject
|
||||||
{
|
{
|
||||||
@ -808,13 +821,13 @@ MyYesNoDialog::MyYesNoDialog(const JsonNode & config):
|
|||||||
|
|
||||||
You can build custom widgets, related to your UI element specifically. Like in example above, there is Item widget, which can be also used on JSON config.
|
You can build custom widgets, related to your UI element specifically. Like in example above, there is Item widget, which can be also used on JSON config.
|
||||||
|
|
||||||
```C++
|
```cpp
|
||||||
REGISTER_BUILDER("myItem", &MyYesNoDialog::buildMyItem);
|
REGISTER_BUILDER("myItem", &MyYesNoDialog::buildMyItem);
|
||||||
```
|
```
|
||||||
|
|
||||||
You have to define function, which takes JsonNode as an argument and return pointer to built widget
|
You have to define function, which takes JsonNode as an argument and return pointer to built widget
|
||||||
|
|
||||||
```C++
|
```cpp
|
||||||
std::shared_ptr<MyYesNoDialog::Item> MyYesNoDialog::buildMyItem(const JsonNode & config)
|
std::shared_ptr<MyYesNoDialog::Item> MyYesNoDialog::buildMyItem(const JsonNode & config)
|
||||||
{
|
{
|
||||||
auto position = readPosition(config["position"]);
|
auto position = readPosition(config["position"]);
|
||||||
@ -824,7 +837,7 @@ std::shared_ptr<MyYesNoDialog::Item> MyYesNoDialog::buildMyItem(const JsonNode &
|
|||||||
|
|
||||||
After that, if your JSON file has items with type "MyItem", the new Item element will be constructed.
|
After that, if your JSON file has items with type "MyItem", the new Item element will be constructed.
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
{
|
{
|
||||||
"items":
|
"items":
|
||||||
[
|
[
|
||||||
@ -840,7 +853,7 @@ After that, if your JSON file has items with type "MyItem", the new Item element
|
|||||||
|
|
||||||
After calling `build(config)` variables defined in config JSON file become available. You can interpret them and use in callbacks or in element code
|
After calling `build(config)` variables defined in config JSON file become available. You can interpret them and use in callbacks or in element code
|
||||||
|
|
||||||
```C++
|
```cpp
|
||||||
build(config);
|
build(config);
|
||||||
|
|
||||||
if(variables["colorfulText"].Bool())
|
if(variables["colorfulText"].Bool())
|
||||||
|
@ -7,7 +7,7 @@ Difficulty configuration is located in [config/difficulty.json](../config/diffic
|
|||||||
|
|
||||||
## Format summary
|
## Format summary
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"human": //parameters impacting human players only
|
"human": //parameters impacting human players only
|
||||||
{
|
{
|
||||||
@ -50,7 +50,7 @@ For both types of bonuses, `source` should be specified as `OTHER`.
|
|||||||
|
|
||||||
## Example
|
## Example
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
{ //will give 150% extra health to all players' creatures if specified in "battleBonuses" array
|
{ //will give 150% extra health to all players' creatures if specified in "battleBonuses" array
|
||||||
"type" : "STACK_HEALTH",
|
"type" : "STACK_HEALTH",
|
||||||
"val" : 150,
|
"val" : 150,
|
||||||
|
@ -12,7 +12,7 @@ In order to make functional artifact you also need:
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
``` jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// Type of this artifact - creature, hero or commander
|
// Type of this artifact - creature, hero or commander
|
||||||
"type": ["HERO", "CREATURE", "COMMANDER"]
|
"type": ["HERO", "CREATURE", "COMMANDER"]
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Battle Obstacle Format
|
# Battle Obstacle Format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
// List of terrains on which this obstacle can be used
|
// List of terrains on which this obstacle can be used
|
||||||
"allowedTerrains" : []
|
"allowedTerrains" : []
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Battlefield Format
|
# Battlefield Format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
// Human-readable name of the battlefield
|
// Human-readable name of the battlefield
|
||||||
"name" : "",
|
"name" : "",
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@ The purpose is to create visually attractive and consistent maps, which will als
|
|||||||
|
|
||||||
If not enough biomes are defined for [terrain type](Terrain_Format.md), map generator will fall back to using all available templates that match this terrain, which was original behavior before 1.5.0.
|
If not enough biomes are defined for [terrain type](Terrain_Format.md), map generator will fall back to using all available templates that match this terrain, which was original behavior before 1.5.0.
|
||||||
|
|
||||||
``` json
|
```json5
|
||||||
"obstacleSetId" : {
|
"obstacleSetId" : {
|
||||||
"biome" : {
|
"biome" : {
|
||||||
"terrain" : "grass", // Id or vector of Ids this obstacle set can spawn at
|
"terrain" : "grass", // Id or vector of Ids this obstacle set can spawn at
|
||||||
@ -38,5 +38,3 @@ Currently algorithm picks randomly:
|
|||||||
- One or two sets of **rocks** (small objects)
|
- One or two sets of **rocks** (small objects)
|
||||||
- One of each remaining types of object (**structure**, **animal**, **other**), until enough number of sets is picked.
|
- One of each remaining types of object (**structure**, **animal**, **other**), until enough number of sets is picked.
|
||||||
- Obstacles marked as **other** are picked last, and are generally rare.
|
- Obstacles marked as **other** are picked last, and are generally rare.
|
||||||
|
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@ In order to make functional creature you also need:
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
// camelCase unique creature identifier
|
// camelCase unique creature identifier
|
||||||
"creatureName" :
|
"creatureName" :
|
||||||
{
|
{
|
||||||
|
@ -5,6 +5,7 @@ This page helps you to create a creature (i.e. a unit that fights in a battle) f
|
|||||||
## Utilities
|
## Utilities
|
||||||
|
|
||||||
You need to download the two utilities [`DefPreview`](https://sourceforge.net/projects/grayface-misc/files/DefPreview-1.2.1/) and [`H3DefTool`](https://sourceforge.net/projects/grayface-misc/files/H3DefTool-3.4.2/) from the internet:
|
You need to download the two utilities [`DefPreview`](https://sourceforge.net/projects/grayface-misc/files/DefPreview-1.2.1/) and [`H3DefTool`](https://sourceforge.net/projects/grayface-misc/files/H3DefTool-3.4.2/) from the internet:
|
||||||
|
|
||||||
- `DefPreview` converts a `.def` file to `.bmp` images
|
- `DefPreview` converts a `.def` file to `.bmp` images
|
||||||
- `H3DefTool` converts `.bmp` images to a `.def` file
|
- `H3DefTool` converts `.bmp` images to a `.def` file
|
||||||
|
|
||||||
@ -21,7 +22,9 @@ The sun is always at zenith, so the shadow is always behind. The reason is that
|
|||||||
We don't know the right elevation angle for the view.
|
We don't know the right elevation angle for the view.
|
||||||
|
|
||||||
### 3D render
|
### 3D render
|
||||||
You can render your creature using a 3D software like _Blender_. You can start with those free-licenced rigged 3D models:
|
|
||||||
|
You can render your creature using a 3D software like *Blender*. You can start with those free-licenced rigged 3D models:
|
||||||
|
|
||||||
- [Fantasy-bandit](https://www.cgtrader.com/free-3d-models/character/man/fantasy-bandit)
|
- [Fantasy-bandit](https://www.cgtrader.com/free-3d-models/character/man/fantasy-bandit)
|
||||||
- [Monster-4](https://www.cgtrader.com/free-3d-models/character/fantasy-character/monster-4-f5757b92-dc9c-4f5e-ad0d-593203d14fe2)
|
- [Monster-4](https://www.cgtrader.com/free-3d-models/character/fantasy-character/monster-4-f5757b92-dc9c-4f5e-ad0d-593203d14fe2)
|
||||||
- [Crypt-fiend-modular-character](https://www.cgtrader.com/free-3d-models/character/fantasy-character/crypt-fiend-modular-character-demo-scene)
|
- [Crypt-fiend-modular-character](https://www.cgtrader.com/free-3d-models/character/fantasy-character/crypt-fiend-modular-character-demo-scene)
|
||||||
@ -33,57 +36,60 @@ You can render your creature using a 3D software like _Blender_. You can start w
|
|||||||
- [Shani](https://www.cgtrader.com/free-3d-models/character/woman/shani-3d-character)
|
- [Shani](https://www.cgtrader.com/free-3d-models/character/woman/shani-3d-character)
|
||||||
|
|
||||||
You can also create your 3D model from a single image:
|
You can also create your 3D model from a single image:
|
||||||
- _Stable Fast 3D_: https://huggingface.co/spaces/stabilityai/stable-fast-3d
|
|
||||||
- _Unique3D_: https://huggingface.co/spaces/abreza/Unique3D
|
|
||||||
|
|
||||||
To use it in _Blender_, create a `.blend` project and import the file. To render the texture:
|
- *Stable Fast 3D*: <https://huggingface.co/spaces/stabilityai/stable-fast-3d>
|
||||||
1. Add a _Principled BSDF_ material to the object
|
- *Unique3D*: <https://huggingface.co/spaces/abreza/Unique3D>
|
||||||
1. Create a _Color Attribute_ in the _Shader Editor_ view
|
|
||||||
1. Link the Color output of the _Color Attribute_ to the _Base color_ input of the _Principled BSDF_
|
|
||||||
|
|
||||||
You can improve details by cropping the source image on a detail and generate a model for this detail. Once both imported in _Blender_, melt them together.
|
To use it in *Blender*, create a `.blend` project and import the file. To render the texture:
|
||||||
|
|
||||||
Render the images without background by selecting png RVBA and disabling background (_Film_ -> _Filter_ -> _Transparent_). It avoids the creatures to have an ugly dark border. Then, to correctly separate the creature from the cyan area, in _GIMP_, apply the threeshold on the transparency by clicking on _Layer_ -> _Transparency_ -> _Alpha threeshold_.
|
1. Add a *Principled BSDF* material to the object
|
||||||
|
1. Create a *Color Attribute* in the *Shader Editor* view
|
||||||
|
1. Link the Color output of the *Color Attribute* to the *Base color* input of the *Principled BSDF*
|
||||||
|
|
||||||
|
You can improve details by cropping the source image on a detail and generate a model for this detail. Once both imported in *Blender*, melt them together.
|
||||||
|
|
||||||
|
Render the images without background by selecting png RVBA and disabling background (*Film* -> *Filter* -> *Transparent*). It avoids the creatures to have an ugly dark border. Then, to correctly separate the creature from the cyan area, in *GIMP*, apply the threeshold on the transparency by clicking on *Layer* -> *Transparency* -> *Alpha threeshold*.
|
||||||
|
|
||||||
The global FPS of the game is 10 f/s but you can render at a higher level and configure it in the `.json` files. We are not in the 1990's.
|
The global FPS of the game is 10 f/s but you can render at a higher level and configure it in the `.json` files. We are not in the 1990's.
|
||||||
|
|
||||||
### IA render
|
### IA render
|
||||||
|
|
||||||
You can also use an AI like _Flux_ to generate the main creature representation: https://huggingface.co/spaces/multimodalart/FLUX.1-merged
|
You can also use an AI like *Flux* to generate the main creature representation: <https://huggingface.co/spaces/multimodalart/FLUX.1-merged>
|
||||||
|
|
||||||
Then you can add random animations for idle states with _SVD_: https://huggingface.co/spaces/xi0v/Stable-Video-Diffusion-Img2Vid
|
Then you can add random animations for idle states with *SVD*: <https://huggingface.co/spaces/xi0v/Stable-Video-Diffusion-Img2Vid>
|
||||||
|
|
||||||
Most of the time, the creatures do not move more than one pixel in an idle animation. The reason may be to avoid too much animation on screen and make the transition with the other animations always seamless. Use poses with _ControlNet_ or _OpenPose_. For specific animations, I recommend to use _Cinemo_ because it adds a description prompt but the resolution is smaller: https://huggingface.co/spaces/maxin-cn/Cinemo
|
Most of the time, the creatures do not move more than one pixel in an idle animation. The reason may be to avoid too much animation on screen and make the transition with the other animations always seamless. Use poses with *ControlNet* or *OpenPose*. For specific animations, I recommend to use *Cinemo* because it adds a description prompt but the resolution is smaller: <https://huggingface.co/spaces/maxin-cn/Cinemo>
|
||||||
|
|
||||||
Make animations seamless from one to another. To do this, you can draw the first and the last images with a prompt with _ToonCrafter_: https://huggingface.co/spaces/ChristianHappy/tooncrafter
|
Make animations seamless from one to another. To do this, you can draw the first and the last images with a prompt with *ToonCrafter*: <https://huggingface.co/spaces/ChristianHappy/tooncrafter>
|
||||||
|
|
||||||
Most of the time, you need to increase the resolution or the quality of your template image, so use _SUPIR_: https://huggingface.co/spaces/Fabrice-TIERCELIN/SUPIR
|
Most of the time, you need to increase the resolution or the quality of your template image, so use *SUPIR*: <https://huggingface.co/spaces/Fabrice-TIERCELIN/SUPIR>
|
||||||
|
|
||||||
## Battle sound effect
|
## Battle sound effect
|
||||||
|
|
||||||
To create the audio effects, I recommend to use _Tango 2_: https://huggingface.co/spaces/declare-lab/tango2
|
To create the audio effects, I recommend to use *Tango 2*: <https://huggingface.co/spaces/declare-lab/tango2>
|
||||||
|
|
||||||
The quality is better than _Stable Audio_.
|
The quality is better than *Stable Audio*.
|
||||||
|
|
||||||
## Map render
|
## Map render
|
||||||
|
|
||||||
We don't know the right elevation angle for the view but 45° elevation seems to be a good choice. For the sunlight direction, I would say 45° elevation and 45° azimut.
|
We don't know the right elevation angle for the view but 45° elevation seems to be a good choice. For the sunlight direction, I would say 45° elevation and 45° azimut.
|
||||||
|
|
||||||
The map creatures are not rendered on the map with vanishing points but in isometric. You can [get an orthogonal render in Blender](https://blender.stackexchange.com/a/135384/2768). If you are creating a creature and its updated version, most of the time, the both creatures are not oriented to the same side on the map. I think that the animation on the map is usually the _Mouse Over_ animation on battle.
|
The map creatures are not rendered on the map with vanishing points but in isometric. You can [get an orthogonal render in Blender](https://blender.stackexchange.com/a/135384/2768). If you are creating a creature and its updated version, most of the time, the both creatures are not oriented to the same side on the map. I think that the animation on the map is usually the *Mouse Over* animation on battle.
|
||||||
|
|
||||||
You can see that the view angle is higher than on a battle. To change the angle from a battle sprite, you can use _Zero 1-to-3_: https://huggingface.co/spaces/cvlab/zero123-live
|
You can see that the view angle is higher than on a battle. To change the angle from a battle sprite, you can use *Zero 1-to-3*: <https://huggingface.co/spaces/cvlab/zero123-live>
|
||||||
|
|
||||||
You can get higher resolution using this Video AI that can control the motion of the camera: https://huggingface.co/spaces/TencentARC/MotionCtrl_SVD
|
You can get higher resolution using this Video AI that can control the motion of the camera: <https://huggingface.co/spaces/TencentARC/MotionCtrl_SVD>
|
||||||
|
|
||||||
If you have a 3D software, you can get better quality by converting your image into 3D model and then render it from another angle using _Stable Fast 3D_: https://huggingface.co/spaces/stabilityai/stable-fast-3d
|
If you have a 3D software, you can get better quality by converting your image into 3D model and then render it from another angle using *Stable Fast 3D*: <https://huggingface.co/spaces/stabilityai/stable-fast-3d>
|
||||||
|
|
||||||
Follow this comment to retrieve the color: https://huggingface.co/stabilityai/TripoSR/discussions/1#65e8a8e5e214f37d85dad366
|
Follow this comment to retrieve the color: <https://huggingface.co/stabilityai/TripoSR/discussions/1#65e8a8e5e214f37d85dad366>
|
||||||
|
|
||||||
### Shadow render
|
### Shadow render
|
||||||
|
|
||||||
There are no strong rules in the original game about the angle of the shadows on the map. Different buildings have inconsistent shadows. To draw the shadow, I recommend the following technique:
|
There are no strong rules in the original game about the angle of the shadows on the map. Different buildings have inconsistent shadows. To draw the shadow, I recommend the following technique:
|
||||||
|
|
||||||
Let's consider that the object is a vertical cone:
|
Let's consider that the object is a vertical cone:
|
||||||
|
|
||||||
| | | | | | | | | | |
|
| | | | | | | | | | |
|
||||||
|---|---|---|---|---|---|---|---|---|---|
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
@ -98,6 +104,7 @@ Let's consider that the object is a vertical cone:
|
|||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
|
|
||||||
Locate the top and its projection to the ground:
|
Locate the top and its projection to the ground:
|
||||||
|
|
||||||
| | | | | | | | | | |
|
| | | | | | | | | | |
|
||||||
|---|---|---|---|---|---|---|---|---|---|
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
@ -112,6 +119,7 @@ Locate the top and its projection to the ground:
|
|||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
|
|
||||||
Then draw a rectangle triangle on the left:
|
Then draw a rectangle triangle on the left:
|
||||||
|
|
||||||
| | | | | | | | | | |
|
| | | | | | | | | | |
|
||||||
|---|---|---|---|---|---|---|---|---|---|
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
@ -126,6 +134,7 @@ Then draw a rectangle triangle on the left:
|
|||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
|
|
||||||
The square top is the projection of the shadow of the top of the cone:
|
The square top is the projection of the shadow of the top of the cone:
|
||||||
|
|
||||||
| | | | | | | | | | |
|
| | | | | | | | | | |
|
||||||
|---|---|---|---|---|---|---|---|---|---|
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
@ -140,6 +149,7 @@ The square top is the projection of the shadow of the top of the cone:
|
|||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
|
|
||||||
Then you can draw the rest of the shadow:
|
Then you can draw the rest of the shadow:
|
||||||
|
|
||||||
| | | | | | | | | | |
|
| | | | | | | | | | |
|
||||||
|---|---|---|---|---|---|---|---|---|---|
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
| 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 | 🟦 |
|
||||||
|
@ -47,7 +47,7 @@ Each town requires a set of buildings (Around 30-45 buildings)
|
|||||||
|
|
||||||
## Faction node (root entry for town configuration)
|
## Faction node (root entry for town configuration)
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
// Unique faction identifier.
|
// Unique faction identifier.
|
||||||
"myFaction" :
|
"myFaction" :
|
||||||
{
|
{
|
||||||
@ -108,7 +108,7 @@ Each town requires a set of buildings (Around 30-45 buildings)
|
|||||||
|
|
||||||
## Town node
|
## Town node
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// Field that describes behavior of map object part of town. Town-specific part of object format
|
// Field that describes behavior of map object part of town. Town-specific part of object format
|
||||||
"mapObject" :
|
"mapObject" :
|
||||||
@ -256,7 +256,7 @@ Each town requires a set of buildings (Around 30-45 buildings)
|
|||||||
|
|
||||||
## Siege node
|
## Siege node
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
// Describes town siege screen
|
// Describes town siege screen
|
||||||
// Comments in the end of each graphic position indicate specify required suffix for image
|
// Comments in the end of each graphic position indicate specify required suffix for image
|
||||||
// Note: one not included image is battlefield background with suffix "BACK"
|
// Note: one not included image is battlefield background with suffix "BACK"
|
||||||
@ -340,7 +340,9 @@ Each town requires a set of buildings (Around 30-45 buildings)
|
|||||||
```
|
```
|
||||||
|
|
||||||
## Building node
|
## Building node
|
||||||
|
|
||||||
See [Town Building Format](Town_Building_Format.md)
|
See [Town Building Format](Town_Building_Format.md)
|
||||||
|
|
||||||
## Structure node
|
## Structure node
|
||||||
|
|
||||||
See [Town Building Format](Town_Building_Format.md)
|
See [Town Building Format](Town_Building_Format.md)
|
@ -3,90 +3,109 @@
|
|||||||
This page helps you to create from scratch a VCMI mod that adds a new faction. The faction mod structure is described [here](Faction_Format.md).
|
This page helps you to create from scratch a VCMI mod that adds a new faction. The faction mod structure is described [here](Faction_Format.md).
|
||||||
|
|
||||||
## Questioning the faction creation
|
## Questioning the faction creation
|
||||||
|
|
||||||
Before creating a faction, be aware that creating a faction mod is lots of work. You can start [creating creatures](Creature_Help.md) in a creature mod that can be converted into a faction mod after. This way, you are sure to release something. The smallest contribution is a hero portrait that you can suggest on an existing mod. You can also restore the former version of the [Ruins faction](https://github.com/vcmi-mods/ruins-town/tree/1bea30a1d915770e2fd0f95d158030815ff462cd). You would only have to remake the similar parts to the new version.
|
Before creating a faction, be aware that creating a faction mod is lots of work. You can start [creating creatures](Creature_Help.md) in a creature mod that can be converted into a faction mod after. This way, you are sure to release something. The smallest contribution is a hero portrait that you can suggest on an existing mod. You can also restore the former version of the [Ruins faction](https://github.com/vcmi-mods/ruins-town/tree/1bea30a1d915770e2fd0f95d158030815ff462cd). You would only have to remake the similar parts to the new version.
|
||||||
|
|
||||||
## Make a playable faction mod
|
## Make a playable faction mod
|
||||||
Before creating your content, retrieve the content of an existing faction mod like [Highlands town](https://github.com/vcmi-mods/highlands-town). To download the project, click on the _Code_ button and click on _Download ZIP_. The first thing to do is to change all the faction identifiers in the files following the [faction format](Faction_Format.md) and manage to play with the faction and the original without any conflict. To play to a faction, you have to add all the files in your _Mods_ folder. When it works, you will be able to modify the content step by step.
|
|
||||||
|
Before creating your content, retrieve the content of an existing faction mod like [Highlands town](https://github.com/vcmi-mods/highlands-town). To download the project, click on the *Code* button and click on *Download ZIP*. The first thing to do is to change all the faction identifiers in the files following the [faction format](Faction_Format.md) and manage to play with the faction and the original without any conflict. To play to a faction, you have to add all the files in your *Mods* folder. When it works, you will be able to modify the content step by step.
|
||||||
|
|
||||||
Keep in mind that the most important part of a faction mod, above the animations, the graphisms and the musics, is the concept because if you have to change it, you have to change everything else. All the remaining content can be improved by the community.
|
Keep in mind that the most important part of a faction mod, above the animations, the graphisms and the musics, is the concept because if you have to change it, you have to change everything else. All the remaining content can be improved by the community.
|
||||||
|
|
||||||
## Town screen
|
## Town screen
|
||||||
### Background
|
|
||||||
Beware to direct all the shadows to the same direction. The easiest way to create the background is to use a text-to-image AI. The free most powerful AI at the moment is _Flux_ available here: https://huggingface.co/spaces/multimodalart/FLUX.1-merged
|
|
||||||
|
|
||||||
In the _Advanced Options_, set the width to 800px and set the height to 374px.
|
### Background
|
||||||
|
|
||||||
|
Beware to direct all the shadows to the same direction. The easiest way to create the background is to use a text-to-image AI. The free most powerful AI at the moment is *Flux* available here: <https://huggingface.co/spaces/multimodalart/FLUX.1-merged>
|
||||||
|
|
||||||
|
In the *Advanced Options*, set the width to 800px and set the height to 374px.
|
||||||
|
|
||||||
### Buildings
|
### Buildings
|
||||||
To render a building upon the background, I recommend to use an inpainting AI like _BRIA Inpaint_: https://huggingface.co/spaces/briaai/BRIA-2.3-Inpainting
|
|
||||||
|
To render a building upon the background, I recommend to use an inpainting AI like *BRIA Inpaint*: <https://huggingface.co/spaces/briaai/BRIA-2.3-Inpainting>
|
||||||
|
|
||||||
The idea is to select the area where you want to add the building. As a prompt, describe the new building. The advantage is a perfect match between the background and the building. Keep in mind that to correctly integrate a building image, it must contain the image of the background on its edges. It simulates the semi-transparency.
|
The idea is to select the area where you want to add the building. As a prompt, describe the new building. The advantage is a perfect match between the background and the building. Keep in mind that to correctly integrate a building image, it must contain the image of the background on its edges. It simulates the semi-transparency.
|
||||||
|
|
||||||
You can also animate the building or the background using _Stable Video Diffusion_: https://huggingface.co/spaces/multimodalart/stable-video-diffusion
|
You can also animate the building or the background using *Stable Video Diffusion*: <https://huggingface.co/spaces/multimodalart/stable-video-diffusion>
|
||||||
|
|
||||||
## Map dwellings
|
## Map dwellings
|
||||||
You may want to get the same render as in the town, so you have to change the angle and the shadows. If you handle a 3D model software, you can start with _Stable Fast 3D_: https://huggingface.co/spaces/stabilityai/stable-fast-3d
|
|
||||||
|
You may want to get the same render as in the town, so you have to change the angle and the shadows. If you handle a 3D model software, you can start with *Stable Fast 3D*: <https://huggingface.co/spaces/stabilityai/stable-fast-3d>
|
||||||
|
|
||||||
The map dwellings are not rendered on the map with vanishing points but in isometric. You can [get an orthogonal render in Blender](https://blender.stackexchange.com/a/135384/2768).
|
The map dwellings are not rendered on the map with vanishing points but in isometric. You can [get an orthogonal render in Blender](https://blender.stackexchange.com/a/135384/2768).
|
||||||
|
|
||||||
Without 3D, you can use _Zero 1-to-3_: https://huggingface.co/spaces/cvlab/zero123-live
|
Without 3D, you can use *Zero 1-to-3*: <https://huggingface.co/spaces/cvlab/zero123-live>
|
||||||
|
|
||||||
You can get higher resolution using this Video AI that can control the motion of the camera: https://huggingface.co/spaces/TencentARC/MotionCtrl_SVD
|
You can get higher resolution using this Video AI that can control the motion of the camera: <https://huggingface.co/spaces/TencentARC/MotionCtrl_SVD>
|
||||||
|
|
||||||
The buildings on the map are more satured than on town screen. If you have to reduce the size of an image, do not use interpolation (LANCZOS, Bilinear...) to get more details, not a blurred image. If you need to increase the resolution or the quality of your template image, use _SUPIR_: https://huggingface.co/spaces/Fabrice-TIERCELIN/SUPIR
|
The buildings on the map are more satured than on town screen. If you have to reduce the size of an image, do not use interpolation (LANCZOS, Bilinear...) to get more details, not a blurred image. If you need to increase the resolution or the quality of your template image, use *SUPIR*: <https://huggingface.co/spaces/Fabrice-TIERCELIN/SUPIR>
|
||||||
|
|
||||||
## Map buildings
|
## Map buildings
|
||||||
|
|
||||||
The AIs badly understand the sun direction and the perspective angles. To generate the buildings on the adventure map:
|
The AIs badly understand the sun direction and the perspective angles. To generate the buildings on the adventure map:
|
||||||
|
|
||||||
1. Open the HOMM3 map editor
|
1. Open the HOMM3 map editor
|
||||||
1. Put items all around a big empty area
|
1. Put items all around a big empty area
|
||||||
1. Make a screenshot
|
1. Make a screenshot
|
||||||
1. Go on an AI like _BRIA Inpaint_: https://huggingface.co/spaces/briaai/BRIA-2.3-Inpainting
|
1. Go on an AI like *BRIA Inpaint*: <https://huggingface.co/spaces/briaai/BRIA-2.3-Inpainting>
|
||||||
1. Inpaint the (big) empty middle with the brush
|
1. Inpaint the (big) empty middle with the brush
|
||||||
1. Use a prompt like: `A dark house, at the center of the image, map, isometric, parallel perspective, sunlight from the bottom right`
|
1. Use a prompt like: `A dark house, at the center of the image, map, isometric, parallel perspective, sunlight from the bottom right`
|
||||||
|
|
||||||
## Music
|
## Music
|
||||||
|
|
||||||
Here are unused available themes:
|
Here are unused available themes:
|
||||||
|
|
||||||
* [Synthetic Horizon](https://github.com/Fabrice-TIERCELIN/forge/raw/theme/content/music/factions/theme.ogg)
|
* [Synthetic Horizon](https://github.com/Fabrice-TIERCELIN/forge/raw/theme/content/music/factions/theme.ogg)
|
||||||
|
|
||||||
1. Prompt: `Dystopy, Cinematic classical, Science fiction, 160 bpm, Best quality, Futuristic`
|
1. Prompt: `Dystopy, Cinematic classical, Science fiction, 160 bpm, Best quality, Futuristic`
|
||||||
1. Initially created for: _Forge town_
|
1. Initially created for: *Forge town*
|
||||||
|
|
||||||
* [Quantum Overture](https://github.com/Fabrice-TIERCELIN/asylum-town/raw/theme/asylum-town/content/Music/factions/AsylumTown.ogg)
|
* [Quantum Overture](https://github.com/Fabrice-TIERCELIN/asylum-town/raw/theme/asylum-town/content/Music/factions/AsylumTown.ogg)
|
||||||
|
|
||||||
1. Prompt: `Clef shifting, Fantasy, Mystical, Overworldly, Cinematic classical`
|
1. Prompt: `Clef shifting, Fantasy, Mystical, Overworldly, Cinematic classical`
|
||||||
1. Initially created for: _Asylum town_
|
1. Initially created for: *Asylum town*
|
||||||
|
|
||||||
* [Warrior s March](https://github.com/vcmi-mods/ruins-town/assets/20668759/964f27de-6feb-4ef6-9d25-455f52938cef)
|
* [Warrior s March](https://github.com/vcmi-mods/ruins-town/assets/20668759/964f27de-6feb-4ef6-9d25-455f52938cef)
|
||||||
|
|
||||||
1. Prompt: `Powerful percussions, Drums, Battle Anthem, Rythm, Warrior, 160 bpm, Celtic, New age, Instrumental, Accoustic, Medieval`
|
1. Prompt: `Powerful percussions, Drums, Battle Anthem, Rythm, Warrior, 160 bpm, Celtic, New age, Instrumental, Accoustic, Medieval`
|
||||||
1. Initially created for: _Ruins town_
|
1. Initially created for: *Ruins town*
|
||||||
|
|
||||||
* [Clan of Echoes](https://github.com/Fabrice-TIERCELIN/ruins-town/raw/theme/ruins-town/content/music/ruins.ogg)
|
* [Clan of Echoes](https://github.com/Fabrice-TIERCELIN/ruins-town/raw/theme/ruins-town/content/music/ruins.ogg)
|
||||||
|
|
||||||
1. Prompt: `new age, medieval, celtic, warrior, battle, soundtrack, accoustic, drums, rythm`
|
1. Prompt: `new age, medieval, celtic, warrior, battle, soundtrack, accoustic, drums, rythm`
|
||||||
1. Initially created for: _Ruins town_
|
1. Initially created for: *Ruins town*
|
||||||
|
|
||||||
* [Enchanted Reverie](https://github.com/Fabrice-TIERCELIN/grove/raw/theme/Grove/content/Music/factions/GroveTown.ogg)
|
* [Enchanted Reverie](https://github.com/Fabrice-TIERCELIN/grove/raw/theme/Grove/content/Music/factions/GroveTown.ogg)
|
||||||
|
|
||||||
1. Prompt: `Classical music, Soundtrack, Score, Instrumental, 160 bpm, ((((fantasy)))), mystic`
|
1. Prompt: `Classical music, Soundtrack, Score, Instrumental, 160 bpm, ((((fantasy)))), mystic`
|
||||||
1. Initially created for: _Grove town_
|
1. Initially created for: *Grove town*
|
||||||
|
|
||||||
* [World Discovery](https://github.com/vcmi-mods/asylum-town/assets/20668759/34438523-8a44-44ca-b493-127501b474a6)
|
* [World Discovery](https://github.com/vcmi-mods/asylum-town/assets/20668759/34438523-8a44-44ca-b493-127501b474a6)
|
||||||
|
|
||||||
1. Prompt: `Clef shifting, fantasy, mystical, overworldly, Cinematic classical`
|
1. Prompt: `Clef shifting, fantasy, mystical, overworldly, Cinematic classical`
|
||||||
1. Initially created for: _Asylum town_
|
1. Initially created for: *Asylum town*
|
||||||
|
|
||||||
* [Enchanted Ballad](https://github.com/vcmi-mods/fairy-town/assets/20668759/619e6e33-d940-4899-8c76-9c1e8d3d20aa)
|
* [Enchanted Ballad](https://github.com/vcmi-mods/fairy-town/assets/20668759/619e6e33-d940-4899-8c76-9c1e8d3d20aa)
|
||||||
|
|
||||||
1. Prompt: `Females vocalize, Cinematic classical, Harp, Fairy tale, Princess, 160 bpm`
|
1. Prompt: `Females vocalize, Cinematic classical, Harp, Fairy tale, Princess, 160 bpm`
|
||||||
1. Initially created for: _Fairy town_
|
1. Initially created for: *Fairy town*
|
||||||
|
|
||||||
* [Baroque Resurgence](https://github.com/Fabrice-TIERCELIN/courtyard_proposal/raw/theme/Courtyard/Content/music/factions/courtyard/CourtTown.ogg)
|
* [Baroque Resurgence](https://github.com/Fabrice-TIERCELIN/courtyard_proposal/raw/theme/Courtyard/Content/music/factions/courtyard/CourtTown.ogg)
|
||||||
|
|
||||||
1. Prompt: `Baroque, Instrumental, 160 bpm, Cinematic classical, Best quality`
|
1. Prompt: `Baroque, Instrumental, 160 bpm, Cinematic classical, Best quality`
|
||||||
1. Initially created for: _Courtyard town_
|
1. Initially created for: *Courtyard town*
|
||||||
|
|
||||||
* [Harvest Parade](https://github.com/Fabrice-TIERCELIN/greenhouse-town/raw/theme/Greenhouse/content/Music/town.ogg)
|
* [Harvest Parade](https://github.com/Fabrice-TIERCELIN/greenhouse-town/raw/theme/Greenhouse/content/Music/town.ogg)
|
||||||
1. Prompt: `Marching band, Best quality, Happy, Vegetables`
|
|
||||||
1. Initially created for: _Green town_
|
|
||||||
|
|
||||||
Those themes have been generated using _[Udio](https://udio.com)_.
|
1. Prompt: `Marching band, Best quality, Happy, Vegetables`
|
||||||
|
1. Initially created for: *Green town*
|
||||||
|
|
||||||
|
Those themes have been generated using *[Udio](https://udio.com)*.
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||
|
|
||||||
Most of the time, the first screenshot is the townscreen because it's the most specific content.
|
Most of the time, the first screenshot is the townscreen because it's the most specific content.
|
||||||
|
|
||||||
## Recycle
|
## Recycle
|
||||||
|
|
||||||
Some mods contain neutral heroes or creatures. You can integrate them in your faction mod. Don't forget to remove the content from the original mod.
|
Some mods contain neutral heroes or creatures. You can integrate them in your faction mod. Don't forget to remove the content from the original mod.
|
@ -9,7 +9,7 @@ In order to make functional hero class you also need:
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
// Unique identifier of hero class, camelCase
|
// Unique identifier of hero class, camelCase
|
||||||
"myClassName" :
|
"myClassName" :
|
||||||
{
|
{
|
||||||
|
@ -9,7 +9,7 @@ In order to make functional hero you also need:
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"myHeroName" :
|
"myHeroName" :
|
||||||
{
|
{
|
||||||
// Identifier of class this hero belongs to. Such as knight or battleMage
|
// Identifier of class this hero belongs to. Such as knight or battleMage
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"newRiver" :
|
"newRiver" :
|
||||||
{
|
{
|
||||||
// Two-letters unique identifier for this river. Used in map format
|
// Two-letters unique identifier for this river. Used in map format
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"newRoad" :
|
"newRoad" :
|
||||||
{
|
{
|
||||||
// Two-letters unique identifier for this road. Used in map format
|
// Two-letters unique identifier for this road. Used in map format
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Main format
|
## Main format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// Skill be only be available on maps with water
|
// Skill be only be available on maps with water
|
||||||
"onlyOnWaterMap" : false,
|
"onlyOnWaterMap" : false,
|
||||||
@ -11,7 +11,7 @@
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
"skillName":
|
"skillName":
|
||||||
{
|
{
|
||||||
@ -55,7 +55,7 @@ level fields become optional if they equal "base" configuration.
|
|||||||
|
|
||||||
## Skill level format
|
## Skill level format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// Localizable description
|
// Localizable description
|
||||||
// Use {xxx} for formatting
|
// Use {xxx} for formatting
|
||||||
@ -87,7 +87,7 @@ level fields become optional if they equal "base" configuration.
|
|||||||
The following modifies the tactics skill to grant an additional speed
|
The following modifies the tactics skill to grant an additional speed
|
||||||
boost at advanced and expert levels.
|
boost at advanced and expert levels.
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"core:tactics" : {
|
"core:tactics" : {
|
||||||
"base" : {
|
"base" : {
|
||||||
"effects" : {
|
"effects" : {
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Main format
|
## Main format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"spellName":
|
"spellName":
|
||||||
{
|
{
|
||||||
@ -156,7 +156,7 @@
|
|||||||
|
|
||||||
TODO
|
TODO
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"projectile": [
|
"projectile": [
|
||||||
{"minimumAngle": 0 ,"defName":"C20SPX4"},
|
{"minimumAngle": 0 ,"defName":"C20SPX4"},
|
||||||
@ -179,7 +179,7 @@ Json object with data common for all levels can be put here. These configuration
|
|||||||
|
|
||||||
This will make spell affect single target on all levels except expert, where it is massive spell.
|
This will make spell affect single target on all levels except expert, where it is massive spell.
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
"base":{
|
"base":{
|
||||||
"range": 0
|
"range": 0
|
||||||
},
|
},
|
||||||
@ -192,7 +192,7 @@ This will make spell affect single target on all levels except expert, where it
|
|||||||
|
|
||||||
TODO
|
TODO
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
{
|
{
|
||||||
//Mandatory, localizable description. Use {xxx} for formatting
|
//Mandatory, localizable description. Use {xxx} for formatting
|
||||||
@ -262,7 +262,7 @@ Configurable spells ignore *offensive* flag, *effects* and *cumulativeEffects*.
|
|||||||
|
|
||||||
TODO
|
TODO
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
"mod:effectId":{
|
"mod:effectId":{
|
||||||
|
|
||||||
@ -283,7 +283,7 @@ TODO
|
|||||||
|
|
||||||
TODO
|
TODO
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
"mod:effectId":{
|
"mod:effectId":{
|
||||||
|
|
||||||
@ -304,7 +304,7 @@ TODO
|
|||||||
|
|
||||||
Configurable version of Clone spell.
|
Configurable version of Clone spell.
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
"mod:effectId":{
|
"mod:effectId":{
|
||||||
|
|
||||||
@ -320,7 +320,7 @@ TODO
|
|||||||
|
|
||||||
If effect is automatic, spell behave like offensive spell (uses power, levelPower etc)
|
If effect is automatic, spell behave like offensive spell (uses power, levelPower etc)
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
"mod:effectId":{
|
"mod:effectId":{
|
||||||
|
|
||||||
@ -368,7 +368,7 @@ TODO
|
|||||||
If effect is automatic, spell behave like \[de\]buff spell (effect and
|
If effect is automatic, spell behave like \[de\]buff spell (effect and
|
||||||
cumulativeEffects ignored)
|
cumulativeEffects ignored)
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
"mod:effectId":{
|
"mod:effectId":{
|
||||||
|
|
||||||
@ -389,21 +389,21 @@ cumulativeEffects ignored)
|
|||||||
TODO
|
TODO
|
||||||
|
|
||||||
- CREATURE target (only battle spells)
|
- CREATURE target (only battle spells)
|
||||||
- range 0: smart assumed single creature target
|
- range 0: smart assumed single creature target
|
||||||
- range "X" + smart modifier = enchanter casting, expert massive spells
|
- range "X" + smart modifier = enchanter casting, expert massive spells
|
||||||
- range "X" + no smart modifier = armageddon, death ripple, destroy undead
|
- range "X" + no smart modifier = armageddon, death ripple, destroy undead
|
||||||
- any other range (including chain effect)
|
- any other range (including chain effect)
|
||||||
- smart modifier: smth like cloud of confusion in H4 (if I remember correctly :) )
|
- smart modifier: smth like cloud of confusion in H4 (if I remember correctly :) )
|
||||||
- no smart modifier: like inferno, fireball etc. but target only creature
|
- no smart modifier: like inferno, fireball etc. but target only creature
|
||||||
|
|
||||||
- NO_TARGET
|
- NO_TARGET
|
||||||
- no target selection,(abilities, most adventure spells)
|
- no target selection,(abilities, most adventure spells)
|
||||||
|
|
||||||
- LOCATION
|
- LOCATION
|
||||||
- any tile on map/battlefield (inferno, fireball etc.), DD also here but with special handling
|
- any tile on map/battlefield (inferno, fireball etc.), DD also here but with special handling
|
||||||
- clearTarget - destination hex must be clear (unused so far)
|
- clearTarget - destination hex must be clear (unused so far)
|
||||||
- clearAfffected - all affected hexes must be clear (forceField, fireWall)
|
- clearAfffected - all affected hexes must be clear (forceField, fireWall)
|
||||||
|
|
||||||
- OBSTACLE target
|
- OBSTACLE target
|
||||||
- range 0: any single obstacle
|
- range 0: any single obstacle
|
||||||
- range X: all obstacles
|
- range X: all obstacles
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Format
|
## Format
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"newTerrain" :
|
"newTerrain" :
|
||||||
{
|
{
|
||||||
// Two-letters unique identifier for this terrain. Used in map format
|
// Two-letters unique identifier for this terrain. Used in map format
|
||||||
|
@ -10,11 +10,14 @@ Each building requires following assets:
|
|||||||
- Town hall icon (1 image)
|
- Town hall icon (1 image)
|
||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
|
|
||||||
These are just a couple of examples of what can be done in VCMI. See vcmi configuration files to check how buildings from Heroes III are implemented or other mods for more examples
|
These are just a couple of examples of what can be done in VCMI. See vcmi configuration files to check how buildings from Heroes III are implemented or other mods for more examples
|
||||||
|
|
||||||
####
|
####
|
||||||
|
|
||||||
##### Order of Fire from Inferno:
|
##### Order of Fire from Inferno
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"special4": {
|
"special4": {
|
||||||
"requires" : [ "mageGuild1" ],
|
"requires" : [ "mageGuild1" ],
|
||||||
"name" : "Order of Fire",
|
"name" : "Order of Fire",
|
||||||
@ -37,7 +40,8 @@ These are just a couple of examples of what can be done in VCMI. See vcmi config
|
|||||||
```
|
```
|
||||||
|
|
||||||
##### Mana Vortex from Dungeon
|
##### Mana Vortex from Dungeon
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"special2": {
|
"special2": {
|
||||||
"requires" : [ "mageGuild1" ],
|
"requires" : [ "mageGuild1" ],
|
||||||
"name" : "Mana Vortex",
|
"name" : "Mana Vortex",
|
||||||
@ -65,7 +69,8 @@ These are just a couple of examples of what can be done in VCMI. See vcmi config
|
|||||||
```
|
```
|
||||||
|
|
||||||
#### Resource Silo with custom production
|
#### Resource Silo with custom production
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"resourceSilo": {
|
"resourceSilo": {
|
||||||
"name" : "Wood Resource Silo",
|
"name" : "Wood Resource Silo",
|
||||||
"description" : "Produces 2 wood every day",
|
"description" : "Produces 2 wood every day",
|
||||||
@ -80,7 +85,8 @@ These are just a couple of examples of what can be done in VCMI. See vcmi config
|
|||||||
```
|
```
|
||||||
|
|
||||||
#### Brotherhood of Sword - bonuses in siege
|
#### Brotherhood of Sword - bonuses in siege
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"special3": {
|
"special3": {
|
||||||
// replaces +1 Morale bonus from Tavern
|
// replaces +1 Morale bonus from Tavern
|
||||||
"upgradeReplacesBonuses" : true,
|
"upgradeReplacesBonuses" : true,
|
||||||
@ -96,7 +102,8 @@ These are just a couple of examples of what can be done in VCMI. See vcmi config
|
|||||||
```
|
```
|
||||||
|
|
||||||
#### Lighthouse - bonus to all heroes under player control
|
#### Lighthouse - bonus to all heroes under player control
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"special1": {
|
"special1": {
|
||||||
"bonuses": [
|
"bonuses": [
|
||||||
{
|
{
|
||||||
@ -112,7 +119,7 @@ These are just a couple of examples of what can be done in VCMI. See vcmi config
|
|||||||
|
|
||||||
## Town Building node
|
## Town Building node
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// Numeric identifier of this building
|
// Numeric identifier of this building
|
||||||
"id" : 0,
|
"id" : 0,
|
||||||
@ -211,7 +218,7 @@ These are just a couple of examples of what can be done in VCMI. See vcmi config
|
|||||||
|
|
||||||
Building requirements can be described using logical expressions:
|
Building requirements can be described using logical expressions:
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"requires" :
|
"requires" :
|
||||||
[
|
[
|
||||||
"allOf", // Normal H3 "build all" mode
|
"allOf", // Normal H3 "build all" mode
|
||||||
@ -228,10 +235,13 @@ Building requirements can be described using logical expressions:
|
|||||||
]
|
]
|
||||||
]
|
]
|
||||||
```
|
```
|
||||||
|
|
||||||
### List of unique town buildings
|
### List of unique town buildings
|
||||||
|
|
||||||
#### Buildings from Heroes III
|
#### Buildings from Heroes III
|
||||||
|
|
||||||
Following Heroes III buildings can be used as unique buildings for a town. Their functionality should be identical to a corresponding H3 building. H3 buildings that are not present in this list contain no hardcoded functionality. See vcmi json configuration to see how such buildings can be implemented in a mod.
|
Following Heroes III buildings can be used as unique buildings for a town. Their functionality should be identical to a corresponding H3 building. H3 buildings that are not present in this list contain no hardcoded functionality. See vcmi json configuration to see how such buildings can be implemented in a mod.
|
||||||
|
|
||||||
- `mysticPond`
|
- `mysticPond`
|
||||||
- `artifactMerchant`
|
- `artifactMerchant`
|
||||||
- `freelancersGuild`
|
- `freelancersGuild`
|
||||||
@ -244,16 +254,18 @@ Following Heroes III buildings can be used as unique buildings for a town. Their
|
|||||||
- `treasury`
|
- `treasury`
|
||||||
|
|
||||||
#### Buildings from other Heroes III mods
|
#### Buildings from other Heroes III mods
|
||||||
|
|
||||||
Following HotA buildings can be used as unique building for a town. Functionality should match corresponding HotA building:
|
Following HotA buildings can be used as unique building for a town. Functionality should match corresponding HotA building:
|
||||||
|
|
||||||
- `bank`
|
- `bank`
|
||||||
|
|
||||||
#### Custom buildings
|
#### Custom buildings
|
||||||
In addition to above, it is possible to use same format as [Rewardable](../Map_Objects/Rewardable.md) map objects for town buildings. In order to do that, configuration of a rewardable object must be placed into `configuration` json node in building config.
|
|
||||||
|
|
||||||
|
In addition to above, it is possible to use same format as [Rewardable](../Map_Objects/Rewardable.md) map objects for town buildings. In order to do that, configuration of a rewardable object must be placed into `configuration` json node in building config.
|
||||||
|
|
||||||
### Town Structure node
|
### Town Structure node
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// Main animation file for this building
|
// Main animation file for this building
|
||||||
"animation" : "",
|
"animation" : "",
|
||||||
@ -281,16 +293,18 @@ In addition to above, it is possible to use same format as [Rewardable](../Map_O
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
#### Markets in towns
|
#### Markets in towns
|
||||||
|
|
||||||
Market buildings require list of available [modes](../Map_Objects/Market.md)
|
Market buildings require list of available [modes](../Map_Objects/Market.md)
|
||||||
|
|
||||||
##### Marketplace
|
##### Marketplace
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"marketplace": { "marketModes" : ["resource-resource", "resource-player"] },
|
"marketplace": { "marketModes" : ["resource-resource", "resource-player"] },
|
||||||
```
|
```
|
||||||
|
|
||||||
##### Artifact merchant
|
##### Artifact merchant
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"special1": { "type" : "artifactMerchant", "requires" : [ "marketplace" ], "marketModes" : ["resource-artifact", "artifact-resource"] },
|
"special1": { "type" : "artifactMerchant", "requires" : [ "marketplace" ], "marketModes" : ["resource-artifact", "artifact-resource"] },
|
||||||
```
|
```
|
@ -7,6 +7,7 @@ In most cases, VCMI supports formats that were supported by Heroes III, with add
|
|||||||
### Images
|
### Images
|
||||||
|
|
||||||
For images VCMI supports:
|
For images VCMI supports:
|
||||||
|
|
||||||
- png. Recommended for usage in mods
|
- png. Recommended for usage in mods
|
||||||
- bmp. While this format is supported, bmp images have no compressions leading to large file sizes
|
- bmp. While this format is supported, bmp images have no compressions leading to large file sizes
|
||||||
- pcx (h3 version). Note that this is format that is specific to Heroes III and has nothing in common with widely known .pcx format. Files in this format generally can only be found inside of .lod archive of Heroes III and are usually extracted as .bmp files
|
- pcx (h3 version). Note that this is format that is specific to Heroes III and has nothing in common with widely known .pcx format. Files in this format generally can only be found inside of .lod archive of Heroes III and are usually extracted as .bmp files
|
||||||
@ -26,6 +27,7 @@ For animations VCMI supports .def format from Heroes III as well as alternative
|
|||||||
### Sounds
|
### Sounds
|
||||||
|
|
||||||
For sounds VCMI currently supports:
|
For sounds VCMI currently supports:
|
||||||
|
|
||||||
- .ogg/vorbis format - preferred for mods. Unlike wav, vorbis uses compression which may cause some data loss, however even 128kbit is generally undistinguishable from lossless formats
|
- .ogg/vorbis format - preferred for mods. Unlike wav, vorbis uses compression which may cause some data loss, however even 128kbit is generally undistinguishable from lossless formats
|
||||||
- .wav format. This is format used by H3. It is supported by vcmi, but it may result in large file sizes (and as result - large mods)
|
- .wav format. This is format used by H3. It is supported by vcmi, but it may result in large file sizes (and as result - large mods)
|
||||||
|
|
||||||
@ -36,6 +38,7 @@ Support for additional formats, such as ogg/opus or flac may be added in future
|
|||||||
### Music
|
### Music
|
||||||
|
|
||||||
For music VCMI currently supports:
|
For music VCMI currently supports:
|
||||||
|
|
||||||
- .ogg/vorbis format - preferred for mods. Generally offers better quality and lower sizes compared to mp3
|
- .ogg/vorbis format - preferred for mods. Generally offers better quality and lower sizes compared to mp3
|
||||||
- .mp3 format. This is format used by H3
|
- .mp3 format. This is format used by H3
|
||||||
|
|
||||||
@ -51,6 +54,7 @@ Starting from VCMI 1.6, following video container formats are supported by VCMI:
|
|||||||
- .webm - modern, free format that is recommended for modding.
|
- .webm - modern, free format that is recommended for modding.
|
||||||
|
|
||||||
Supported video codecs:
|
Supported video codecs:
|
||||||
|
|
||||||
- bink and smacker - formats used by Heroes III, should be used only to avoid re-encoding
|
- bink and smacker - formats used by Heroes III, should be used only to avoid re-encoding
|
||||||
- theora - used by Heroes III: HD Edition
|
- theora - used by Heroes III: HD Edition
|
||||||
- vp8 - modern format with way better compression compared to formats used by Heroes III
|
- vp8 - modern format with way better compression compared to formats used by Heroes III
|
||||||
@ -59,6 +63,7 @@ Supported video codecs:
|
|||||||
Support for av1 video codec is likely to be added in future.
|
Support for av1 video codec is likely to be added in future.
|
||||||
|
|
||||||
Supported audio codecs:
|
Supported audio codecs:
|
||||||
|
|
||||||
- binkaudio and smackaud - formats used by Heroes III
|
- binkaudio and smackaud - formats used by Heroes III
|
||||||
- vorbis - modern format with good compression level
|
- vorbis - modern format with good compression level
|
||||||
- opus - recommended, improvement over vorbis. Any bitrate is supported, with 128 kbit probably being the best option
|
- opus - recommended, improvement over vorbis. Any bitrate is supported, with 128 kbit probably being the best option
|
||||||
|
@ -13,6 +13,7 @@ If user for example selects 3x resolution and only 2x exists in mod then the 2x
|
|||||||
## Mod
|
## Mod
|
||||||
|
|
||||||
For upscaled images you have to use following folders (next to `sprites` and `data` folders):
|
For upscaled images you have to use following folders (next to `sprites` and `data` folders):
|
||||||
|
|
||||||
- `sprites2x`, `sprites3x`, `sprites4x` for sprites
|
- `sprites2x`, `sprites3x`, `sprites4x` for sprites
|
||||||
- `data2x`, `data3x`, `data4x` for images
|
- `data2x`, `data3x`, `data4x` for images
|
||||||
|
|
||||||
@ -24,11 +25,13 @@ It's also possible (but not necessary) to add high-definition shadows: Just plac
|
|||||||
In future, such shadows will likely become required to correctly exclude shadow from effects such as Clone spell.
|
In future, such shadows will likely become required to correctly exclude shadow from effects such as Clone spell.
|
||||||
|
|
||||||
Shadow images are used only for animations of following objects:
|
Shadow images are used only for animations of following objects:
|
||||||
|
|
||||||
- All adventure map objects
|
- All adventure map objects
|
||||||
- All creature animations in combat
|
- All creature animations in combat
|
||||||
|
|
||||||
Same for overlays with `-overlay`. But overlays are **necessary** for some animation graphics. They will be colorized by VCMI.
|
Same for overlays with `-overlay`. But overlays are **necessary** for some animation graphics. They will be colorized by VCMI.
|
||||||
|
|
||||||
Currently needed for:
|
Currently needed for:
|
||||||
|
|
||||||
- Flaggable adventure map objects. Overlay must contain a transparent image with white flags on it and will be used to colorize flags to owning player
|
- Flaggable adventure map objects. Overlay must contain a transparent image with white flags on it and will be used to colorize flags to owning player
|
||||||
- Creature battle animations, idle and mouse hover group. Overlay must contain a transparent image with white outline of creature for highlighting on mouse hover)
|
- Creature battle animations, idle and mouse hover group. Overlay must contain a transparent image with white outline of creature for highlighting on mouse hover)
|
||||||
|
@ -38,7 +38,7 @@ Templates are dynamically filtered depending on parameters you choose.
|
|||||||
|
|
||||||
To load the map, press open and select map file from the browser.
|
To load the map, press open and select map file from the browser.
|
||||||
|
|
||||||
You can load both *.h3m and *.vmap formats but for saving *.vmap is allowed only.
|
You can load both *.h3m and*.vmap formats but for saving *.vmap is allowed only.
|
||||||
|
|
||||||
## Views
|
## Views
|
||||||
|
|
||||||
@ -60,7 +60,7 @@ There are 3 buttons switching views <img width="131" alt="Снимок экра
|
|||||||
|
|
||||||
<img width="190" src="https://user-images.githubusercontent.com/9308612/188778010-a1d45d59-7333-4432-b83f-57190fbe09f4.png">
|
<img width="190" src="https://user-images.githubusercontent.com/9308612/188778010-a1d45d59-7333-4432-b83f-57190fbe09f4.png">
|
||||||
|
|
||||||
# Setup terrain
|
## Setup terrain
|
||||||
|
|
||||||
1. Select brush you want
|
1. Select brush you want
|
||||||
<img width="124" src="https://user-images.githubusercontent.com/9308612/188776299-fd688696-a98d-4f89-8bef-e81c90d3724b.png">
|
<img width="124" src="https://user-images.githubusercontent.com/9308612/188776299-fd688696-a98d-4f89-8bef-e81c90d3724b.png">
|
||||||
@ -85,9 +85,10 @@ To erase roads or rivers, you need to select tiles to be cleaned and press empty
|
|||||||
|
|
||||||
<img width="130" src="https://user-images.githubusercontent.com/9308612/189969309-87ff7818-2915-4b38-b9db-ec4a616d18e7.png">
|
<img width="130" src="https://user-images.githubusercontent.com/9308612/189969309-87ff7818-2915-4b38-b9db-ec4a616d18e7.png">
|
||||||
|
|
||||||
_Erasing works either for roads or for rivers, e.g. empty button from the roads tab erases roads only, but not rivers. You also can safely select bigger area, because it won't erase anything on tiles without roads/rivers accordingly_
|
*Erasing works either for roads or for rivers, e.g. empty button from the roads tab erases roads only, but not rivers. You also can safely select bigger area, because it won't erase anything on tiles without roads/rivers accordingly*
|
||||||
|
|
||||||
### About brushes
|
### About brushes
|
||||||
|
|
||||||
* Buttons "1", "2", "4" - 1x1, 2x2, 4x4 brush sizes accordingly
|
* Buttons "1", "2", "4" - 1x1, 2x2, 4x4 brush sizes accordingly
|
||||||
* Button "[]" - non-additive rectangle selection
|
* Button "[]" - non-additive rectangle selection
|
||||||
* Button "O" - lasso brush (not implemented yet)
|
* Button "O" - lasso brush (not implemented yet)
|
||||||
@ -171,7 +172,7 @@ You can modify general properties of the map
|
|||||||
|
|
||||||
## Player settings
|
## Player settings
|
||||||
|
|
||||||
Open **Map** menu on the top and select **Player settings"
|
Open **Map** menu on the top and select **Player settings**
|
||||||
|
|
||||||
<img width="141" src="https://user-images.githubusercontent.com/9308612/188781357-4a6091cf-f175-4649-a000-620f306d2c46.png">
|
<img width="141" src="https://user-images.githubusercontent.com/9308612/188781357-4a6091cf-f175-4649-a000-620f306d2c46.png">
|
||||||
|
|
||||||
@ -208,6 +209,7 @@ vcmieditor loads set of mods using exactly same mechanism as game uses and mod m
|
|||||||
The mods mechanism used in map editor is the same as in game.
|
The mods mechanism used in map editor is the same as in game.
|
||||||
|
|
||||||
To enable or disable mods
|
To enable or disable mods
|
||||||
|
|
||||||
* Start launcher, activate or deactivate mods you want
|
* Start launcher, activate or deactivate mods you want
|
||||||
* Close launcher
|
* Close launcher
|
||||||
* Run map editor
|
* Run map editor
|
||||||
|
@ -16,7 +16,7 @@ Full object consists from 3 parts:
|
|||||||
|
|
||||||
## Object group format
|
## Object group format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
{
|
{
|
||||||
"myCoolObjectGroup":
|
"myCoolObjectGroup":
|
||||||
@ -42,6 +42,7 @@ Full object consists from 3 parts:
|
|||||||
## Object types
|
## Object types
|
||||||
|
|
||||||
### Moddable types
|
### Moddable types
|
||||||
|
|
||||||
These are object types that are available for modding and have configurable properties
|
These are object types that are available for modding and have configurable properties
|
||||||
|
|
||||||
- `configurable` - see [Rewardable](Map_Objects/Rewardable.md). Visitable object which grants all kinds of rewards (gold, experience, Bonuses etc...)
|
- `configurable` - see [Rewardable](Map_Objects/Rewardable.md). Visitable object which grants all kinds of rewards (gold, experience, Bonuses etc...)
|
||||||
@ -55,6 +56,7 @@ These are object types that are available for modding and have configurable prop
|
|||||||
- `terrain` - Defines terrain overlays such as magic grounds. TODO: documentation. See config files in vcmi installation for reference
|
- `terrain` - Defines terrain overlays such as magic grounds. TODO: documentation. See config files in vcmi installation for reference
|
||||||
|
|
||||||
### Common types
|
### Common types
|
||||||
|
|
||||||
These are types that don't have configurable properties, however it is possible to add additional map templates for this objects, for use in editor or in random maps generator
|
These are types that don't have configurable properties, however it is possible to add additional map templates for this objects, for use in editor or in random maps generator
|
||||||
|
|
||||||
- `static` - Defines unpassable static map obstacles that can be used by RMG
|
- `static` - Defines unpassable static map obstacles that can be used by RMG
|
||||||
@ -79,6 +81,7 @@ These are types that don't have configurable properties, however it is possible
|
|||||||
- `monolith`
|
- `monolith`
|
||||||
|
|
||||||
### Internal types
|
### Internal types
|
||||||
|
|
||||||
These are internal types that are generally not available for modding and are handled by vcmi internally.
|
These are internal types that are generally not available for modding and are handled by vcmi internally.
|
||||||
|
|
||||||
- `hero`
|
- `hero`
|
||||||
@ -96,7 +99,7 @@ These are internal types that are generally not available for modding and are ha
|
|||||||
|
|
||||||
## Object type format
|
## Object type format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"myCoolObject":
|
"myCoolObject":
|
||||||
{
|
{
|
||||||
@ -150,7 +153,7 @@ These are internal types that are generally not available for modding and are ha
|
|||||||
|
|
||||||
## Object template format
|
## Object template format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"myCoolObjectTemplate" :
|
"myCoolObjectTemplate" :
|
||||||
{
|
{
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Boat
|
# Boat
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
// Layer on which this boat moves. Possible values:
|
// Layer on which this boat moves. Possible values:
|
||||||
// "land" - same rules as movement of hero on land
|
// "land" - same rules as movement of hero on land
|
||||||
|
@ -6,9 +6,11 @@ Format of rewards is same as in [Rewardable Objects](Rewardable.md)
|
|||||||
Deprecated in 1.6. Please use [Rewardable Objects](Rewardable.md) instead. See Conversion from 1.5 format section below for help with migration
|
Deprecated in 1.6. Please use [Rewardable Objects](Rewardable.md) instead. See Conversion from 1.5 format section below for help with migration
|
||||||
|
|
||||||
### Example
|
### Example
|
||||||
|
|
||||||
This example defines a rewardable object with functionality similar of H3 creature bank.
|
This example defines a rewardable object with functionality similar of H3 creature bank.
|
||||||
See [Rewardable Objects](Rewardable.md) for detailed documentation of these properties.
|
See [Rewardable Objects](Rewardable.md) for detailed documentation of these properties.
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"name" : "Cyclops Stockpile",
|
"name" : "Cyclops Stockpile",
|
||||||
|
|
||||||
@ -93,6 +95,7 @@ See [Rewardable Objects](Rewardable.md) for detailed documentation of these prop
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Conversion from 1.5 format
|
### Conversion from 1.5 format
|
||||||
|
|
||||||
This is a list of changes that needs to be done to bank config to migrate it to 1.6 system. See [Rewardable Objects](Rewardable.md) documentation for description of new fields
|
This is a list of changes that needs to be done to bank config to migrate it to 1.6 system. See [Rewardable Objects](Rewardable.md) documentation for description of new fields
|
||||||
|
|
||||||
- If your object type has defined `handler`, change its value from `bank` to `configurable`
|
- If your object type has defined `handler`, change its value from `bank` to `configurable`
|
||||||
@ -112,7 +115,7 @@ This is a list of changes that needs to be done to bank config to migrate it to
|
|||||||
|
|
||||||
### Old format (1.5 or earlier)
|
### Old format (1.5 or earlier)
|
||||||
|
|
||||||
``` jsonc
|
```json5
|
||||||
{
|
{
|
||||||
/// If true, battle setup will be like normal - Attacking player on the left, enemy on the right
|
/// If true, battle setup will be like normal - Attacking player on the left, enemy on the right
|
||||||
"regularUnitPlacement" : true,
|
"regularUnitPlacement" : true,
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Dwelling
|
# Dwelling
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
/// List of creatures in this bank. Each list represents one "level" of bank
|
/// List of creatures in this bank. Each list represents one "level" of bank
|
||||||
/// Creatures on the same level will have shared growth and available number (similar to towns)
|
/// Creatures on the same level will have shared growth and available number (similar to towns)
|
||||||
|
@ -3,12 +3,13 @@
|
|||||||
Flaggable object are those that can be captured by a visiting hero. H3 examples are mines, dwellings, or lighthouse.
|
Flaggable object are those that can be captured by a visiting hero. H3 examples are mines, dwellings, or lighthouse.
|
||||||
|
|
||||||
Currently, it is possible to make flaggable objects that provide player with:
|
Currently, it is possible to make flaggable objects that provide player with:
|
||||||
|
|
||||||
- Any [Bonus](Bonus_Format.md) supported by bonus system
|
- Any [Bonus](Bonus_Format.md) supported by bonus system
|
||||||
- Daily resources income (wood, ore, gold, etc)
|
- Daily resources income (wood, ore, gold, etc)
|
||||||
|
|
||||||
## Format description
|
## Format description
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
"baseObjectName" : {
|
"baseObjectName" : {
|
||||||
"name" : "Object name",
|
"name" : "Object name",
|
||||||
|
@ -7,7 +7,7 @@ Markets can be added as any other object with special handler called "market".
|
|||||||
|
|
||||||
Here is schema describing such object
|
Here is schema describing such object
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
"seafaringAcademy" : //object name
|
"seafaringAcademy" : //object name
|
||||||
{
|
{
|
||||||
"handler" : "market", //market handler
|
"handler" : "market", //market handler
|
||||||
@ -34,6 +34,7 @@ Here is schema describing such object
|
|||||||
|
|
||||||
Mode parameter defines a way to exchange different entities. Multiple modes can be specified to support several types of exchange.
|
Mode parameter defines a way to exchange different entities. Multiple modes can be specified to support several types of exchange.
|
||||||
Following options are supported:
|
Following options are supported:
|
||||||
|
|
||||||
* `"resource-resource"` - regular resource exchange, like trading post
|
* `"resource-resource"` - regular resource exchange, like trading post
|
||||||
* `"resource-player"` - allows to send resources to another player
|
* `"resource-player"` - allows to send resources to another player
|
||||||
* `"creature-resource"` - acts like freelance guild
|
* `"creature-resource"` - acts like freelance guild
|
||||||
@ -49,19 +50,20 @@ Following options are supported:
|
|||||||
### Trading post
|
### Trading post
|
||||||
|
|
||||||
Trading post allows to exchange resources and send resources to another player, so it shall be configured this way:
|
Trading post allows to exchange resources and send resources to another player, so it shall be configured this way:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
"modes" : ["resource-resource", "resource-player"]
|
"modes" : ["resource-resource", "resource-player"]
|
||||||
```
|
```
|
||||||
|
|
||||||
### Black market
|
### Black market
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"modes" : ["resource-artifact"]
|
"modes" : ["resource-artifact"]
|
||||||
```
|
```
|
||||||
|
|
||||||
### Freelance guild
|
### Freelance guild
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"modes" : ["creature-resource"]
|
"modes" : ["creature-resource"]
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -71,7 +73,7 @@ Altar of sacrifice allows exchange creatures for experience for evil factions an
|
|||||||
So both modes shall be available in the market.
|
So both modes shall be available in the market.
|
||||||
Game logic prohibits using modes unavailable for faction
|
Game logic prohibits using modes unavailable for faction
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"modes" : ["creature-experience", "artifact-experience"]
|
"modes" : ["creature-experience", "artifact-experience"]
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -83,14 +85,14 @@ See [Secondary skills](Rewardable.md#secondary-skills) description for more deta
|
|||||||
|
|
||||||
### Example for University of magic (e.g conflux building)
|
### Example for University of magic (e.g conflux building)
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
"modes" : ["resource-skill"],
|
"modes" : ["resource-skill"],
|
||||||
"offer" : ["airMagic", "waterMagic", "earthMagic", "fireMagic"]
|
"offer" : ["airMagic", "waterMagic", "earthMagic", "fireMagic"]
|
||||||
```
|
```
|
||||||
|
|
||||||
### Example for regular University
|
### Example for regular University
|
||||||
|
|
||||||
```js
|
```json5
|
||||||
"modes" : ["resource-skill"],
|
"modes" : ["resource-skill"],
|
||||||
"offer" : [ //4 random skills except necromancy
|
"offer" : [ //4 random skills except necromancy
|
||||||
{ "noneOf" : ["necromancy"] },
|
{ "noneOf" : ["necromancy"] },
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
# Rewardable
|
# Rewardable
|
||||||
|
|
||||||
## Base object definition
|
## Base object definition
|
||||||
|
|
||||||
Rewardable object is defined similarly to other objects, with key difference being `handler`. This field must be set to `"handler" : "configurable"` in order for vcmi to use this mode.
|
Rewardable object is defined similarly to other objects, with key difference being `handler`. This field must be set to `"handler" : "configurable"` in order for vcmi to use this mode.
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
{
|
{
|
||||||
"baseObjectName" : {
|
"baseObjectName" : {
|
||||||
"name" : "Object name",
|
"name" : "Object name",
|
||||||
@ -34,7 +36,8 @@ Rewardable object is defined similarly to other objects, with key difference bei
|
|||||||
```
|
```
|
||||||
|
|
||||||
## Configurable object definition
|
## Configurable object definition
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
// List of potential rewards
|
// List of potential rewards
|
||||||
"rewards" : [
|
"rewards" : [
|
||||||
{
|
{
|
||||||
@ -173,7 +176,8 @@ This property allows defining "variables" that are shared between all rewards an
|
|||||||
Variables are randomized only once, so you can use them multiple times for example, to give skill only if hero does not have this skill (e.g. Witch Hut).
|
Variables are randomized only once, so you can use them multiple times for example, to give skill only if hero does not have this skill (e.g. Witch Hut).
|
||||||
|
|
||||||
Example of creation of a variable named "gainedSkill" of type "secondarySkill":
|
Example of creation of a variable named "gainedSkill" of type "secondarySkill":
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
"variables" : {
|
"variables" : {
|
||||||
"secondarySkill" : {
|
"secondarySkill" : {
|
||||||
"gainedSkill" : {
|
"gainedSkill" : {
|
||||||
@ -187,6 +191,7 @@ Example of creation of a variable named "gainedSkill" of type "secondarySkill":
|
|||||||
```
|
```
|
||||||
|
|
||||||
Possible variable types:
|
Possible variable types:
|
||||||
|
|
||||||
- number: can be used in any place that expects a number
|
- number: can be used in any place that expects a number
|
||||||
- artifact
|
- artifact
|
||||||
- spell
|
- spell
|
||||||
@ -194,19 +199,22 @@ Possible variable types:
|
|||||||
- secondarySkill
|
- secondarySkill
|
||||||
|
|
||||||
To reference variable in limiter prepend variable name with '@' symbol:
|
To reference variable in limiter prepend variable name with '@' symbol:
|
||||||
```json
|
|
||||||
|
```json5
|
||||||
"secondary" : {
|
"secondary" : {
|
||||||
"@gainedSkill" : 1
|
"@gainedSkill" : 1
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
## Reset Parameters definition
|
## Reset Parameters definition
|
||||||
|
|
||||||
This property describes how object state should be reset. Objects without this field will never reset its state.
|
This property describes how object state should be reset. Objects without this field will never reset its state.
|
||||||
|
|
||||||
- Period describes interval between object resets in day. Periods are counted from game start and not from hero visit, so reset duration of 7 will always reset object on new week & duration of 28 will always reset on new month.
|
- Period describes interval between object resets in day. Periods are counted from game start and not from hero visit, so reset duration of 7 will always reset object on new week & duration of 28 will always reset on new month.
|
||||||
- If `visitors` is set to true, game will reset list of visitors (heroes and players) on start of new period, allowing revisits of objects with `visitMode` set to `once`, `hero`, or `player`. Objects with visit mode set to `bonus` are not affected. In order to allow revisit such objects use appropriate bonus duration (e.g. `ONE_DAY` or `ONE_WEEK`) instead.
|
- If `visitors` is set to true, game will reset list of visitors (heroes and players) on start of new period, allowing revisits of objects with `visitMode` set to `once`, `hero`, or `player`. Objects with visit mode set to `bonus` are not affected. In order to allow revisit such objects use appropriate bonus duration (e.g. `ONE_DAY` or `ONE_WEEK`) instead.
|
||||||
- If `rewards` is set to true, object will re-randomize its provided rewards, similar to such H3 objects as "Fountain of Fortune" or "Windmill"
|
- If `rewards` is set to true, object will re-randomize its provided rewards, similar to such H3 objects as "Fountain of Fortune" or "Windmill"
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"resetParameters" : {
|
"resetParameters" : {
|
||||||
"period" : 7,
|
"period" : 7,
|
||||||
"visitors" : true,
|
"visitors" : true,
|
||||||
@ -215,15 +223,17 @@ This property describes how object state should be reset. Objects without this f
|
|||||||
```
|
```
|
||||||
|
|
||||||
## Appear Chance definition
|
## Appear Chance definition
|
||||||
|
|
||||||
This property describes chance for reward to be selected.
|
This property describes chance for reward to be selected.
|
||||||
When object is initialized on map load, game will roll a "dice" - random number in range 0-99, and pick all awards that have appear chance within selected number.
|
When object is initialized on map load, game will roll a "dice" - random number in range 0-99, and pick all awards that have appear chance within selected number.
|
||||||
Note that object that uses appearChance MUST have continuous range for every value in 0-99 range. For example, object with 3 different rewards may want to define them as
|
Note that object that uses appearChance MUST have continuous range for every value in 0-99 range. For example, object with 3 different rewards may want to define them as
|
||||||
|
|
||||||
- `"min" : 0, "max" : 33`
|
- `"min" : 0, "max" : 33`
|
||||||
- `"min" : 33, "max" : 66`
|
- `"min" : 33, "max" : 66`
|
||||||
- `"min" : 66, "max" : 100`
|
- `"min" : 66, "max" : 100`
|
||||||
In other words, min chance of second reward must be equal to max chance of previous reward
|
In other words, min chance of second reward must be equal to max chance of previous reward
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"appearChance":
|
"appearChance":
|
||||||
{
|
{
|
||||||
// (Advanced) rewards with different dice number will get different dice number
|
// (Advanced) rewards with different dice number will get different dice number
|
||||||
@ -240,8 +250,10 @@ In other words, min chance of second reward must be equal to max chance of previ
|
|||||||
```
|
```
|
||||||
|
|
||||||
## Configurable Properties
|
## Configurable Properties
|
||||||
|
|
||||||
Unless stated othervice, all numbers in this section can be replaced with random values, e.g.
|
Unless stated othervice, all numbers in this section can be replaced with random values, e.g.
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"minLevel" : { "min" : 5, "max" : 10 } // select random number between 5-10, including both 5 & 10
|
"minLevel" : { "min" : 5, "max" : 10 } // select random number between 5-10, including both 5 & 10
|
||||||
"minLevel" : [ 2, 4, 6, 8, 10] // (VCMI 1.2) select random number out of provided list, with equal chance for each
|
"minLevel" : [ 2, 4, 6, 8, 10] // (VCMI 1.2) select random number out of provided list, with equal chance for each
|
||||||
```
|
```
|
||||||
@ -250,32 +262,35 @@ In this case, actual value for minLevel will be picked randomly.
|
|||||||
Keep in mind, that all randomization is performed on map load and on object reset (if `rewards` field in `resetParameter` was set).
|
Keep in mind, that all randomization is performed on map load and on object reset (if `rewards` field in `resetParameter` was set).
|
||||||
|
|
||||||
### Current Day
|
### Current Day
|
||||||
|
|
||||||
- Can only be used as limiter. To pass, current day of week should be equal to this value. 1 = first day of the week, 7 = last day
|
- Can only be used as limiter. To pass, current day of week should be equal to this value. 1 = first day of the week, 7 = last day
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"dayOfWeek" : 0
|
"dayOfWeek" : 0
|
||||||
```
|
```
|
||||||
|
|
||||||
- Can only be used as limiter. To pass, number of days since game started must be at equal or greater than this value
|
- Can only be used as limiter. To pass, number of days since game started must be at equal or greater than this value
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"daysPassed" : 8
|
"daysPassed" : 8
|
||||||
```
|
```
|
||||||
|
|
||||||
### Resource
|
### Resource
|
||||||
|
|
||||||
- Can be used as limiter. To pass, player needs to have specified resources. Note that limiter will NOT take resources.
|
- Can be used as limiter. To pass, player needs to have specified resources. Note that limiter will NOT take resources.
|
||||||
- Can be used as reward to grant resources to player
|
- Can be used as reward to grant resources to player
|
||||||
- If negative value is used as reward, it will be used as cost and take resources from player
|
- If negative value is used as reward, it will be used as cost and take resources from player
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"resources": {
|
"resources": {
|
||||||
"crystal" : 6,
|
"crystal" : 6,
|
||||||
"gold" : -1000,
|
"gold" : -1000,
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
- Alternative format that allows random selection of a resource type
|
- Alternative format that allows random selection of a resource type
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"resources": [
|
"resources": [
|
||||||
{
|
{
|
||||||
"anyOf" : [ "wood", "ore" ],
|
"anyOf" : [ "wood", "ore" ],
|
||||||
@ -289,68 +304,75 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Experience
|
### Experience
|
||||||
|
|
||||||
- Can be used as limiter
|
- Can be used as limiter
|
||||||
- Can be used as reward to grant experience to hero
|
- Can be used as reward to grant experience to hero
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"heroExperience" : 1000,
|
"heroExperience" : 1000,
|
||||||
```
|
```
|
||||||
|
|
||||||
### Hero Level
|
### Hero Level
|
||||||
|
|
||||||
- Can be used as limiter. Hero requires to have at least specified level
|
- Can be used as limiter. Hero requires to have at least specified level
|
||||||
- Can be used as reward, will grant hero experience amount equal to the difference between the hero's next level and current level (Tree of Knowledge)
|
- Can be used as reward, will grant hero experience amount equal to the difference between the hero's next level and current level (Tree of Knowledge)
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"heroLevel" : 1,
|
"heroLevel" : 1,
|
||||||
```
|
```
|
||||||
|
|
||||||
### Mana Points
|
### Mana Points
|
||||||
|
|
||||||
- Can be used as limiter. Hero must have at least specific mana amount
|
- Can be used as limiter. Hero must have at least specific mana amount
|
||||||
- Can be used as reward, to give mana points to hero. Mana points may go above mana pool limit.
|
- Can be used as reward, to give mana points to hero. Mana points may go above mana pool limit.
|
||||||
- If negative value is used as reward, it will be used as cost and take mana from player
|
- If negative value is used as reward, it will be used as cost and take mana from player
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"manaPoints": -10,
|
"manaPoints": -10,
|
||||||
```
|
```
|
||||||
|
|
||||||
- If giving mana points puts hero above mana pool limit, any overflow will be multiplied by specified percentage. If set to 0, mana will not go above mana pool limit.
|
- If giving mana points puts hero above mana pool limit, any overflow will be multiplied by specified percentage. If set to 0, mana will not go above mana pool limit.
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"manaOverflowFactor" : 50,
|
"manaOverflowFactor" : 50,
|
||||||
```
|
```
|
||||||
|
|
||||||
### Mana Percentage
|
### Mana Percentage
|
||||||
|
|
||||||
- Can be used as limiter. Hero must have at least specific mana percentage
|
- Can be used as limiter. Hero must have at least specific mana percentage
|
||||||
- Can be used to set hero mana level to specified percentage value, not restricted to mana pool limit (Magic Well, Mana Spring)
|
- Can be used to set hero mana level to specified percentage value, not restricted to mana pool limit (Magic Well, Mana Spring)
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"manaPercentage": 200,
|
"manaPercentage": 200,
|
||||||
```
|
```
|
||||||
|
|
||||||
### Movement Points
|
### Movement Points
|
||||||
|
|
||||||
- Can NOT be used as limiter
|
- Can NOT be used as limiter
|
||||||
- Can be used as reward, to give movement points to hero. Movement points may go above mana pool limit.
|
- Can be used as reward, to give movement points to hero. Movement points may go above mana pool limit.
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"movePoints": 200,
|
"movePoints": 200,
|
||||||
```
|
```
|
||||||
|
|
||||||
### Movement Percentage
|
### Movement Percentage
|
||||||
|
|
||||||
- Can NOT be used as limiter
|
- Can NOT be used as limiter
|
||||||
- Can be used to set hero movement points level to specified percentage value. Value of 0 will take away any remaining movement points
|
- Can be used to set hero movement points level to specified percentage value. Value of 0 will take away any remaining movement points
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"movePercentage": 50,
|
"movePercentage": 50,
|
||||||
```
|
```
|
||||||
|
|
||||||
### Primary Skills
|
### Primary Skills
|
||||||
|
|
||||||
- Can be used as limiter, hero must have primary skill at least at specified level
|
- Can be used as limiter, hero must have primary skill at least at specified level
|
||||||
- Can be used as reward, to increase hero primary skills by selected value
|
- Can be used as reward, to increase hero primary skills by selected value
|
||||||
- If reward value is negative, value will be used as cost, decreasing primary skill
|
- If reward value is negative, value will be used as cost, decreasing primary skill
|
||||||
- Each primary skill can be explicitly specified or randomly selected
|
- Each primary skill can be explicitly specified or randomly selected
|
||||||
- Possible values: `"attack", "defence", "spellpower", "knowledge"`
|
- Possible values: `"attack", "defence", "spellpower", "knowledge"`
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"primary": [
|
"primary": [
|
||||||
{
|
{
|
||||||
// Specific primary skill
|
// Specific primary skill
|
||||||
@ -376,13 +398,15 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Secondary Skills
|
### Secondary Skills
|
||||||
|
|
||||||
- Can be used as limiter, hero must have secondary skill at least at specified level
|
- Can be used as limiter, hero must have secondary skill at least at specified level
|
||||||
- Can be used as reward, to grant secondary skills to hero
|
- Can be used as reward, to grant secondary skills to hero
|
||||||
- If hero already has specified skill, the skills will be leveled up specified number of times
|
- If hero already has specified skill, the skills will be leveled up specified number of times
|
||||||
- If hero does not have selected skill and have free skill slots, he will receive skill at specified level
|
- If hero does not have selected skill and have free skill slots, he will receive skill at specified level
|
||||||
- Possible values: 1 (basic), 2 (advanced), 3 (expert)
|
- Possible values: 1 (basic), 2 (advanced), 3 (expert)
|
||||||
- Each secondary skill can be explicitly specified or randomly selected
|
- Each secondary skill can be explicitly specified or randomly selected
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"secondary": [
|
"secondary": [
|
||||||
{
|
{
|
||||||
// Specific skill
|
// Specific skill
|
||||||
@ -411,17 +435,18 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
|
|
||||||
- Can be used as limiter. Hero must have free skill slot to pass limiter
|
- Can be used as limiter. Hero must have free skill slot to pass limiter
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"canLearnSkills" : true
|
"canLearnSkills" : true
|
||||||
```
|
```
|
||||||
|
|
||||||
### Bonus System
|
### Bonus System
|
||||||
|
|
||||||
- Can be used as reward, to grant bonus to player
|
- Can be used as reward, to grant bonus to player
|
||||||
- if present, MORALE and LUCK bonus will add corresponding image component to UI.
|
- if present, MORALE and LUCK bonus will add corresponding image component to UI.
|
||||||
- Note that unlike most values, parameter of bonuses can NOT be randomized
|
- Note that unlike most values, parameter of bonuses can NOT be randomized
|
||||||
- Description can be string or number of corresponding string from `arraytxt.txt`
|
- Description can be string or number of corresponding string from `arraytxt.txt`
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"bonuses" : [
|
"bonuses" : [
|
||||||
{
|
{
|
||||||
"type" : "MORALE",
|
"type" : "MORALE",
|
||||||
@ -433,11 +458,12 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Artifacts
|
### Artifacts
|
||||||
|
|
||||||
- Can be used as limiter, hero must have artifact either equipped or in backpack
|
- Can be used as limiter, hero must have artifact either equipped or in backpack
|
||||||
- Can be used as reward, to give new artifact to a hero
|
- Can be used as reward, to give new artifact to a hero
|
||||||
- Artifacts added as reward will be used for text substitution. First `%s` in text string will be replaced with name of an artifact
|
- Artifacts added as reward will be used for text substitution. First `%s` in text string will be replaced with name of an artifact
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"artifacts": [
|
"artifacts": [
|
||||||
"ribCage"
|
"ribCage"
|
||||||
],
|
],
|
||||||
@ -447,7 +473,7 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
- For artifact class possible values are "TREASURE", "MINOR", "MAJOR", "RELIC"
|
- For artifact class possible values are "TREASURE", "MINOR", "MAJOR", "RELIC"
|
||||||
- Artifact value range can be specified with min value and max value
|
- Artifact value range can be specified with min value and max value
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"artifacts": [
|
"artifacts": [
|
||||||
{
|
{
|
||||||
"class" : "TREASURE",
|
"class" : "TREASURE",
|
||||||
@ -458,11 +484,12 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Spells
|
### Spells
|
||||||
|
|
||||||
- Can be used as limiter
|
- Can be used as limiter
|
||||||
- Can be used as reward, to give new spell to a hero
|
- Can be used as reward, to give new spell to a hero
|
||||||
- Spells added as reward will be used for text substitution. First `%s` in text string will be replaced with spell name
|
- Spells added as reward will be used for text substitution. First `%s` in text string will be replaced with spell name
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"spells": [
|
"spells": [
|
||||||
"magicArrow"
|
"magicArrow"
|
||||||
],
|
],
|
||||||
@ -471,7 +498,7 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
- Alternative format, random spell selection
|
- Alternative format, random spell selection
|
||||||
- Spell can be selected from specifically selected school
|
- Spell can be selected from specifically selected school
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"spells": [
|
"spells": [
|
||||||
{
|
{
|
||||||
"level" : 1,
|
"level" : 1,
|
||||||
@ -488,18 +515,20 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
- - he does not have a spellbook
|
- - he does not have a spellbook
|
||||||
- - he does not have sufficient Wisdom level for this spell
|
- - he does not have sufficient Wisdom level for this spell
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"canLearnSpells" : [
|
"canLearnSpells" : [
|
||||||
"magicArrow"
|
"magicArrow"
|
||||||
],
|
],
|
||||||
```
|
```
|
||||||
|
|
||||||
### Creatures
|
### Creatures
|
||||||
|
|
||||||
- Can be used as limiter
|
- Can be used as limiter
|
||||||
- Can be used as reward, to give new creatures to a hero
|
- Can be used as reward, to give new creatures to a hero
|
||||||
- If hero does not have enough free slots, game will show selection dialog to pick troops to keep
|
- If hero does not have enough free slots, game will show selection dialog to pick troops to keep
|
||||||
- It is possible to specify probability to receive upgraded creature
|
- It is possible to specify probability to receive upgraded creature
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"creatures" : [
|
"creatures" : [
|
||||||
{
|
{
|
||||||
"type" : "archer",
|
"type" : "archer",
|
||||||
@ -510,13 +539,15 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Guards
|
### Guards
|
||||||
|
|
||||||
- When used in a reward, these creatures will be added to guards of the objects
|
- When used in a reward, these creatures will be added to guards of the objects
|
||||||
- Hero must defeat all guards before being able to receive rewards
|
- Hero must defeat all guards before being able to receive rewards
|
||||||
- Guards are only reset when object rewards are reset
|
- Guards are only reset when object rewards are reset
|
||||||
- Requires `guardsLayout` property to be set in main part of object configuration
|
- Requires `guardsLayout` property to be set in main part of object configuration
|
||||||
- It is possible to add up to 7 slots of creatures
|
- It is possible to add up to 7 slots of creatures
|
||||||
- Guards of the same creature type will never merge or rearrange their stacks
|
- Guards of the same creature type will never merge or rearrange their stacks
|
||||||
```jsonc
|
|
||||||
|
```json5
|
||||||
"guards" : [
|
"guards" : [
|
||||||
{ "type" : "archer", "amount" : 20 },
|
{ "type" : "archer", "amount" : 20 },
|
||||||
{ "type" : "archer", "amount" : 20, "upgradeChance" : 30 },
|
{ "type" : "archer", "amount" : 20, "upgradeChance" : 30 },
|
||||||
@ -525,22 +556,24 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Creatures Change
|
### Creatures Change
|
||||||
|
|
||||||
- Can NOT be used as limiter
|
- Can NOT be used as limiter
|
||||||
- Can be used as reward, to replace creatures in hero army. It is possible to use this parameter both for upgrades of creatures as well as for changing them into completely unrelated creature, e.g. similar to Skeleton Transformer
|
- Can be used as reward, to replace creatures in hero army. It is possible to use this parameter both for upgrades of creatures as well as for changing them into completely unrelated creature, e.g. similar to Skeleton Transformer
|
||||||
- This parameter will not change creatures given by `creatures` parameter on the same visit
|
- This parameter will not change creatures given by `creatures` parameter on the same visit
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"changeCreatures" : {
|
"changeCreatures" : {
|
||||||
"cavalier" : "champion"
|
"cavalier" : "champion"
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
### Spell cast
|
### Spell cast
|
||||||
|
|
||||||
- Can NOT be used as limiter
|
- Can NOT be used as limiter
|
||||||
- As reward, instantly casts adventure map spell for visiting hero. All checks for spell book, wisdom or presence of mana will be ignored. It's possible to specify school level at which spell will be casted. If it's necessary to reduce player's mana or do some checks, they shall be introduced as limiters and other rewards
|
- As reward, instantly casts adventure map spell for visiting hero. All checks for spell book, wisdom or presence of mana will be ignored. It's possible to specify school level at which spell will be casted. If it's necessary to reduce player's mana or do some checks, they shall be introduced as limiters and other rewards
|
||||||
- School level possible values: 1 (basic), 2 (advanced), 3 (expert)
|
- School level possible values: 1 (basic), 2 (advanced), 3 (expert)
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"spellCast" : {
|
"spellCast" : {
|
||||||
"spell" : "townPortal",
|
"spell" : "townPortal",
|
||||||
"schoolLevel": 3
|
"schoolLevel": 3
|
||||||
@ -555,7 +588,7 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
- It is possible to specify which terrain classes should be affected. Tile will be affected if sum of values its classes is positive. For example, `"water" : 1` will affect all water tiles, while `"surface" : 1, "subterra" : -1` will include terrains that have "surface" flag but do not have "subterra" flag
|
- It is possible to specify which terrain classes should be affected. Tile will be affected if sum of values its classes is positive. For example, `"water" : 1` will affect all water tiles, while `"surface" : 1, "subterra" : -1` will include terrains that have "surface" flag but do not have "subterra" flag
|
||||||
- If 'hide' is set to true, then instead of revealing terrain, game will hide affected tiles for all other players
|
- If 'hide' is set to true, then instead of revealing terrain, game will hide affected tiles for all other players
|
||||||
|
|
||||||
```json
|
```json5
|
||||||
"revealTiles" : {
|
"revealTiles" : {
|
||||||
"radius" : 20,
|
"radius" : 20,
|
||||||
"surface" : 1,
|
"surface" : 1,
|
||||||
@ -567,28 +600,31 @@ Keep in mind, that all randomization is performed on map load and on object rese
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Player color
|
### Player color
|
||||||
|
|
||||||
- Can be used as limiter
|
- Can be used as limiter
|
||||||
- Can NOT be used as reward
|
- Can NOT be used as reward
|
||||||
- Only players with specific color can pass the limiter
|
- Only players with specific color can pass the limiter
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"colors" : [ "red", "blue", "tan", "green", "orange", "purple", "teal", "pink" ]
|
"colors" : [ "red", "blue", "tan", "green", "orange", "purple", "teal", "pink" ]
|
||||||
```
|
```
|
||||||
|
|
||||||
### Hero types
|
### Hero types
|
||||||
|
|
||||||
- Can be used as limiter
|
- Can be used as limiter
|
||||||
- Can NOT be used as reward
|
- Can NOT be used as reward
|
||||||
- Only specific heroes can pass the limiter
|
- Only specific heroes can pass the limiter
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"heroes" : [ "orrin" ]
|
"heroes" : [ "orrin" ]
|
||||||
```
|
```
|
||||||
|
|
||||||
### Hero classes
|
### Hero classes
|
||||||
|
|
||||||
- Can be used as limiter
|
- Can be used as limiter
|
||||||
- Can NOT be used as reward
|
- Can NOT be used as reward
|
||||||
- Only heroes belonging to specific classes can pass the limiter
|
- Only heroes belonging to specific classes can pass the limiter
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
"heroClasses" : [ "battlemage" ]
|
"heroClasses" : [ "battlemage" ]
|
||||||
```
|
```
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Fields with description of mod
|
## Fields with description of mod
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
// Name of your mod. While it does not have hard length limit
|
// Name of your mod. While it does not have hard length limit
|
||||||
// it should not be longer than ~30 symbols to fit into allowed space
|
// it should not be longer than ~30 symbols to fit into allowed space
|
||||||
@ -91,7 +91,7 @@
|
|||||||
|
|
||||||
These are fields that are present only in local mod.json file
|
These are fields that are present only in local mod.json file
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
|
|
||||||
{
|
{
|
||||||
// Following section describes configuration files with content added by mod
|
// Following section describes configuration files with content added by mod
|
||||||
@ -210,7 +210,7 @@ See [Translations](Translations.md) for more information
|
|||||||
|
|
||||||
These are fields that are present only in remote repository and are generally not used in mod.json
|
These are fields that are present only in remote repository and are generally not used in mod.json
|
||||||
|
|
||||||
```jsonc
|
```json5
|
||||||
{
|
{
|
||||||
// URL to mod.json that describes this mod
|
// URL to mod.json that describes this mod
|
||||||
"mod" : "https://raw.githubusercontent.com/vcmi-mods/vcmi-extras/vcmi-1.4/mod.json",
|
"mod" : "https://raw.githubusercontent.com/vcmi-mods/vcmi-extras/vcmi-1.4/mod.json",
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Template format
|
## Template format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
/// Unique template name
|
/// Unique template name
|
||||||
"Triangle" :
|
"Triangle" :
|
||||||
{
|
{
|
||||||
@ -56,7 +56,7 @@
|
|||||||
|
|
||||||
## Zone format
|
## Zone format
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
// Type of this zone. Possible values are:
|
// Type of this zone. Possible values are:
|
||||||
// "playerStart", "cpuStart", "treasure", "junction"
|
// "playerStart", "cpuStart", "treasure", "junction"
|
||||||
|
@ -21,8 +21,8 @@ Example of how directory structure of your mod may look like:
|
|||||||
sprites/ - animation, image sets (H3 .def files or VCMI .json files)
|
sprites/ - animation, image sets (H3 .def files or VCMI .json files)
|
||||||
video/ - video files, .bik, .smk, .ogv .webm
|
video/ - video files, .bik, .smk, .ogv .webm
|
||||||
```
|
```
|
||||||
See [File Formats](File_Formats.md) page for more information on which formats are supported or recommended for vcmi
|
|
||||||
|
|
||||||
|
See [File Formats](File_Formats.md) page for more information on which formats are supported or recommended for vcmi
|
||||||
|
|
||||||
## Creating mod file
|
## Creating mod file
|
||||||
|
|
||||||
@ -30,7 +30,7 @@ All VCMI configuration files use [JSON format](http://en.wikipedia.org/wiki/Json
|
|||||||
Mod.json is main file in your mod and must be present in any mod. This file contains basic description of your mod, dependencies or conflicting mods (if present), list of new content and so on.
|
Mod.json is main file in your mod and must be present in any mod. This file contains basic description of your mod, dependencies or conflicting mods (if present), list of new content and so on.
|
||||||
Minimalistic version of this file:
|
Minimalistic version of this file:
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
"name" : "My test mod",
|
"name" : "My test mod",
|
||||||
"description" : "My test mod that add a lot of useless stuff into the game",
|
"description" : "My test mod that add a lot of useless stuff into the game",
|
||||||
@ -45,6 +45,7 @@ See [Mod file Format](Mod_File_Format.md) for its full description.
|
|||||||
## Creation of new objects
|
## Creation of new objects
|
||||||
|
|
||||||
In order to create new object use following steps:
|
In order to create new object use following steps:
|
||||||
|
|
||||||
1. Create json file with definition of new object. See list of supported object types below.
|
1. Create json file with definition of new object. See list of supported object types below.
|
||||||
2. Add any resources needed for this object, such as images, animations or sounds.
|
2. Add any resources needed for this object, such as images, animations or sounds.
|
||||||
2. Add reference to new object in corresponding section of mod.json file
|
2. Add reference to new object in corresponding section of mod.json file
|
||||||
@ -52,9 +53,11 @@ In order to create new object use following steps:
|
|||||||
### List of supported new object types
|
### List of supported new object types
|
||||||
|
|
||||||
Random Map Generator:
|
Random Map Generator:
|
||||||
|
|
||||||
- [Random Map Template](Random_Map_Template.md)
|
- [Random Map Template](Random_Map_Template.md)
|
||||||
|
|
||||||
Game Entities:
|
Game Entities:
|
||||||
|
|
||||||
- [Artifact](Entities_Format/Artifact_Format.md)
|
- [Artifact](Entities_Format/Artifact_Format.md)
|
||||||
- [Creature Requirement](Entities_Format/Creature_Format.md)
|
- [Creature Requirement](Entities_Format/Creature_Format.md)
|
||||||
- [Creature Help](Entities_Format/Creature_Help.md)
|
- [Creature Help](Entities_Format/Creature_Help.md)
|
||||||
@ -66,6 +69,7 @@ Game Entities:
|
|||||||
- [Secondary Skill](Entities_Format/Secondary_Skill_Format.md)
|
- [Secondary Skill](Entities_Format/Secondary_Skill_Format.md)
|
||||||
|
|
||||||
Map objects:
|
Map objects:
|
||||||
|
|
||||||
- [Map Objects](Map_Object_Format.md)
|
- [Map Objects](Map_Object_Format.md)
|
||||||
- - [Rewardable](Map_Objects/Rewardable.md)
|
- - [Rewardable](Map_Objects/Rewardable.md)
|
||||||
- - [Creature Bank](Map_Objects/Creature_Bank.md)
|
- - [Creature Bank](Map_Objects/Creature_Bank.md)
|
||||||
@ -74,6 +78,7 @@ Map objects:
|
|||||||
- - [Boat](Map_Objects/Boat.md)
|
- - [Boat](Map_Objects/Boat.md)
|
||||||
|
|
||||||
Other:
|
Other:
|
||||||
|
|
||||||
- [Terrain](Entities_Format/Terrain_Format.md)
|
- [Terrain](Entities_Format/Terrain_Format.md)
|
||||||
- [River](Entities_Format/River_Format.md)
|
- [River](Entities_Format/River_Format.md)
|
||||||
- [Road](Entities_Format/Road_Format.md)
|
- [Road](Entities_Format/Road_Format.md)
|
||||||
@ -96,7 +101,8 @@ VCMI uses strings to reference objects. Examples:
|
|||||||
### Modifying existing objects
|
### Modifying existing objects
|
||||||
|
|
||||||
Alternatively to creating new objects, you can edit existing objects. Normally, when creating new objects you specify object name as:
|
Alternatively to creating new objects, you can edit existing objects. Normally, when creating new objects you specify object name as:
|
||||||
``` javascript
|
|
||||||
|
```json5
|
||||||
"newCreature" : {
|
"newCreature" : {
|
||||||
// creature parameters
|
// creature parameters
|
||||||
}
|
}
|
||||||
@ -104,7 +110,7 @@ Alternatively to creating new objects, you can edit existing objects. Normally,
|
|||||||
|
|
||||||
In order to access and modify existing object you need to specify mod that you wish to edit:
|
In order to access and modify existing object you need to specify mod that you wish to edit:
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
/// "core" specifier refers to objects that exist in H3
|
/// "core" specifier refers to objects that exist in H3
|
||||||
"core:archer" : {
|
"core:archer" : {
|
||||||
/// This will set health of Archer to 10
|
/// This will set health of Archer to 10
|
||||||
@ -123,6 +129,7 @@ In order to access and modify existing object you need to specify mod that you w
|
|||||||
"speed" : 10
|
"speed" : 10
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that modification of existing objects does not requires a dependency on edited mod. Such definitions will only be used by game if corresponding mod is installed and active.
|
Note that modification of existing objects does not requires a dependency on edited mod. Such definitions will only be used by game if corresponding mod is installed and active.
|
||||||
|
|
||||||
This allows using objects editing not just for rebalancing mods but also to provide compatibility between two different mods or to add interaction between two mods.
|
This allows using objects editing not just for rebalancing mods but also to provide compatibility between two different mods or to add interaction between two mods.
|
||||||
@ -132,6 +139,7 @@ This allows using objects editing not just for rebalancing mods but also to prov
|
|||||||
Any graphical replacer mods fall under this category. In VCMI directory **<mod name>/Content** acts as mod-specific game root directory. So for example file **<mod name>/Content/Data/AISHIELD.PNG** will replace file with same name from **H3Bitmap.lod** game archive.
|
Any graphical replacer mods fall under this category. In VCMI directory **<mod name>/Content** acts as mod-specific game root directory. So for example file **<mod name>/Content/Data/AISHIELD.PNG** will replace file with same name from **H3Bitmap.lod** game archive.
|
||||||
Any other files can be replaced in exactly same way.
|
Any other files can be replaced in exactly same way.
|
||||||
Note that replacing files from archives requires placing them into specific location:
|
Note that replacing files from archives requires placing them into specific location:
|
||||||
|
|
||||||
- H3Bitmap.lod -> Data
|
- H3Bitmap.lod -> Data
|
||||||
- H3Sprite.lod -> Sprites
|
- H3Sprite.lod -> Sprites
|
||||||
- Heroes3.snd -> Sounds
|
- Heroes3.snd -> Sounds
|
||||||
@ -145,12 +153,13 @@ This includes archives added by expansions (e.g. **H3ab_bmp.lod** uses same rule
|
|||||||
|
|
||||||
Heroes III uses custom format for storing animation: def files. These files are used to store all in-game animations as well as for some GUI elements like buttons and for icon sets.
|
Heroes III uses custom format for storing animation: def files. These files are used to store all in-game animations as well as for some GUI elements like buttons and for icon sets.
|
||||||
These files can be replaced by another def file but in some cases original format can't be used. This includes but not limited to:
|
These files can be replaced by another def file but in some cases original format can't be used. This includes but not limited to:
|
||||||
|
|
||||||
- Replacing one (or several) icons in set
|
- Replacing one (or several) icons in set
|
||||||
- Replacing animation with fully-colored 32-bit images
|
- Replacing animation with fully-colored 32-bit images
|
||||||
In VCMI these animation files can also be replaced by json description of their content. See [Animation Format](Animation_Format.md) for full description of this format.
|
In VCMI these animation files can also be replaced by json description of their content. See [Animation Format](Animation_Format.md) for full description of this format.
|
||||||
Example: replacing single icon
|
Example: replacing single icon
|
||||||
|
|
||||||
``` javascript
|
```json5
|
||||||
{
|
{
|
||||||
// List of replaced images
|
// List of replaced images
|
||||||
"images" :
|
"images" :
|
||||||
@ -191,7 +200,7 @@ Same way we can also create special stable branch for every mod under "vcmi-mods
|
|||||||
### Getting into vcmi-mods organization
|
### Getting into vcmi-mods organization
|
||||||
|
|
||||||
Before your mod can be accepted into official mod list you need to get it into repository under "vcmi-mods" organization umbrella. To do this contact one of mod repository maintainers. If needed you can get own team within "vcmi-mods" organization.
|
Before your mod can be accepted into official mod list you need to get it into repository under "vcmi-mods" organization umbrella. To do this contact one of mod repository maintainers. If needed you can get own team within "vcmi-mods" organization.
|
||||||
Link to our mod will looks like that: https://github.com/vcmi-mods/adventure-ai-trace
|
Link to our mod will looks like that: <https://github.com/vcmi-mods/adventure-ai-trace>
|
||||||
|
|
||||||
## Rules of repository
|
## Rules of repository
|
||||||
|
|
||||||
@ -199,8 +208,10 @@ Link to our mod will looks like that: https://github.com/vcmi-mods/adventure-ai-
|
|||||||
|
|
||||||
For sanity reasons mod identifier must only contain lower-case English characters, numbers and hyphens.
|
For sanity reasons mod identifier must only contain lower-case English characters, numbers and hyphens.
|
||||||
|
|
||||||
my-mod-name
|
```
|
||||||
2000-new-maps
|
my-mod-name
|
||||||
|
2000-new-maps
|
||||||
|
```
|
||||||
|
|
||||||
Sub-mods can be named as you like, but we strongly encourage everyone to use proper identifiers for them as well.
|
Sub-mods can be named as you like, but we strongly encourage everyone to use proper identifiers for them as well.
|
||||||
|
|
||||||
|
@ -70,9 +70,11 @@ Alternative usage: `vcmiexp <amount>` - gives selected hero specified amount of
|
|||||||
`nwcbluepill` or `vcmimelkor` or `vcmilose` - player loses
|
`nwcbluepill` or `vcmimelkor` or `vcmilose` - player loses
|
||||||
|
|
||||||
### Misc
|
### Misc
|
||||||
|
|
||||||
`nwctheone` or `vcmigod` - reveals the whole map, gives 5 archangels in each empty slot, unlimited movement points and permanent flight
|
`nwctheone` or `vcmigod` - reveals the whole map, gives 5 archangels in each empty slot, unlimited movement points and permanent flight
|
||||||
|
|
||||||
## Using cheat codes on other players
|
## Using cheat codes on other players
|
||||||
|
|
||||||
By default, all cheat codes apply to current player. Alternatively, it is possible to specify player that you want to target:
|
By default, all cheat codes apply to current player. Alternatively, it is possible to specify player that you want to target:
|
||||||
|
|
||||||
- Specific players: `red`/`blue`/`green`...
|
- Specific players: `red`/`blue`/`green`...
|
||||||
@ -89,12 +91,14 @@ By default, all cheat codes apply to current player. Alternatively, it is possib
|
|||||||
## Multiplayer chat commands
|
## Multiplayer chat commands
|
||||||
|
|
||||||
Following commands can be used in multiplayer only by host player to control the session:
|
Following commands can be used in multiplayer only by host player to control the session:
|
||||||
|
|
||||||
- `!exit` - finish the game
|
- `!exit` - finish the game
|
||||||
- `!save <filename>` - save the game into the specified file
|
- `!save <filename>` - save the game into the specified file
|
||||||
- `!kick red/blue/tan/green/orange/purple/teal/pink` - kick player of specified color from the game
|
- `!kick red/blue/tan/green/orange/purple/teal/pink` - kick player of specified color from the game
|
||||||
- `!kick 0/1/2/3/4/5/6/7/8` - kick player of specified ID from the game (_zero indexed!_) (`0: red, 1: blue, tan: 2, green: 3, orange: 4, purple: 5, teal: 6, pink: 7`)
|
- `!kick 0/1/2/3/4/5/6/7/8` - kick player of specified ID from the game (*zero indexed!*) (`0: red, 1: blue, tan: 2, green: 3, orange: 4, purple: 5, teal: 6, pink: 7`)
|
||||||
|
|
||||||
Following commands can be used by any player in multiplayer:
|
Following commands can be used by any player in multiplayer:
|
||||||
|
|
||||||
- `!help` - displays in-game list of available commands
|
- `!help` - displays in-game list of available commands
|
||||||
- `!cheaters` - lists players that have entered cheat at any point of the game
|
- `!cheaters` - lists players that have entered cheat at any point of the game
|
||||||
- `!vote` - initiates voting to change one of the possible options:
|
- `!vote` - initiates voting to change one of the possible options:
|
||||||
@ -114,12 +118,14 @@ Windows builds of VCMI run separate console window by default, on other platform
|
|||||||
Below a list of supported commands, with their arguments wrapped in `<>`
|
Below a list of supported commands, with their arguments wrapped in `<>`
|
||||||
|
|
||||||
#### Game Commands
|
#### Game Commands
|
||||||
|
|
||||||
`die, fool` - quits game
|
`die, fool` - quits game
|
||||||
`save <filename>` - saves game in given file (at the moment doesn't work)
|
`save <filename>` - saves game in given file (at the moment doesn't work)
|
||||||
`mp` - on adventure map with a hero selected, shows heroes current movement points, max movement points on land and on water
|
`mp` - on adventure map with a hero selected, shows heroes current movement points, max movement points on land and on water
|
||||||
`bonuses` - shows bonuses of currently selected adventure map object
|
`bonuses` - shows bonuses of currently selected adventure map object
|
||||||
|
|
||||||
#### Extract commands
|
#### Extract commands
|
||||||
|
|
||||||
`translate` - save game texts into json files
|
`translate` - save game texts into json files
|
||||||
`translate missing` - save untranslated game texts into json files
|
`translate missing` - save untranslated game texts into json files
|
||||||
`translate maps` - save map and campaign texts into json files
|
`translate maps` - save map and campaign texts into json files
|
||||||
@ -131,15 +137,17 @@ Below a list of supported commands, with their arguments wrapped in `<>`
|
|||||||
`generate assets` - generate all assets at once
|
`generate assets` - generate all assets at once
|
||||||
|
|
||||||
#### AI commands
|
#### AI commands
|
||||||
|
|
||||||
`setBattleAI <ai name>` - change battle AI used by neutral creatures to the one specified, persists through game quit
|
`setBattleAI <ai name>` - change battle AI used by neutral creatures to the one specified, persists through game quit
|
||||||
`gosolo` - AI takes over until the end of turn (unlike original H3 currently causes AI to take over until typed again)
|
`gosolo` - AI takes over until the end of turn (unlike original H3 currently causes AI to take over until typed again)
|
||||||
`controlai <[red][blue][tan][green][orange][purple][teal][pink]>` - gives you control over specified AI player. If none is specified gives you control over all AI players
|
`controlai <[red][blue][tan][green][orange][purple][teal][pink]>` - gives you control over specified AI player. If none is specified gives you control over all AI players
|
||||||
`autoskip` - Toggles autoskip mode on and off. In this mode, player turns are automatically skipped and only AI moves. However, GUI is still present and allows to observe AI moves. After this option is activated, you need to end first turn manually. Press `[Shift]` before your turn starts to not skip it
|
`autoskip` - Toggles autoskip mode on and off. In this mode, player turns are automatically skipped and only AI moves. However, GUI is still present and allows to observe AI moves. After this option is activated, you need to end first turn manually. Press `[Shift]` before your turn starts to not skip it
|
||||||
|
|
||||||
#### Settings
|
#### Settings
|
||||||
|
|
||||||
`set <command> <on/off>` - sets special temporary settings that reset on game quit. Below some of the most notable commands:
|
`set <command> <on/off>` - sets special temporary settings that reset on game quit. Below some of the most notable commands:
|
||||||
-`autoskip` - identical to `autoskip` option
|
-`autoskip` - identical to `autoskip` option
|
||||||
-`onlyAI` - run without human player, all players will be _default AI_
|
-`onlyAI` - run without human player, all players will be *default AI*
|
||||||
-`headless` - run without GUI, implies `onlyAI` is set
|
-`headless` - run without GUI, implies `onlyAI` is set
|
||||||
-`showGrid` - display a square grid overlay on top of adventure map
|
-`showGrid` - display a square grid overlay on top of adventure map
|
||||||
-`showBlocked` - show blocked tiles on map
|
-`showBlocked` - show blocked tiles on map
|
||||||
@ -147,6 +155,7 @@ Below a list of supported commands, with their arguments wrapped in `<>`
|
|||||||
-`hideSystemMessages` - suppress server messages in chat
|
-`hideSystemMessages` - suppress server messages in chat
|
||||||
|
|
||||||
#### Developer Commands
|
#### Developer Commands
|
||||||
|
|
||||||
`crash` - force a game crash. It is sometimes useful to generate memory dump file in certain situations, for example game freeze
|
`crash` - force a game crash. It is sometimes useful to generate memory dump file in certain situations, for example game freeze
|
||||||
`gui` - displays tree view of currently present VCMI common GUI elements
|
`gui` - displays tree view of currently present VCMI common GUI elements
|
||||||
`activate <0/1/2>` - activate game windows (no current use, apparently broken long ago)
|
`activate <0/1/2>` - activate game windows (no current use, apparently broken long ago)
|
||||||
|
@ -174,6 +174,7 @@ TODO
|
|||||||
Simultaneous turns allow multiple players to act at the same time, speeding up early game phase in multiplayer games. During this phase if different players (allies or not) attempt to interact with each other, such as capture objects owned by other players (mines, dwellings, towns) or attack their heroes, game will block such actions. Interaction with same map objects at the same time, such as attacking same wandering monster is also blocked.
|
Simultaneous turns allow multiple players to act at the same time, speeding up early game phase in multiplayer games. During this phase if different players (allies or not) attempt to interact with each other, such as capture objects owned by other players (mines, dwellings, towns) or attack their heroes, game will block such actions. Interaction with same map objects at the same time, such as attacking same wandering monster is also blocked.
|
||||||
|
|
||||||
Following options can be used to configure simultaneous turns:
|
Following options can be used to configure simultaneous turns:
|
||||||
|
|
||||||
- Minimal duration (at least for): this is duration during which simultaneous turns will run unconditionally. Until specified number of days have passed, simultaneous turns will never break and game will not attempt to detect contacts.
|
- Minimal duration (at least for): this is duration during which simultaneous turns will run unconditionally. Until specified number of days have passed, simultaneous turns will never break and game will not attempt to detect contacts.
|
||||||
- Maximal duration (at most for): this is duration after which simultaneous turns will end unconditionally, even if players still have not contacted each other. However if contact detection discovers contact between two players, simultaneous turns between them might end before specified duration.
|
- Maximal duration (at most for): this is duration after which simultaneous turns will end unconditionally, even if players still have not contacted each other. However if contact detection discovers contact between two players, simultaneous turns between them might end before specified duration.
|
||||||
- Simultaneous turns for AI: If this option is on, AI can act at the same time as human players. Note that AI shares settings for simultaneous turns with human players - if no simultaneous turns have been set up this option has no effect.
|
- Simultaneous turns for AI: If this option is on, AI can act at the same time as human players. Note that AI shares settings for simultaneous turns with human players - if no simultaneous turns have been set up this option has no effect.
|
||||||
@ -185,6 +186,7 @@ Players are considered to be "in contact" if movement range of their heroes at t
|
|||||||
Once detected, contact can never be "lost". If game detected contact between two players, this contact will remain active till the end of the game, even if their heroes move far enough from each other.
|
Once detected, contact can never be "lost". If game detected contact between two players, this contact will remain active till the end of the game, even if their heroes move far enough from each other.
|
||||||
|
|
||||||
Game performs contact detection once per turn, at the very start of each in-game day. Once contact detection has been performed, players that are not in contact with each other can start making turn. For example, in game with 4 players: red, blue, brown and green. If game detected contact between red and blue following will happen:
|
Game performs contact detection once per turn, at the very start of each in-game day. Once contact detection has been performed, players that are not in contact with each other can start making turn. For example, in game with 4 players: red, blue, brown and green. If game detected contact between red and blue following will happen:
|
||||||
|
|
||||||
- red, brown and green will all instantly start turn
|
- red, brown and green will all instantly start turn
|
||||||
- once red ends his turn, blue will be able to start his own turn (even if brown or green are still making turn)
|
- once red ends his turn, blue will be able to start his own turn (even if brown or green are still making turn)
|
||||||
|
|
||||||
@ -197,4 +199,4 @@ Differences compared to HD Mod version:
|
|||||||
|
|
||||||
## Manuals and guides
|
## Manuals and guides
|
||||||
|
|
||||||
- https://heroes.thelazy.net/index.php/Main_Page Wiki that aims to be a complete reference to Heroes of Might and Magic III.
|
- <https://heroes.thelazy.net/index.php/Main_Page> Wiki that aims to be a complete reference to Heroes of Might and Magic III.
|
||||||
|
@ -9,6 +9,7 @@ VCMI requires data from original Heroes 3: Shadow of Death or Complete editions.
|
|||||||
Up-to-date releases can be found in our PPA here: <https://launchpad.net/~vcmi/+archive/ubuntu/ppa>
|
Up-to-date releases can be found in our PPA here: <https://launchpad.net/~vcmi/+archive/ubuntu/ppa>
|
||||||
|
|
||||||
To install VCMI from PPA use:
|
To install VCMI from PPA use:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt-add-repository ppa:vcmi/ppa
|
sudo apt-add-repository ppa:vcmi/ppa
|
||||||
sudo apt update
|
sudo apt update
|
||||||
@ -20,25 +21,30 @@ To install VCMI from PPA use:
|
|||||||
We also provide latest, unstable builds mostly suitable for testing here: <https://launchpad.net/~vcmi/+archive/ubuntu/vcmi-latest>
|
We also provide latest, unstable builds mostly suitable for testing here: <https://launchpad.net/~vcmi/+archive/ubuntu/vcmi-latest>
|
||||||
|
|
||||||
In order to install from this PPA use:
|
In order to install from this PPA use:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo add-apt-repository ppa:vcmi/vcmi-latest
|
sudo add-apt-repository ppa:vcmi/vcmi-latest
|
||||||
sudo apt update
|
sudo apt update
|
||||||
sudo apt install vcmi
|
sudo apt install vcmi
|
||||||
```
|
```
|
||||||
|
|
||||||
### Ubuntu - From Ubuntu repository
|
### Ubuntu - From Ubuntu repository
|
||||||
|
|
||||||
VCMI stable builds available in "multiverse" repository. Learn how to enable it in [Ubuntu wiki](https://help.ubuntu.com/community/Repositories/Ubuntu).
|
VCMI stable builds available in "multiverse" repository. Learn how to enable it in [Ubuntu wiki](https://help.ubuntu.com/community/Repositories/Ubuntu).
|
||||||
Once enabled, you can install VCMI using Ubuntu Store or in terminal using following commands:
|
Once enabled, you can install VCMI using Ubuntu Store or in terminal using following commands:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt update
|
sudo apt update
|
||||||
sudo apt install vcmi
|
sudo apt install vcmi
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that version available in Ubuntu is outdated. Install via PPA is preferred.
|
Note that version available in Ubuntu is outdated. Install via PPA is preferred.
|
||||||
|
|
||||||
### Debian
|
### Debian
|
||||||
|
|
||||||
Stable VCMI version is available in "contrib" repository. Learn how to enable it in [Debian wiki](https://wiki.debian.org/SourcesList).
|
Stable VCMI version is available in "contrib" repository. Learn how to enable it in [Debian wiki](https://wiki.debian.org/SourcesList).
|
||||||
To install VCMI from repository:
|
To install VCMI from repository:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install vcmi
|
sudo apt-get install vcmi
|
||||||
@ -52,6 +58,7 @@ Stable VCMI version is available in RPM Fusion repository. Learn how to enable i
|
|||||||
sudo dnf update
|
sudo dnf update
|
||||||
sudo dnf install vcmi
|
sudo dnf install vcmi
|
||||||
```
|
```
|
||||||
|
|
||||||
### Flatpak (distribution-agnostic)
|
### Flatpak (distribution-agnostic)
|
||||||
|
|
||||||
Latest public release build can be installed via Flatpak.
|
Latest public release build can be installed via Flatpak.
|
||||||
@ -87,11 +94,13 @@ To install Heroes 3 data using automated script you need any of:
|
|||||||
- One or two CD's or CD images
|
- One or two CD's or CD images
|
||||||
|
|
||||||
Run the script using options appropriate to your input files:
|
Run the script using options appropriate to your input files:
|
||||||
|
|
||||||
```
|
```
|
||||||
vcmibuilder --cd1 /path/to/iso/or/cd --cd2 /path/to/second/cd
|
vcmibuilder --cd1 /path/to/iso/or/cd --cd2 /path/to/second/cd
|
||||||
vcmibuilder --gog /path/to/gog.com/installer.exe
|
vcmibuilder --gog /path/to/gog.com/installer.exe
|
||||||
vcmibuilder --data /path/to/h3/data
|
vcmibuilder --data /path/to/h3/data
|
||||||
```
|
```
|
||||||
|
|
||||||
You should use only one of these commands.
|
You should use only one of these commands.
|
||||||
|
|
||||||
On flatpak install, it's also possible to run the script, but any path seems to be interpreted from within the Flatpak sandbox:
|
On flatpak install, it's also possible to run the script, but any path seems to be interpreted from within the Flatpak sandbox:
|
||||||
@ -107,9 +116,11 @@ Download both files for the "offline backup game installers" and extract them us
|
|||||||
You can select both downloaded files in launcher to extract automatically.
|
You can select both downloaded files in launcher to extract automatically.
|
||||||
|
|
||||||
Alternatively you can use the classic way:
|
Alternatively you can use the classic way:
|
||||||
|
|
||||||
```
|
```
|
||||||
innoextract --output-dir=~/Downloads/HoMM3 "setup_heroes_of_might_and_magic_3_complete_4.0_(28740).exe"
|
innoextract --output-dir=~/Downloads/HoMM3 "setup_heroes_of_might_and_magic_3_complete_4.0_(28740).exe"
|
||||||
```
|
```
|
||||||
|
|
||||||
(note that installer file name might be different)
|
(note that installer file name might be different)
|
||||||
|
|
||||||
Once innoextract completes, start VCMI Launcher and choose to place existing files. Select the ~/Downloads/HoMM3 directory. Once placing is complete, you can delete both offline installer files as well as ~/Downloads/HoMM3.
|
Once innoextract completes, start VCMI Launcher and choose to place existing files. Select the ~/Downloads/HoMM3 directory. Once placing is complete, you can delete both offline installer files as well as ~/Downloads/HoMM3.
|
||||||
|
@ -19,15 +19,14 @@ iii) To install the .ipa file on your device do one of the following:
|
|||||||
- In AltStore go to >My Apps > press + in the top left corner. Select VCMI-iOS.ipa to install,
|
- In AltStore go to >My Apps > press + in the top left corner. Select VCMI-iOS.ipa to install,
|
||||||
- or drag and drop the .ipa file into your iOS device in iTunes
|
- or drag and drop the .ipa file into your iOS device in iTunes
|
||||||
|
|
||||||
|
|
||||||
## Step 2: Installing Heroes III data files
|
## Step 2: Installing Heroes III data files
|
||||||
|
|
||||||
If you bought HoMM3 on [GOG](https://www.gog.com/de/game/heroes_of_might_and_magic_3_complete_edition), you can download the files directly from the browser in the device.
|
If you bought HoMM3 on [GOG](https://www.gog.com/de/game/heroes_of_might_and_magic_3_complete_edition), you can download the files directly from the browser in the device.
|
||||||
|
|
||||||
Launch VCMI app on the device and the launcher will prompt two files to complete the installation. Select the **.bin** file first, then the **.exe** file. This may take a few seconds. Please be patient.
|
Launch VCMI app on the device and the launcher will prompt two files to complete the installation. Select the **.bin** file first, then the **.exe** file. This may take a few seconds. Please be patient.
|
||||||
|
|
||||||
|
|
||||||
## Step 3: Configuration settings
|
## Step 3: Configuration settings
|
||||||
|
|
||||||
Once you have installed VCMI and have the launcher opened, select Settings on the left bar. The following Video settings are recommended:
|
Once you have installed VCMI and have the launcher opened, select Settings on the left bar. The following Video settings are recommended:
|
||||||
|
|
||||||
- Lower reserved screen area to zero.
|
- Lower reserved screen area to zero.
|
||||||
@ -54,7 +53,9 @@ The easiest way to install the ipa on your device is to do one of the following:
|
|||||||
|
|
||||||
Alternatively, to install the signed ipa on your device, you can use Xcode or Apple Configurator (available on the Mac App Store for free). The latter also allows installing ipa from the command line, here's an example that assumes you have only 1 device connected to your Mac and the signed ipa is on your desktop:
|
Alternatively, to install the signed ipa on your device, you can use Xcode or Apple Configurator (available on the Mac App Store for free). The latter also allows installing ipa from the command line, here's an example that assumes you have only 1 device connected to your Mac and the signed ipa is on your desktop:
|
||||||
|
|
||||||
/Applications/Apple\ Configurator.app/Contents/MacOS/cfgutil install-app ~/Desktop/vcmi.ipa
|
```
|
||||||
|
/Applications/Apple\ Configurator.app/Contents/MacOS/cfgutil install-app ~/Desktop/vcmi.ipa
|
||||||
|
```
|
||||||
|
|
||||||
## Alternative Step 2: Installing Heroes III data files
|
## Alternative Step 2: Installing Heroes III data files
|
||||||
|
|
||||||
|
@ -23,6 +23,7 @@ This is list of all languages that are currently supported by VCMI. If your lang
|
|||||||
- Vietnamese
|
- Vietnamese
|
||||||
|
|
||||||
## Progress of the translations
|
## Progress of the translations
|
||||||
|
|
||||||
You can see the current progress of the different translations here:
|
You can see the current progress of the different translations here:
|
||||||
[Translation progress](https://github.com/vcmi/vcmi-translation-status)
|
[Translation progress](https://github.com/vcmi/vcmi-translation-status)
|
||||||
|
|
||||||
@ -32,17 +33,18 @@ The page will be automatically updated once a week.
|
|||||||
|
|
||||||
VCMI allows translating game data into languages other than English. In order to translate Heroes III in your language easiest approach is to:
|
VCMI allows translating game data into languages other than English. In order to translate Heroes III in your language easiest approach is to:
|
||||||
|
|
||||||
- Copy existing translation, such as English translation from here: https://github.com/vcmi-mods/h3-for-vcmi-englisation (delete sound and video folders)
|
- Copy existing translation, such as English translation from here: <https://github.com/vcmi-mods/h3-for-vcmi-englisation> (delete sound and video folders)
|
||||||
- Copy text-free images from here: https://github.com/vcmi-mods/empty-translation
|
- Copy text-free images from here: <https://github.com/vcmi-mods/empty-translation>
|
||||||
- Rename mod to indicate your language, preferred form is "(language)-translation"
|
- Rename mod to indicate your language, preferred form is "(language)-translation"
|
||||||
- Update mod.json to match your mod
|
- Update mod.json to match your mod
|
||||||
- Translate all texts strings from `game.json`, `campaigns.json` and `maps.json`
|
- Translate all texts strings from `game.json`, `campaigns.json` and `maps.json`
|
||||||
- Replace images in data and sprites with translated ones (or delete it if you don't want to translate them)
|
- Replace images in data and sprites with translated ones (or delete it if you don't want to translate them)
|
||||||
- If unicode characters needed for language: Create a submod with a free font like here: https://github.com/vcmi-mods/vietnamese-translation/tree/vcmi-1.4/vietnamese-translation/mods/VietnameseTrueTypeFonts
|
- If unicode characters needed for language: Create a submod with a free font like here: <https://github.com/vcmi-mods/vietnamese-translation/tree/vcmi-1.4/vietnamese-translation/mods/VietnameseTrueTypeFonts>
|
||||||
|
|
||||||
If you can't produce some content on your own (like the images or the sounds):
|
If you can't produce some content on your own (like the images or the sounds):
|
||||||
|
|
||||||
- Create a `README.md` file at the root of the mod
|
- Create a `README.md` file at the root of the mod
|
||||||
- Write into the file the translations and the <ins>detailled</ins> location
|
- Write into the file the translations and the **detailled** location
|
||||||
|
|
||||||
This way, a contributor that is not a native speaker can do it for you in the future.
|
This way, a contributor that is not a native speaker can do it for you in the future.
|
||||||
|
|
||||||
@ -57,8 +59,10 @@ This will export all strings from game into `Documents/My Games/VCMI/extracted/t
|
|||||||
To export maps and campaigns, use '/translate maps' command instead.
|
To export maps and campaigns, use '/translate maps' command instead.
|
||||||
|
|
||||||
### Video subtitles
|
### Video subtitles
|
||||||
|
|
||||||
It's possible to add video subtitles. Create a JSON file in `video` folder of translation mod with the name of the video (e.g. `H3Intro.json`):
|
It's possible to add video subtitles. Create a JSON file in `video` folder of translation mod with the name of the video (e.g. `H3Intro.json`):
|
||||||
```
|
|
||||||
|
```json5
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
"timeStart" : 5.640, // start time, seconds
|
"timeStart" : 5.640, // start time, seconds
|
||||||
@ -84,6 +88,7 @@ Before you start, make sure that you have copy of VCMI source code. If you are n
|
|||||||
### Translation of in-game data
|
### Translation of in-game data
|
||||||
|
|
||||||
In order to translate in-game data you need:
|
In order to translate in-game data you need:
|
||||||
|
|
||||||
- Add section with your language to `<VCMI>/Mods/VCMI/mod.json`, similar to other languages
|
- Add section with your language to `<VCMI>/Mods/VCMI/mod.json`, similar to other languages
|
||||||
- Copy English translation file in `<VCMI>/Mods/VCMI/config/vcmi/english.json` and rename it to name of your language. Note that while you can copy any language other than English, other files might not be up to date and may have missing strings.
|
- Copy English translation file in `<VCMI>/Mods/VCMI/config/vcmi/english.json` and rename it to name of your language. Note that while you can copy any language other than English, other files might not be up to date and may have missing strings.
|
||||||
- Translate copied file to your language.
|
- Translate copied file to your language.
|
||||||
@ -94,7 +99,7 @@ After this, you can set language in Launcher to your language and start game. Al
|
|||||||
|
|
||||||
VCMI Launcher and Map Editor use translation system provided by Qt framework so it requires slightly different approach than in-game translations:
|
VCMI Launcher and Map Editor use translation system provided by Qt framework so it requires slightly different approach than in-game translations:
|
||||||
|
|
||||||
- Install Qt Linguist. You can find find standalone version here: https://download.qt.io/linguist_releases/
|
- Install Qt Linguist. You can find find standalone version here: <https://download.qt.io/linguist_releases/>
|
||||||
- Open `<VCMI Sources>/launcher/translation/` directory, copy `english.ts` file and rename it to your language
|
- Open `<VCMI Sources>/launcher/translation/` directory, copy `english.ts` file and rename it to your language
|
||||||
- Open `<VCMI Sources>/launcher/CMakeLists.txt` file with a text editor. In there you need to find list of existing translation files and add new file to the list.
|
- Open `<VCMI Sources>/launcher/CMakeLists.txt` file with a text editor. In there you need to find list of existing translation files and add new file to the list.
|
||||||
- Launch Qt Linguist, select Open and navigate to your copied file
|
- Launch Qt Linguist, select Open and navigate to your copied file
|
||||||
@ -112,23 +117,27 @@ TODO: how to test translation locally
|
|||||||
The [AppStream](https://freedesktop.org/software/appstream/docs/chap-Metadata.html) [metainfo file](https://github.com/vcmi/vcmi/blob/develop/launcher/eu.vcmi.VCMI.metainfo.xml) is used for Linux software centers.
|
The [AppStream](https://freedesktop.org/software/appstream/docs/chap-Metadata.html) [metainfo file](https://github.com/vcmi/vcmi/blob/develop/launcher/eu.vcmi.VCMI.metainfo.xml) is used for Linux software centers.
|
||||||
|
|
||||||
It can be translated using a text editor or using [jdAppStreamEdit](https://flathub.org/apps/page.codeberg.JakobDev.jdAppStreamEdit):
|
It can be translated using a text editor or using [jdAppStreamEdit](https://flathub.org/apps/page.codeberg.JakobDev.jdAppStreamEdit):
|
||||||
|
|
||||||
- Install jdAppStreamEdit
|
- Install jdAppStreamEdit
|
||||||
- Open `<VCMI>/launcher/eu.vcmi.VCMI.metainfo.xml`
|
- Open `<VCMI>/launcher/eu.vcmi.VCMI.metainfo.xml`
|
||||||
- Translate and save the file
|
- Translate and save the file
|
||||||
|
|
||||||
##### Desktop file
|
##### Desktop file
|
||||||
|
|
||||||
- Edit `<VCMI>/launcher/vcmilauncher.desktop` and `<VCMI>/launcher/vcmieditor.desktop`
|
- Edit `<VCMI>/launcher/vcmilauncher.desktop` and `<VCMI>/launcher/vcmieditor.desktop`
|
||||||
- Add `GenericName[xyz]` and `Comment[xyz]` with your language code and translation
|
- Add `GenericName[xyz]` and `Comment[xyz]` with your language code and translation
|
||||||
|
|
||||||
##### Translation of Android Launcher
|
##### Translation of Android Launcher
|
||||||
|
|
||||||
- Copy `<VCMI>/android/vcmi-app/src/main/res/values/strings.xml` to `<VCMI>/android/vcmi-app/src/main/res/values-xyz/strings.xml` (`xyz` is your language code)
|
- Copy `<VCMI>/android/vcmi-app/src/main/res/values/strings.xml` to `<VCMI>/android/vcmi-app/src/main/res/values-xyz/strings.xml` (`xyz` is your language code)
|
||||||
- Translate this file
|
- Translate this file
|
||||||
|
|
||||||
See also here: https://developer.android.com/guide/topics/resources/localization
|
See also here: <https://developer.android.com/guide/topics/resources/localization>
|
||||||
|
|
||||||
### Submitting changes
|
### Submitting changes
|
||||||
|
|
||||||
Once you have finished with translation you need to submit these changes to vcmi team using git or Github Desktop
|
Once you have finished with translation you need to submit these changes to vcmi team using git or Github Desktop
|
||||||
|
|
||||||
- Commit all your changed files
|
- Commit all your changed files
|
||||||
- Push changes to your forked repository
|
- Push changes to your forked repository
|
||||||
- Create pull request in VCMI repository with your changes
|
- Create pull request in VCMI repository with your changes
|
||||||
@ -152,8 +161,10 @@ If your mod also contains maps or campaigns that you want to translate, then use
|
|||||||
If you want to update existing translation, you can use '/translate missing' command that will export only strings that were not translated
|
If you want to update existing translation, you can use '/translate missing' command that will export only strings that were not translated
|
||||||
|
|
||||||
### Translating mod information
|
### Translating mod information
|
||||||
|
|
||||||
In order to display information in Launcher in language selected by user add following block into your `mod.json`:
|
In order to display information in Launcher in language selected by user add following block into your `mod.json`:
|
||||||
```
|
|
||||||
|
```json5
|
||||||
"<language>" : {
|
"<language>" : {
|
||||||
"name" : "<translated name>",
|
"name" : "<translated name>",
|
||||||
"description" : "<translated description>",
|
"description" : "<translated description>",
|
||||||
@ -163,6 +174,7 @@ In order to display information in Launcher in language selected by user add fol
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
|
|
||||||
However, normally you don't need to use block for English. Instead, English text should remain in root section of your `mod.json` file, to be used when game can not find translated version.
|
However, normally you don't need to use block for English. Instead, English text should remain in root section of your `mod.json` file, to be used when game can not find translated version.
|
||||||
|
|
||||||
### Translating in-game strings
|
### Translating in-game strings
|
||||||
@ -174,7 +186,9 @@ Use any text editor (Notepad++ is recommended for Windows) and translate all str
|
|||||||
## Developers documentation
|
## Developers documentation
|
||||||
|
|
||||||
### Adding new languages
|
### Adding new languages
|
||||||
|
|
||||||
In order to add new language it needs to be added in multiple locations in source code:
|
In order to add new language it needs to be added in multiple locations in source code:
|
||||||
|
|
||||||
- Generate new .ts files for launcher and map editor, either by running `lupdate` with name of new .ts or by copying `english.ts` and editing language tag in the header.
|
- Generate new .ts files for launcher and map editor, either by running `lupdate` with name of new .ts or by copying `english.ts` and editing language tag in the header.
|
||||||
- Add new language into `lib/Languages.h` entry. This will trigger static_assert's in places that needs an update in code
|
- Add new language into `lib/Languages.h` entry. This will trigger static_assert's in places that needs an update in code
|
||||||
- Add new language into json schemas validation list - settings schema and mod schema
|
- Add new language into json schemas validation list - settings schema and mod schema
|
||||||
@ -187,7 +201,8 @@ Also, make full search for a name of an existing language to ensure that there a
|
|||||||
At the moment, build system will generate binary translation files (`.qs`) that can be opened by Qt.
|
At the moment, build system will generate binary translation files (`.qs`) that can be opened by Qt.
|
||||||
However, any new or changed lines will not be added into existing .ts files.
|
However, any new or changed lines will not be added into existing .ts files.
|
||||||
In order to update `.ts` files manually, open command line shell in `mapeditor` or `launcher` source directories and execute command
|
In order to update `.ts` files manually, open command line shell in `mapeditor` or `launcher` source directories and execute command
|
||||||
```
|
|
||||||
|
```sh
|
||||||
lupdate -no-obsolete * -ts translation/*.ts
|
lupdate -no-obsolete * -ts translation/*.ts
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -197,5 +212,6 @@ There *may* be a way to do the same via QtCreator UI or via CMake, if you find o
|
|||||||
### Updating translation of Launcher and Map Editor using new .ts file from translators
|
### Updating translation of Launcher and Map Editor using new .ts file from translators
|
||||||
|
|
||||||
Generally, this should be as simple as overwriting old files. Things that may be necessary if translation update is not visible in executable:
|
Generally, this should be as simple as overwriting old files. Things that may be necessary if translation update is not visible in executable:
|
||||||
|
|
||||||
- Rebuild subproject (map editor/launcher).
|
- Rebuild subproject (map editor/launcher).
|
||||||
- Regenerate translations via `lupdate -no-obsolete * -ts translation/*.ts`
|
- Regenerate translations via `lupdate -no-obsolete * -ts translation/*.ts`
|
||||||
|
Loading…
x
Reference in New Issue
Block a user