L.TileLayer.GoogleBase = L.TileLayer.extend({
    getTileUrl: function (coords) {
        return '/Menggu/GetMap?type=47626774&zoom={0}&x={1}&y={2}'.format(coords.z, coords.x, coords.y);
    }
});

L.TileLayer.GoogleDetail = L.TileLayer.extend({
    getTileUrl: function (coords) {
        return '/Menggu/GetMap?type=1024577166&zoom={0}&x={1}&y={2}'.format(coords.z, coords.x, coords.y);
    }
});

L.tileLayer.googleBase = function () {
    return new L.TileLayer.GoogleBase();
};

L.tileLayer.googleDetail = function () {
    return new L.TileLayer.GoogleDetail();
};

var Map = function (parent) {
    this.Parent = parent;
    this.WindTemplate = null;
    this.MultiLayers = false;
    this.isMark = false;
    this.isSwitch = false;
    this.isSelected = false;
    this.showTimeTags = true;
    this.currentButton = {
        cursorSelected: true,
        markSelected: true,
        drawLineSelected: true,
        clearSelected: true,
        tagSelected: true,
        exportSelected: true
    };

    this.InfoPoint = new InfoPoint(this);
    this.SelectPoint = new SelectPoint(this);
    this.LatLngSwitch = new LatLngSwitch(this);

    this.Startup = function () {
        this.CreateMap();
        this.InitCapture();
        this.LoadWindTemplate();
        this.LoadInfoPoint();
        this.InfoPoint.Startup();
        this.SelectPoint.Startup();
        this.LatLngSwitch.Startup();

        $('#select-point-button').on('click', this.onSelectPointClick.bind(this));
        $('#mark-button').on('click', this.onMarkClick.bind(this));
        $('#switch-button').on('click', this.onSwitchClick.bind(this));
        $('#draw-button').on('click', this.OnDrawButtonClick.bind(this));
        $('#clear-button').on('click', this.OnClearButtonClick.bind(this));
        $('#particle-switch a').on('click', this.OnParticleButtonClick.bind(this));
        $('#label-switch').on('click', this.OnRemoveMarkersButtonClick.bind(this));

        $('.control-btn-groups .control-item').find('.icon').on('click', this.OnItemClick.bind(this));
    };

    this.CreateMap = function () {
        this.map = L.map('map', {
            zoomControl: false,
            minZoom: 4,
            maxZoom: 16,
            zoomDelta: 0.05
        }).setView([40.854662, 111.746303], 7);

        L.tileLayer.googleBase().addTo(this.map);
        L.tileLayer.googleDetail().addTo(this.map);
        $('.leaflet-control-attribution').hide();

        $('.latlng').text('当前经纬度:{0}, {1}'.format(this.getLatLng(111.746303), this.getLatLng(40.854662)));
        this.map.on('mousemove', this.OnMapMove.bind(this));
    };

    this.CenterMap = function (lat, lng) {
        this.map.setView([lat, lng], 11);
    };

    this.OnMapMove = function (e) {
        $('.latlng').text('当前经纬度:{0}, {1}'.format(this.getLatLng(e.latlng.lng), this.getLatLng(e.latlng.lat)))
    };

    this.getLatLng = function (value) {
        value = value.toString()
        let index = value.indexOf('.')
        if (index !== -1) {
            value = value.substring(0, 6 + index + 1)
        } else {
            value = value.substring(0)
        }
        return parseFloat(value).toFixed(6)
    }

    this.OnParticleButtonClick = function () {
        // Toggle switch on/off
        var parent = $(event.target).parent();
        parent.toggleClass('switch-on');

        // Hide wind layer
        if (!parent.hasClass('switch-on')) {
            this.HideWindLayer();
            return;
        }

        // Add wind layer
        var url = 'http://{0}/bj/getwind/{1}'.format(Config.ApiRoot, this.GetWindFileName());
        $.getJSON(url).done(function (data) {
            this.WindTemplate[0].data = [];
            this.WindTemplate[1].data = [];

            $(data).each(function (index, item) {
                this.WindTemplate[0].data.push(item.u);
                this.WindTemplate[1].data.push(item.v);
            }.bind(this));

            this.ShowWindLayer(this.WindTemplate);
        }.bind(this)).fail(function (xhr, status, error) {
            var message = "Json文件加载失败: Status={0}, Error={1}".format(status, error);
            this.MessageBox("error", message);
        }.bind(this));
    };

    this.OnRemoveMarkersButtonClick = function (event) {
        this.showTimeTags = !this.showTimeTags;

        if (this.showTimeTags) {
            $('.point').show();
            $('.time-label').show();
            $(event.target).removeClass('default-text');
        } else {
            $('.point').hide();
            $('.time-label').hide();
            $(event.target).addClass('default-text');
        }
    };

    this.OnDrawButtonClick = function () {
        if (this.map.hasLayer(this.lineLayer))
            this.map.removeLayer(this.lineLayer);

        this.point = [];
        this.lineLayer = new L.FeatureGroup();
        this.map.addLayer(this.lineLayer);

        this.map.on(L.Draw.Event.CREATED, function (e) {
            this.lineLayer.addLayer(e.layer);
            this.point.push([e.layer._latlngs[0].lng, e.layer._latlngs[0].lat], [e.layer._latlngs[1].lng, e.layer._latlngs[1].lat]);
            this.AddPoints(this.lineLayer, this.point, 2, '#000000', '#ff0000');
            this.SetDistance(e.layer._latlngs[0].lng, e.layer._latlngs[0].lat, e.layer._latlngs[1].lng, e.layer._latlngs[1].lat, this.lineLayer);
        }.bind(this));

        this.map.on(L.Draw.Event.DRAWVERTEX, function (e) {
            var layerIds = Object.keys(e.layers._layers);
            if (layerIds.length > 1) {
                var secondVertex = e.layers._layers[layerIds[1]]._icon;
                requestAnimationFrame(() => secondVertex.click());
            }
        }.bind(this));

        var drawer = new L.Draw.Polyline(this.map, {});
        drawer.enable();
    };

    this.OnClearButtonClick = function () {
        if (this.map.hasLayer(this.lineLayer))
            this.map.removeLayer(this.lineLayer);
    };

    this.SetDistance = function (fromLon, fromLat, toLon, toLat, features) {
        var lon = (fromLon + toLon) / 2;
        var lat = (fromLat + toLat) / 2;
        var from = L.latLng(fromLat, fromLon);
        var to = L.latLng(toLat, toLon);
        var bounds = from.distanceTo(to);

        L.marker([lat, lon], {
            icon: L.divIcon({
                className: 'distance',
                html: '<span>两点之间距离为:{0}公里</span>'.format(Math.round((bounds / 1000) * 100) / 100)
            })
        }).addTo(features);
    };

    this.AddPoints = function (features, points, weight, color, fillColor) {
        $(points).each(function (index, point) {
            L.circleMarker([point[1], point[0]], {
                opacity: 1,
                weight: weight,
                color: color,
                fillColor: fillColor,
                fillOpacity: 1,
                radius: 4,
                className: 'point point{0}'.format(point[1].toString().split(".").join(""))
            }).addTo(features);
        }.bind(this));
    };

    this.GetWindFileName = function () {
        var value = $('#release-date').datebox('getValue');
        var time = moment(value, 'YYYY/MM/DD HH:mm');

        var hour = time.format('YYYY-MM-DD_HH');
        var minute = Math.floor(time.minutes() / 10) * 10;
        return '{0}_{1}.json'.format(hour, minute === 0 ? '00' : minute);
    };

    this.LoadWindTemplate = function () {
        $.getJSON('/Content/scripts/menggu/wind-template.json', function (data) {
            this.WindTemplate = data;
        }.bind(this));
    };

    this.RemoveLayer = function (name) {
        this.map.eachLayer(function (layer) {
            if (layer.name === name)
                this.map.removeLayer(layer);
        }.bind(this));
    };

    this.ClearLayers = function () {
        this.map.eachLayer(function (layer) {
            if (layer.name !== undefined)
                this.map.removeLayer(layer);
        }.bind(this));
    };

    this.LoadAverageData = function (name, param, result) {
        if (this.MultiLayers)
            this.RemoveLayer(name);
        else
            this.ClearLayers();

        var layer = this.CreateAverageLayer(name, param, result);
        this.map.addLayer(layer);
        this.BindCloseButtons();
    };

    this.BindCloseButtons = function () {
        $('.close-button').on('click', function (event) {
            $('.{0}'.format($(event.target).next().attr('lng'))).hide();
            $(event.target).parent().hide();
        });
    };

    this.LoadDetailData = function (name, values) {
        this.ClearLayers();
        var layer = this.CreateDetailLayer(name, values);
        this.map.addLayer(layer);
    };

    this.CreateDetailLayer = function (name, values) {
        var features = new L.FeatureGroup();
        features.name = name;

        this.AddData(features, values);
        return features;
    };

    this.CreateAverageLayer = function (name, param, result) {
        var features = new L.FeatureGroup();
        features.name = name;

        this.AddData(features, result.backward.average);
        this.AddData(features, result.forward.average);

        this.AddPolyline(features, result.backward.points);
        this.AddPolyline(features, result.forward.points);

        this.AddCenter(features, param);
        this.AddPoints(features, result.backward.points);
        this.AddPoints(features, result.forward.points);

        this.AddLabels(features, result.backward.points);
        this.AddLabels(features, result.forward.points);
        return features;
    };

    this.AddData = function (features, values) {
        $(values).each(function (index, value) {
            var color = this.GetPointsColor(value[2]);
            var bounds = [[value[1] - 0.01, value[0] - 0.01], [value[1] + 0.01, value[0] + 0.01]];
            L.rectangle(bounds, {
                color: color,
                fillColor: color,
                fillOpacity: 0.75,
                weight: 0
            }).addTo(features);
        }.bind(this));
    };

    this.AddPolyline = function (features, points) {
        var polyline = [];
        $(points).each(function (index, point) {
            polyline.push([point[1], point[0]]);
        });
        L.polyline(polyline, {
            fillColor: '#1c9099',
            color: 'white',
            weight: 2
        }).addTo(features);
    };

    this.AddCenter = function (features, param) {
        var latlng = [param.Latitude, param.Longitude];
        L.circleMarker(latlng, {
            opacity: 1,
            weight: 0.5,
            color: 'black',
            fillColor: 'yellow',
            fillOpacity: 0.5,
            radius: 6
        }).addTo(features);

        L.marker(latlng, {
            icon: L.divIcon({
                className: 'center-label',
                html: this.GetCenterLabel(param)
            })
        }).addTo(features);
    };

    this.AddPoints = function (features, points) {
        $(points).each(function (index, point) {
            L.circleMarker([point[1], point[0]], {
                opacity: 1,
                weight: 0.5,
                color: 'black',
                fillColor: '#fff',
                fillOpacity: 1,
                radius: 4,
                className: 'point point{0}'.format(point[1].toString().split(".").join(""))
            }).addTo(features);
        }.bind(this));
    };

    this.AddLabels = function (features, points) {
        $(points).each(function (index, point) {
            L.marker([point[1], point[0]], {
                icon: L.divIcon({
                    className: 'time-label',
                    html: this.NormalizeTime(point[1], point[2])
                })
            }).addTo(features);
        }.bind(this));
    };

    this.GetCenterLabel = function (param) {
        var pattern = '<p>经度:{0}</p><p>纬度:{1}</p><p>高度:{2}米</p><p>释放时间:{3}</p>';
        return pattern.format(param.Longitude, param.Latitude, param.Height, moment(param.ReleaseTime).format("MM/DD HH:mm"));
    };

    this.NormalizeTime = function (lng, time) {
        var attr = lng.toString().split(".").join("");
        var value = moment(time, "YYYYMMDD_HHmmss");
        return '<a href="javascript:;" class="close-button" title="关闭"></a><span lng="point{0}">{1}</span>'.format(attr, value.format("MM/DD HH:mm"));
    };

    this.ShowWindLayer = function (data) {
        this.WindLayer = L.velocityLayer({
            displayValues: false,
            displayOptions: {
                velocityType: 'GBR Wind',
                displayPosition: 'bottomleft',
                displayEmptyString: 'No wind data'
            },
            data: data,
            maxVelocity: 15
        });
        this.map.addLayer(this.WindLayer);
    };

    this.HideWindLayer = function () {
        if (this.map.hasLayer(this.WindLayer))
            this.map.removeLayer(this.WindLayer);
    };

    this.InitCapture = function () {
        Init();
        $('#capture-button').on('click', function () {
            var returned = Capture();
            if (returned === emCaptureFailed || returned === emCaptureUnknown) {
                $('#capture-button').hide();
                $('#download-button').css("opacity", "1");
            }
        });
    };

    this.GetPointsColor = function (value) {
        if (value <= 0.0001)
            return 'rgb(255,255,255)';
        if (value <= 0.001)
            return 'rgb(178, 226, 249)';
        if (value <= 0.01)
            return 'rgb(93, 160, 213)';
        if (value <= 0.1)
            return 'rgb(73, 170, 106)';
        if (value <= 1)
            return 'rgb(159, 206, 81)';
        if (value <= 10)
            return 'rgb(248, 171, 67)';
        if (value <= 100)
            return 'rgb(239, 94, 41)';
        if (value <= 1000)
            return 'rgb(192, 28, 36)';
    };

    this.LoadInfoPoint = function () {
        $.ajax({
            type: "POST",
            dataType: 'text',
            url: '/Point/Query',
            data: {
                region: 'mg'
            },
            success: function (result) {
                var data = JSON.parse(result);
                data.forEach(function (item, index) {
                    var point = {
                        lat: item.Latitude,
                        lng: item.Longitude
                    };
                    this.AddInfoPoint(point, item.Title, item.Icon, item.Id)
                }.bind(this))
            }.bind(this)
        });
    };

    this.onSelectPointClick = function () {
        this.isSelected = true;

        $('#map').css('cursor', 'crosshair');
        this.SelectPiontInfo();
    };

    this.SelectPiontInfo = function () {
        this.map.on('click', function (e) {
            if (this.isSelected)
                this.SelectPoint.ShowDialog(e.latlng);
            else
                return;
        }.bind(this));
    };

    this.onMarkClick = function () {
        this.isMark = true;

        $('#map').css('cursor', 'crosshair');
        this.DrewInfoPoint();
    };

    this.DrewInfoPoint = function () {
        this.map.on('click', function (e) {
            if (this.isMark)
                this.InfoPoint.ShowDialog(e.latlng);
            else
                return;
        }.bind(this))
    };

    this.AddInfoPoint = function (point, title, icon, id) {
        let label = '<div class="info-point-block info-point{0}"><div class="info-point-top"><p>{1}</p><div class="remove-info-point""><span class="remove-info-point-btn" id="close-btn{2}"></span></div></div><div class="icon icon{3}"/></div></div>'.format(id, title, id, icon.slice(0, 1));
        L.marker([point.lat, point.lng], {
            icon: L.divIcon({
                className: 'info-point-content',
                html: label
            })
        }).addTo(this.map);

        var button = $('#close-btn{0}'.format(id));
        button.on('click', this.RemoveInfoPoint.bind(this, id));
    };

    this.RemoveInfoPoint = function (id, event) {
        $.ajax({
            type: "POST",
            dataType: 'text',
            url: '/Point/Delete',
            data: {
                id: id,
            },
            success: function () {
                $('.info-point{0}'.format(id)).hide();
            }.bind(this)
        });
    };

    this.OnItemClick = function (event) {
        var label = $(event.target).is('img') ? $(event.target).parent('span') : $(event.target);
        var index = parseInt(label.attr('index'));

        if (label.attr('name') === 'cursor-select') {
            this.currentButton.cursorSelected = !this.currentButton.cursorSelected;
            this.ElementActive(this.currentButton.cursorSelected, label, index);
        } else if (label.attr('name') === 'point-mark') {
            this.currentButton.markSelected = !this.currentButton.markSelected;
            this.ElementActive(this.currentButton.markSelected, label, index);
        } else if (label.attr('name') === 'draw-line') {
            this.currentButton.drawLineSelected = !this.currentButton.drawLineSelected;
            this.ElementActive(this.currentButton.drawLineSelected, label, index);
        } else if (label.attr('name') === 'clear') {
            this.currentButton.clearSelected = !this.currentButton.clearSelected;
            this.ElementActive(this.currentButton.clearSelected, label, index);
        } else if (label.attr('name') === 'tag') {
            this.currentButton.tagSelected = !this.currentButton.tagSelected;
            this.ElementActive(this.currentButton.tagSelected, label, index);
        } else if (label.attr('name') === 'switch') {
            this.currentButton.switchSelected = !this.currentButton.switchSelected;
            this.ElementActive(this.currentButton.switchSelected, label, index);
        } else if (label.attr('name') === 'export') {
            this.currentButton.exportSelected = !this.currentButton.exportSelected;
            this.ElementActive(this.currentButton.exportSelected, label, index);
        }
    };

    this.ElementActive = function (selected, label, index) {
        if (selected) {
            label.parent('div').addClass("control-active-item");
        } else {
            $('.control-btn-groups .control-item').eq(index).removeClass("control-active-item");
        }
    };

    this.onSwitchClick = function () {
        this.isSwitch = true;
        this.LatLngSwitch.ShowDialog();
    };
};