Source: modules/Config.js

  1. /**
  2. * @module modules/Config.js
  3. * @name Config
  4. * @copyright 2023 3Liz
  5. * @author DHONT René-Luc
  6. * @license MPL-2.0
  7. */
  8. import { ValidationError } from './Errors.js';
  9. import { deepFreeze } from './config/Tools.js';
  10. import { MetadataConfig } from './config/Metadata.js';
  11. import { OptionsConfig } from './config/Options.js';
  12. import { LayersConfig } from './config/Layer.js';
  13. import { BaseLayersConfig } from './config/BaseLayer.js';
  14. import { LocateByLayerConfig } from './config/Locate.js';
  15. import { AttributeLayersConfig } from './config/AttributeTable.js';
  16. import { TooltipLayersConfig } from './config/Tooltip.js';
  17. import { EditionLayersConfig } from './config/Edition.js';
  18. import { TimeManagerLayersConfig } from './config/TimeManager.js';
  19. import { FormFilterConfig } from './config/FormFilter.js';
  20. import { ThemesConfig } from './config/Theme.js';
  21. import { DatavizOptionsConfig, DatavizLayersConfig } from './config/Dataviz.js';
  22. import { buildLayerTreeConfig, LayerTreeGroupConfig } from './config/LayerTree.js';
  23. import { buildLayersOrder } from './config/LayersOrder.js';
  24. /**
  25. * @typedef WfsFeatureType
  26. * @type {object}
  27. * @property {string} Name - The layer typename.
  28. * @property {string} Title - The layer title.
  29. * @property {string} SRS - The layer CRS code.
  30. * @property {number[]} LatLongBoundingBox - The layer bounding box in the layer CRS
  31. */
  32. /**
  33. * @class
  34. * @name Config
  35. */
  36. export class Config {
  37. /**
  38. * Create a new Config object
  39. * @param {object} cfg - the lizmap config object
  40. * @param {object} wmsCapabilities - the WMS capabilities
  41. * @param {object} [wfsCapabilities] - the WFS capabilities
  42. */
  43. constructor(cfg, wmsCapabilities, wfsCapabilities) {
  44. if (!cfg || typeof cfg !== "object") {
  45. throw new ValidationError('The config is not an Object! It\'s '+(typeof cfg));
  46. }
  47. if (Object.getOwnPropertyNames(cfg).length == 0) {
  48. throw new ValidationError('The config is empty!');
  49. }
  50. if (!wmsCapabilities || typeof wmsCapabilities !== "object") {
  51. throw new ValidationError('The WMS Capabilities is not an Object! It\'s '+(typeof wmsCapabilities));
  52. }
  53. if (Object.getOwnPropertyNames(wmsCapabilities).length == 0) {
  54. throw new ValidationError('The WMS Capabilities is empty!');
  55. }
  56. this._theConfig = null;
  57. this._theWmsCapabilities = null;
  58. this._theWfsCapabilities = null;
  59. this._options = null;
  60. this._layers = null;
  61. this._layerTree = null;
  62. this._invalidLayers = null;
  63. this._baselayers = null;
  64. this._layersOrder = null;
  65. this._hasMetadata = true;
  66. this._metadata = null;
  67. this._hasLocateByLayer = true;
  68. this._locateByLayer = null;
  69. this._hasAttributeLayers = true;
  70. this._attributeLayers = null;
  71. this._hasTimemanagerLayers = true;
  72. this._timemanagerLayers = null;
  73. this._hasRelations = true;
  74. this._hasPrintTemplates = true;
  75. this._hasTooltipLayers = true;
  76. this._tooltipLayers = null;
  77. this._hasEditionLayers = true;
  78. this._editionLayers = null;
  79. this._hasFormFilterLayers = true;
  80. this._formFilterLayers = null;
  81. this._hasLoginFilteredLayers = true;
  82. this._hasThemes = true;
  83. this._themes = null;
  84. this._hasDatavizConfig = true;
  85. this._datavizLayers = null;
  86. this._datavizOptions = null;
  87. const theConfig = deepFreeze(cfg);
  88. // checking config
  89. const mandatoryConfigProperties = [
  90. 'options',
  91. 'layers',
  92. 'datavizLayers' // needed for locale property to build plot
  93. ];
  94. for (const prop of mandatoryConfigProperties) {
  95. if (!theConfig.hasOwnProperty(prop)) {
  96. throw new ValidationError('No `' + prop + '` in the config!');
  97. }
  98. }
  99. const theWmsCapabilities = deepFreeze(wmsCapabilities);
  100. // checking WMS Capabilities
  101. const mandatoryWmsCapabilitiesgProperties = [
  102. 'Capability', //needed for layers tree
  103. ];
  104. for (const prop of mandatoryWmsCapabilitiesgProperties) {
  105. if (!theWmsCapabilities.hasOwnProperty(prop)) {
  106. throw new ValidationError('No `' + prop + '` in the WMS Capabilities!');
  107. }
  108. }
  109. this._theConfig = theConfig;
  110. this._theWmsCapabilities = theWmsCapabilities;
  111. if (wfsCapabilities) {
  112. this._theWfsCapabilities = deepFreeze(wfsCapabilities);
  113. }
  114. const optionalConfigProperties = [
  115. 'metadata',
  116. 'locateByLayer',
  117. 'attributeLayers',
  118. 'timemanagerLayers',
  119. 'relations',
  120. 'printTemplates',
  121. 'tooltipLayers',
  122. 'editionLayers',
  123. 'formFilterLayers',
  124. 'loginFilteredLayers',
  125. 'themes'
  126. ];
  127. for (const prop of optionalConfigProperties) {
  128. if (!theConfig.hasOwnProperty(prop)
  129. || Object.getOwnPropertyNames(theConfig[prop]).length == 0) {
  130. this['_has'+prop.charAt(0).toUpperCase() + prop.slice(1)] = false;
  131. }
  132. }
  133. // check datavizConfig
  134. if ((!theConfig.datavizLayers.hasOwnProperty('layers')
  135. || Object.getOwnPropertyNames(theConfig.datavizLayers.layers).length == 0)
  136. && (!theConfig.datavizLayers.hasOwnProperty('dataviz')
  137. || Object.getOwnPropertyNames(theConfig.datavizLayers.dataviz).length == 0)) {
  138. this._hasDatavizConfig = false;
  139. }
  140. }
  141. /**
  142. * Config options
  143. * @type {OptionsConfig}
  144. */
  145. get options() {
  146. if (this._options != null) {
  147. return this._options;
  148. }
  149. this._options = new OptionsConfig(this._theConfig.options);
  150. return this._options;
  151. }
  152. /**
  153. * Config layers
  154. * @type {LayersConfig}
  155. */
  156. get layers() {
  157. if (this._layers != null) {
  158. return this._layers;
  159. }
  160. this._layers = new LayersConfig(this._theConfig.layers);
  161. return this._layers;
  162. }
  163. /**
  164. * Root tree layer group
  165. * @type {LayerTreeGroupConfig}
  166. */
  167. get layerTree() {
  168. if (this._layerTree == null) {
  169. this._invalidLayers = [];
  170. this._layerTree = buildLayerTreeConfig(
  171. this._theWmsCapabilities.Capability.Layer,
  172. this.layers,
  173. this._invalidLayers,
  174. );
  175. }
  176. return this._layerTree;
  177. }
  178. /**
  179. * List of invalid layers, not found in the Lizmap configuration file, but found in the WMS GetCapabilities.
  180. * @type {string[]}
  181. */
  182. get invalidLayersNotFoundInCfg() {
  183. if (this._invalidLayers == null) {
  184. this.layerTree;
  185. }
  186. return this._invalidLayers;
  187. }
  188. /**
  189. * Config base layers
  190. * @type {BaseLayersConfig}
  191. */
  192. get baseLayers() {
  193. if (this._baselayers != null) {
  194. return this._baselayers;
  195. }
  196. let baseLayersCfg = {};
  197. if (this._theConfig.hasOwnProperty('baseLayers')) {
  198. baseLayersCfg = this._theConfig.baseLayers;
  199. }
  200. let baseLayerTreeItem = null;
  201. for (const layerTreeItem of this.layerTree.getChildren()) {
  202. if ( layerTreeItem.name.toLowerCase() == 'baselayers') {
  203. baseLayerTreeItem = layerTreeItem;
  204. break;
  205. }
  206. }
  207. let hiddenTreeItem = null;
  208. for (const layerTreeItem of this.layerTree.getChildren()) {
  209. if ( layerTreeItem.name.toLowerCase() == 'hidden') {
  210. hiddenTreeItem = layerTreeItem;
  211. break;
  212. }
  213. }
  214. this._baselayers = new BaseLayersConfig(baseLayersCfg, this._theConfig.options, this.layers, baseLayerTreeItem, hiddenTreeItem);
  215. return this._baselayers;
  216. }
  217. /**
  218. * Layer names displaying order like in QGIS
  219. *
  220. * The first one in this list is the top one in the map
  221. * The last one in this list is the bottom one in the map
  222. * @type {string[]}
  223. */
  224. get layersOrder() {
  225. if (this._layersOrder == null) {
  226. this._layersOrder = buildLayersOrder(this._theConfig, this.layerTree);
  227. }
  228. return [...this._layersOrder];
  229. }
  230. /**
  231. * The list of format for file export
  232. * @type {string[]}
  233. */
  234. get vectorLayerResultFormat() {
  235. const formats = [];
  236. if ( this._theWfsCapabilities == null ){
  237. return formats;
  238. }
  239. for (const request of this._theWfsCapabilities.Capability.Request) {
  240. if (request.name != 'GetFeature') {
  241. continue;
  242. }
  243. return request.ResultFormat
  244. }
  245. return formats;
  246. }
  247. /**
  248. * The list of WFS feature type
  249. * @type {WfsFeatureType[]}
  250. */
  251. get vectorLayerFeatureTypeList() {
  252. const featureTypes = [];
  253. if ( this._theWfsCapabilities == null ){
  254. return featureTypes;
  255. }
  256. if ( this._theWfsCapabilities.FeatureTypeList &&
  257. this._theWfsCapabilities.FeatureTypeList.FeatureType
  258. ) {
  259. return this._theWfsCapabilities.FeatureTypeList.FeatureType;
  260. }
  261. return featureTypes;
  262. }
  263. /**
  264. * Config metadata
  265. * @type {MetadataConfig}
  266. */
  267. get metadata() {
  268. if (this._metadata != null) {
  269. return this._metadata;
  270. }
  271. if (this._hasMetadata) {
  272. this._metadata = new MetadataConfig(this._theConfig.metadata);
  273. } else {
  274. this._metadata = new MetadataConfig();
  275. }
  276. return this._metadata;
  277. }
  278. /**
  279. * Locate by layer config is defined
  280. * @type {boolean}
  281. */
  282. get hasLocateByLayer() {
  283. return this._hasLocateByLayer;
  284. }
  285. /**
  286. * Config locateByLayer
  287. * @type {LocateByLayerConfig|null}
  288. */
  289. get locateByLayer() {
  290. if (this._hasLocateByLayer) {
  291. if (this._locateByLayer != null) {
  292. return this._locateByLayer;
  293. }
  294. this._locateByLayer = new LocateByLayerConfig(this._theConfig.locateByLayer);
  295. }
  296. return this._locateByLayer;
  297. }
  298. /**
  299. * Attribute layers config is defined
  300. * @type {boolean}
  301. */
  302. get hasAttributeLayers() {
  303. return this._hasAttributeLayers;
  304. }
  305. /**
  306. * Config attribueLayers
  307. * @type {AttributeLayersConfig|null}
  308. */
  309. get attributeLayers() {
  310. if (this._hasAttributeLayers) {
  311. if (this._attributeLayers != null) {
  312. return this._attributeLayers;
  313. }
  314. this._attributeLayers = new AttributeLayersConfig(this._theConfig.attributeLayers);
  315. }
  316. return this._attributeLayers;
  317. }
  318. /**
  319. * Time manager config is defined
  320. * @type {boolean}
  321. */
  322. get hasTimemanagerLayers() {
  323. return this._hasTimemanagerLayers;
  324. }
  325. /**
  326. * Config timemanagerLayers
  327. * @type {AttributeLayersConfig|null}
  328. */
  329. get timemanagerLayers() {
  330. if (this._hasTimemanagerLayers) {
  331. if (this._timemanagerLayers != null) {
  332. return this._timemanagerLayers;
  333. }
  334. this._timemanagerLayers = new TimeManagerLayersConfig(this._theConfig.timemanagerLayers);
  335. }
  336. return this._timemanagerLayers;
  337. }
  338. /**
  339. * Relations config is defined
  340. * @type {boolean}
  341. */
  342. get hasRelations() {
  343. return this._hasRelations;
  344. }
  345. /**
  346. * Print templates config is defined
  347. * @type {boolean}
  348. */
  349. get hasPrintTemplates() {
  350. return this._hasPrintTemplates;
  351. }
  352. /**
  353. * Tooltip layers config is defined
  354. * @type {boolean}
  355. */
  356. get hasTooltipLayers() {
  357. return this._hasTooltipLayers;
  358. }
  359. /**
  360. * Config tooltipLayers
  361. * @type {TooltipLayersConfig|null}
  362. */
  363. get tooltipLayers() {
  364. if (this._hasTooltipLayers) {
  365. if (this._tooltipLayers != null) {
  366. return this._tooltipLayers;
  367. }
  368. this._tooltipLayers = new TooltipLayersConfig(this._theConfig.tooltipLayers);
  369. }
  370. return this._tooltipLayers;
  371. }
  372. /**
  373. * Edition layers config is defined
  374. * @type {boolean}
  375. */
  376. get hasEditionLayers() {
  377. return this._hasEditionLayers;
  378. }
  379. /**
  380. * Config editionLayers
  381. * @type {EditionLayersConfig|null}
  382. */
  383. get editionLayers() {
  384. if (this._hasEditionLayers) {
  385. if (this._editionLayers != null) {
  386. return this._editionLayers;
  387. }
  388. this._editionLayers = new EditionLayersConfig(this._theConfig.editionLayers);
  389. }
  390. return this._editionLayers;
  391. }
  392. /**
  393. * Form filter layers config is defined
  394. * @type {boolean}
  395. */
  396. get hasFormFilterLayers() {
  397. return this._hasFormFilterLayers;
  398. }
  399. /**
  400. * Config formFilterLayers
  401. * @type {FormFilterConfig|null}
  402. */
  403. get formFilterLayers() {
  404. if (this.hasFormFilterLayers) {
  405. if (this._formFilterLayers != null) {
  406. return this._formFilterLayers;
  407. }
  408. this._formFilterLayers = new FormFilterConfig(this._theConfig.formFilterLayers);
  409. }
  410. return this._formFilterLayers;
  411. }
  412. /**
  413. * Login filtered layers config is defined
  414. * @type {boolean}
  415. */
  416. get hasLoginFilteredLayers() {
  417. return this._hasLoginFilteredLayers;
  418. }
  419. /**
  420. * Themes config is defined
  421. * @type {boolean}
  422. */
  423. get hasThemes() {
  424. return this._hasThemes;
  425. }
  426. /**
  427. * Config themes
  428. * @type {ThemesConfig|null}
  429. */
  430. get themes() {
  431. if (this.hasThemes) {
  432. if (this._themes != null) {
  433. return this._themes;
  434. }
  435. this._themes = new ThemesConfig(this._theConfig.themes);
  436. }
  437. return this._themes;
  438. }
  439. /**
  440. * Dataviz locale
  441. * @type {string}
  442. */
  443. get datavizLocale() {
  444. return this._theConfig.datavizLayers.locale;
  445. }
  446. /**
  447. * Dataviz config is defined
  448. * @type {boolean}
  449. */
  450. get hasDatavizConfig() {
  451. return this._hasDatavizConfig;
  452. }
  453. /**
  454. * Config datavizLayers
  455. * @type {DatavizLayersConfig|null}
  456. */
  457. get datavizLayers() {
  458. if (this._hasDatavizConfig) {
  459. if (this._datavizLayers != null) {
  460. return this._datavizLayers;
  461. }
  462. this._datavizLayers = new DatavizLayersConfig(this._theConfig.datavizLayers.layers);
  463. }
  464. return this._datavizLayers;
  465. }
  466. /**
  467. * Config datavizOptions
  468. * @type {DatavizOptionsConfig|null}
  469. */
  470. get datavizOptions() {
  471. if (this._hasDatavizConfig) {
  472. if (this._datavizOptions != null) {
  473. return this._datavizOptions;
  474. }
  475. this._datavizOptions = new DatavizOptionsConfig(this._theConfig.datavizLayers.dataviz);
  476. }
  477. return this._datavizOptions;
  478. }
  479. }