Image.js 6.84 KB
import template from "./Image.html";
import style from "antd/lib/image/style/index.css";
angular.module("esNgAntd").directive("antdImage", ["$timeout", "esNgAntd", "$compile", function ($timeout, esNgAntd, $compile) {
  return {
    template: template,
    restrict: "E",
    replace: true,
    transclude: true,
    scope: {
      src: "@",
      dSrc: "@",
      fallback: "@",
      width: "=",
      height: "=",
      preview: "="
    },
    require: ["^?antdImagePreviewGroup"],
    controller: function ($scope, $element, $attrs) {
      $scope.watch = {
        dSrc: function (newVal) {
          if (!["undefined", undefined, null, ""].includes(newVal)) {
            $scope.state.hasImage = true;
            $scope.state.src = newVal;
          }
        }
      };
      $scope.state = {
        src: $scope.fallback || "",
        // 可见性
        visible: false,
        // 有图片?
        hasImage: false,
        // 索引
        index: null
      };

      $scope.handleOpen = function () {
        if ($scope.antdImagePreviewGroup) {
          $scope.antdImagePreviewGroup.handleOpen($scope.state.index);
          return;
        }

        $scope.state.visible = true;
      };

      $scope.handleClose = function () {
        $scope.state.visible = false;
      };

      $scope.handlePreview = function () {
        let div = document.createElement("div");
        div.innerHTML = `
            <div class="ant-image-preview-root">
                <div class="ant-image-preview-mask" ng-if="state.visible"></div>
                <div class="ant-image-preview-wrap" ng-show="state.visible">
                    <div class="ant-image-preview">
                        <div class="ant-image-preview-content">
                            <div class="ant-image-preview-body">
                                <ul class="ant-image-preview-operations">
                                    <li class="ant-image-preview-operations-operation">
                                        <span class="anticon anticon-close ant-image-preview-operations-icon" ng-click="handleClose()">
                                            <antd-icon type="CloseOutlined"></antd-icon>
                                        </span>
                                    </li> 
                                </ul>
                                <div class="ant-image-preview-img-wrapper">
                                    <img class="ant-image-preview-img" ng-src="{{state.src}}"/>
                                </div>
                            </div>
                        </div>
                    </div>
                </div>
            </div>
        `;
        document.body.appendChild(div);
        $compile(div)($scope);
      };
    },
    link: function ($scope, $element, $attrs, $controllers) {
      for (const key in $scope.watch) {
        $scope.$watch(key, $scope.watch[key], true);
      }

      [$element, $attrs, $controllers].forEach(function (value, key) {
        if ([undefined, null, ""].includes(value)) {
          throw new Error(`${["$element", "$attrs", "$controllers"][key]} parameter of constructor method is required.`);
        }
      });
      esNgAntd.createStyle("ant-image", style);
      let src = $scope.state.src;
      let [antdImagePreviewGroup] = $controllers;

      $element.children("img")[0].onerror = function () {
        $timeout(function () {
          $scope.state.hasImage = false;
          $scope.state.src = src;
        }, 0);
      };

      if (antdImagePreviewGroup) {
        $scope.antdImagePreviewGroup = antdImagePreviewGroup.getContext();
        $scope.state.index = $scope.antdImagePreviewGroup.addChildren($scope);
      } else {
        $scope.handlePreview();
      }
    }
  };
}]);