forked from MapComplete/MapComplete
		
	Refactoring: move documentation about tagsFilters into TagUtils
This commit is contained in:
		
							parent
							
								
									00776c1e6b
								
							
						
					
					
						commit
						48e976e6b7
					
				
					 1 changed files with 209 additions and 6 deletions
				
			
		|  | @ -13,13 +13,201 @@ type Tags = Record<string, string> | ||||||
| export type UploadableTag = Tag | SubstitutingTag | And | export type UploadableTag = Tag | SubstitutingTag | And | ||||||
| 
 | 
 | ||||||
| export class TagUtils { | export class TagUtils { | ||||||
|  |     public static readonly comparators: ReadonlyArray<[string, (a: number, b: number) => boolean]> = | ||||||
|  |         [ | ||||||
|  |             ["<=", (a, b) => a <= b], | ||||||
|  |             [">=", (a, b) => a >= b], | ||||||
|  |             ["<", (a, b) => a < b], | ||||||
|  |             [">", (a, b) => a > b], | ||||||
|  |         ] | ||||||
|  |     public static modeDocumentation: Record< | ||||||
|  |         string, | ||||||
|  |         { name: string; docs: string; uploadable?: boolean; overpassSupport: boolean } | ||||||
|  |     > = { | ||||||
|  |         "=": { | ||||||
|  |             name: "strict equality", | ||||||
|  |             uploadable: true, | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: | ||||||
|  |                 "Strict equality is denoted by `key=value`. This key matches __only if__ the keypair is present exactly as stated.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "**Only normal tags (eventually in an `and`) can be used in places where they are uploaded**. Normal tags are used in " + | ||||||
|  |                 "the `mappings` of a [TagRendering] (unless `hideInAnswer` is specified), they are used in `addExtraTags` of [Freeform] " + | ||||||
|  |                 "and are used in the `tags`-list of a preset.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "If a different kind of tag specification is given, your theme will fail to parse.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "### If key is not present\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "If you want to check if a key is not present, use `key=` (pronounce as *key is empty*). A tag collection will match this\n" + | ||||||
|  |                 "if `key` is missing or if `key` is a literal empty value.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "### Removing a key\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "If a key should be deleted in the OpenStreetMap-database, specify `key=` as well. This can be used e.g. to remove a\n" + | ||||||
|  |                 "fixme or value from another mapping if another field is filled out.", | ||||||
|  |         }, | ||||||
|  |         "!=": { | ||||||
|  |             name: "strict not equals", | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: | ||||||
|  |                 "To check if a key does _not_ equal a certain value, use `key!=value`. This is converted behind the scenes\n" + | ||||||
|  |                 "to `key!~^value$`\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "If `key` is not present or empty, this will match too.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "### If key is present\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "This implies that, to check if a key is present, `key!=` can be used. This will only match if the key is present and not\n" + | ||||||
|  |                 "empty.", | ||||||
|  |         }, | ||||||
|  |         "~": { | ||||||
|  |             name: "Value matches regex", | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: | ||||||
|  |                 "A tag can also be tested against a regex with `key~regex`. Note that this regex __must match__ the entire value. If the\n" + | ||||||
|  |                 "value is allowed to appear anywhere as substring, use `key~.*regex.*`.\n" + | ||||||
|  |                 "The regex is put within braces as to prevent runaway values.\n" + | ||||||
|  |                 "\nUse `key~*` to indicate that any value is allowed. This is effectively the check that the attribute is present (defined _and_ not empty)." + | ||||||
|  |                 "\n" + | ||||||
|  |                 "Regexes will match the newline character with `.` too - the `s`-flag is enabled by default.", | ||||||
|  |         }, | ||||||
|  |         "~i~": { | ||||||
|  |             name: "Value matches case-invariant regex", | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: "A tag can also be tested against a regex with `key~i~regex`, where the case of the value will be ignored. The regex is still matched against the _entire_ value", | ||||||
|  |         }, | ||||||
|  |         "!~": { | ||||||
|  |             name: "Value should _not_ match regex", | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: | ||||||
|  |                 "A tag can also be tested against a regex with `key!~regex`. This filter will match if the value does *not* match the regex. " + | ||||||
|  |                 "\n If the\n" + | ||||||
|  |                 "value is allowed to appear anywhere as substring, use `key~.*regex.*`.\n" + | ||||||
|  |                 "The regex is put within braces as to prevent runaway values.\n", | ||||||
|  |         }, | ||||||
|  |         "!~i~": { | ||||||
|  |             name: "Value does *not* match case-invariant regex", | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: "A tag can also be tested against a regex with `key~i~regex`, where the case of the value will be ignored. The regex is still matched against the _entire_ value. This filter returns true if the value does *not* match", | ||||||
|  |         }, | ||||||
|  |         "~~": { | ||||||
|  |             name: "Key and value should match given regex", | ||||||
|  |             overpassSupport: true, | ||||||
|  |             docs: "Both the `key` and `value` part of this specification are interpreted as regexes, both the key and value musth completely match their respective regexes", | ||||||
|  |         }, | ||||||
|  |         ":=": { | ||||||
|  |             name: "Substitute `{some_key}` should match `key`", | ||||||
|  |             overpassSupport: false, | ||||||
|  |             docs: | ||||||
|  |                 "**This is an advanced feature - use with caution**\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "Some tags are automatically set or calculated - see [CalculatedTags](CalculatedTags.md) for an entire overview. If one\n" + | ||||||
|  |                 "wants to apply such a value as tag, use a substituting-tag such, for example`survey:date:={_date:now}`. Note that the\n" + | ||||||
|  |                 "separator between key and value here is `:=`. The text between `{` and `}` is interpreted as a key, and the respective\n" + | ||||||
|  |                 "value is substituted into the string.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "One can also append, e.g. `key:={some_key} fixed text {some_other_key}`.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "An assigning tag _cannot_ be used to query OpenStreetMap/Overpass.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "If using a key or variable which might not be defined, add a condition in the mapping to hide the option. This is\n" + | ||||||
|  |                 "because, if `some_other_key` is not defined, one might actually upload the literal text `key={some_other_key}` to OSM -\n" + | ||||||
|  |                 "which we do not want.\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "To mitigate this, use:\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "```json\n" + | ||||||
|  |                 "{\n" + | ||||||
|  |                 '    "mappings": [\n' + | ||||||
|  |                 "        {\n" + | ||||||
|  |                 '            "if":"key:={some_other_key}",\n' + | ||||||
|  |                 '            "then": "...",\n' + | ||||||
|  |                 '            "hideInAnswer": "some_other_key="\n' + | ||||||
|  |                 "        }\n" + | ||||||
|  |                 "    ]\n" + | ||||||
|  |                 "}\n" + | ||||||
|  |                 "```\n" + | ||||||
|  |                 "\n" + | ||||||
|  |                 "One can use `key!:=prefix-{other_key}-postfix` as well, to match if `key` is _not_ the same\n" + | ||||||
|  |                 "as `prefix-{other_key}-postfix` (with `other_key` substituted by the value)", | ||||||
|  |         }, | ||||||
|  |         "!:=": { | ||||||
|  |             name: "Substitute `{some_key}` should not match `key`", | ||||||
|  |             overpassSupport: false, | ||||||
|  |             docs: "See `:=`, except that this filter is inverted", | ||||||
|  |         }, | ||||||
|  |     } | ||||||
|     private static keyCounts: { keys: any; tags: any } = key_counts |     private static keyCounts: { keys: any; tags: any } = key_counts | ||||||
|     private static comparators: [string, (a: number, b: number) => boolean][] = [ |     public static readonly numberAndDateComparisonDocs = | ||||||
|         ["<=", (a, b) => a <= b], |         "If the value of a tag is a number (e.g. `key=42`), one can use a filter `key<=42`, `key>=35`, `key>40` or `key<50` to\n" + | ||||||
|         [">=", (a, b) => a >= b], |         "match this, e.g. in conditions for renderings. These tags cannot be used to generate an answer nor can they be used to\n" + | ||||||
|         ["<", (a, b) => a < b], |         "request data upstream from overpass.\n" + | ||||||
|         [">", (a, b) => a > b], |         "\n" + | ||||||
|     ] |         "Note that the value coming from OSM will first be stripped by removing all non-numeric characters. For\n" + | ||||||
|  |         "example, `length=42 meter` will be interpreted as `length=42` and will thus match `length<=42` and `length>=42`. In\n" + | ||||||
|  |         "special circumstances (e.g. `surface_area=42 m2` or `length=100 feet`), this will result in erronous\n" + | ||||||
|  |         "values (`surface=422` or if a length in meters is compared to). However, this can be partially alleviated by using '\n" + | ||||||
|  |         "Units' to rewrite to a default format.\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "Dates can be compared with the same expression: `somekey<2022-06-22` will match if `somekey` is a date and is smaller\n" + | ||||||
|  |         "then 22nd june '22." | ||||||
|  | 
 | ||||||
|  |     public static readonly logicalOperator = | ||||||
|  |         "\n" + | ||||||
|  |         "## Logical operators\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "One can combine multiple tags by using `and` or `or`, e.g.:\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "```json\n" + | ||||||
|  |         "{\n" + | ||||||
|  |         '  "osmTags": {\n' + | ||||||
|  |         '    "or": [\n' + | ||||||
|  |         '      "amenity=school",\n' + | ||||||
|  |         '      "amenity=kindergarten"\n' + | ||||||
|  |         "    ]\n" + | ||||||
|  |         "  }\n" + | ||||||
|  |         "}\n" + | ||||||
|  |         "```\n" | ||||||
|  | 
 | ||||||
|  |     public static readonly intro = | ||||||
|  |         "Tags format\n" + | ||||||
|  |         "=============\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "When creating the `json` file describing your layer or theme, you'll have to add a few tags to describe what you want.\n" + | ||||||
|  |         "This document gives an overview of what every expression means and how it behaves in edge cases.\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "If the schema-files note a type [`TagConfigJson`](https://github.com/pietervdvn/MapComplete/blob/develop/Models/ThemeConfig/Json/TagConfigJson.ts), you can use one of these values.\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "In some cases, not every type of tags-filter can be used. For example,  _rendering_ an option with a regex is\n" + | ||||||
|  |         'fine (`"if": "brand~[Bb]randname", "then":" The brand is Brandname"`); but this regex can not be used to write a value\n' + | ||||||
|  |         "into the database. The theme loader will however refuse to work with such inconsistencies and notify you of this while\n" + | ||||||
|  |         "you are building your theme.\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "Example\n" + | ||||||
|  |         "-------\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "This example shows the most common options on how to specify tags:\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "```json\n" + | ||||||
|  |         "{\n" + | ||||||
|  |         '  "and": [\n' + | ||||||
|  |         '    "key=value",\n' + | ||||||
|  |         "    {\n" + | ||||||
|  |         '      "or": [\n' + | ||||||
|  |         '        "other_key=value",\n' + | ||||||
|  |         '        "other_key=some_other_value"\n' + | ||||||
|  |         "      ]\n" + | ||||||
|  |         "    },\n" + | ||||||
|  |         '    "key_which_should_be_missing=",\n' + | ||||||
|  |         '    "key_which_should_have_a_value~*",\n' + | ||||||
|  |         '    "key~.*some_regex_a*_b+_[a-z]?",\n' + | ||||||
|  |         '    "height<1"\n' + | ||||||
|  |         "  ]\n" + | ||||||
|  |         "}\n" + | ||||||
|  |         "```\n" + | ||||||
|  |         "\n" + | ||||||
|  |         "\n" | ||||||
| 
 | 
 | ||||||
|     static KVtoProperties(tags: Tag[]): Record<string, string> { |     static KVtoProperties(tags: Tag[]): Record<string, string> { | ||||||
|         const properties: Record<string, string> = {} |         const properties: Record<string, string> = {} | ||||||
|  | @ -688,4 +876,19 @@ export class TagUtils { | ||||||
|         } |         } | ||||||
|         return " (" + joined + ") " |         return " (" + joined + ") " | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public static generateDocs(): string { | ||||||
|  |         return [ | ||||||
|  |             TagUtils.intro, | ||||||
|  |             ...Object.keys(TagUtils.modeDocumentation).map((mode) => { | ||||||
|  |                 const doc = TagUtils.modeDocumentation[mode] | ||||||
|  |                 return ["", "## `" + mode + "` " + doc.name, "", doc.docs, "", ""].join("\n") | ||||||
|  |             }), | ||||||
|  |             "## " + | ||||||
|  |                 TagUtils.comparators.map((comparator) => "`" + comparator[0] + "`").join(" ") + | ||||||
|  |                 " Logical comparators", | ||||||
|  |             TagUtils.numberAndDateComparisonDocs, | ||||||
|  |             TagUtils.logicalOperator, | ||||||
|  |         ].join("\n") | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue