summaryrefslogtreecommitdiff
path: root/include/mbgl/style/conversion/data_driven_property_value.hpp
blob: 07ed201c99578bced9f913601814c214001c625f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
#pragma once

#include <mbgl/style/data_driven_property_value.hpp>
#include <mbgl/style/conversion.hpp>
#include <mbgl/style/conversion/constant.hpp>
#include <mbgl/style/conversion/function.hpp>
#include <mbgl/style/expression/is_expression.hpp>
#include <mbgl/style/expression/is_constant.hpp>
#include <mbgl/style/expression/find_zoom_curve.hpp>
#include <mbgl/style/expression/literal.hpp>
#include <mbgl/style/expression/value.hpp>
#include <mbgl/style/expression/parsing_context.hpp>

#include <unordered_set>


namespace mbgl {
namespace style {
namespace conversion {

template <class T>
struct Converter<DataDrivenPropertyValue<T>> {

    optional<DataDrivenPropertyValue<T>> operator()(const Convertible& value, Error& error) const {
        using namespace mbgl::style::expression;
        
        if (isUndefined(value)) {
            return DataDrivenPropertyValue<T>();
        } else if (isExpression(value)) {
            ParsingContext ctx(valueTypeToExpressionType<T>());
            ParseResult expression = ctx.parseLayerPropertyExpression(value);
            if (!expression) {
                error = { ctx.getCombinedErrors() };
                return {};
            }
            
            bool featureConstant = isFeatureConstant(**expression);
            bool zoomConstant = isZoomConstant(**expression);
            
            if (featureConstant && !zoomConstant) {
                return DataDrivenPropertyValue<T>(CameraFunction<T>(std::move(*expression)));
            } else if (!featureConstant && zoomConstant) {
                return DataDrivenPropertyValue<T>(SourceFunction<T>(std::move(*expression)));
            } else if (!featureConstant && !zoomConstant) {
                return DataDrivenPropertyValue<T>(CompositeFunction<T>(std::move(*expression)));
            } else {
                auto literal = dynamic_cast<Literal*>(expression->get());
                assert(literal);
                optional<T> constant = fromExpressionValue<T>(literal->getValue());
                if (!constant) {
                    return {};
                }
                return DataDrivenPropertyValue<T>(*constant);
            }
        } else if (!isObject(value)) {
            optional<T> constant = convert<T>(value, error);
            if (!constant) {
                return {};
            }
            return DataDrivenPropertyValue<T>(*constant);
        } else if (!objectMember(value, "property")) {
            optional<CameraFunction<T>> function = convert<CameraFunction<T>>(value, error);
            if (!function) {
                return {};
            }
            return DataDrivenPropertyValue<T>(*function);
        } else {
            optional<CompositeFunction<T>> composite = convert<CompositeFunction<T>>(value, error);
            if (composite) {
                return DataDrivenPropertyValue<T>(*composite);
            }
            optional<SourceFunction<T>> source = convert<SourceFunction<T>>(value, error);
            if (!source) {
                return {};
            }
            return DataDrivenPropertyValue<T>(*source);
        }
    }
};

} // namespace conversion
} // namespace style
} // namespace mbgl