"use strict";

const _ = require("lodash");
const isCustomProperty = require("../../utils/isCustomProperty");
const isStandardSyntaxProperty = require("../../utils/isStandardSyntaxProperty");
const optionsMatches = require("../../utils/optionsMatches");
const report = require("../../utils/report");
const ruleMessages = require("../../utils/ruleMessages");
const validateOptions = require("../../utils/validateOptions");

const ruleName = "declaration-block-no-duplicate-properties";

const messages = ruleMessages(ruleName, {
  rejected: property => `Unexpected duplicate "${property}"`
});

const rule = function(on, options) {
  return (root, result) => {
    const validOptions = validateOptions(
      result,
      ruleName,
      { actual: on },
      {
        actual: options,
        possible: {
          ignore: [
            "consecutive-duplicates",
            "consecutive-duplicates-with-different-values"
          ],
          ignoreProperties: [_.isString]
        },
        optional: true
      }
    );
    if (!validOptions) {
      return;
    }

    // In order to accommodate nested blocks (postcss-nested),
    // we need to run a shallow loop (instead of eachDecl() or eachRule(),
    // which loop recursively) and allow each nested block to accumulate
    // its own list of properties -- so that a property in a nested rule
    // does not conflict with the same property in the parent rule
    root.each(node => {
      if (node.type === "rule" || node.type === "atrule") {
        checkRulesInNode(node);
      }
    });

    function checkRulesInNode(node) {
      const decls = [];
      const values = [];

      node.each(child => {
        if (child.nodes && child.nodes.length) {
          checkRulesInNode(child);
        }

        if (child.type !== "decl") {
          return;
        }

        const prop = child.prop;
        const value = child.value;

        if (!isStandardSyntaxProperty(prop)) {
          return;
        }
        if (isCustomProperty(prop)) {
          return;
        }

        // Return early if the property is to be ignored
        if (optionsMatches(options, "ignoreProperties", prop)) {
          return;
        }

        // Ignore the src property as commonly duplicated in at-fontface
        if (prop.toLowerCase() === "src") {
          return;
        }

        const indexDuplicate = decls.indexOf(prop.toLowerCase());

        if (indexDuplicate !== -1) {
          if (
            optionsMatches(
              options,
              "ignore",
              "consecutive-duplicates-with-different-values"
            )
          ) {
            // if duplicates are not consecutive
            if (indexDuplicate !== decls.length - 1) {
              report({
                message: messages.rejected(prop),
                node: child,
                result,
                ruleName
              });
              return;
            }
            // if values of consecutive duplicates are equal
            if (value === values[indexDuplicate]) {
              report({
                message: messages.rejected(value),
                node: child,
                result,
                ruleName
              });
              return;
            }
            return;
          }

          if (
            optionsMatches(options, "ignore", "consecutive-duplicates") &&
            indexDuplicate === decls.length - 1
          ) {
            return;
          }

          report({
            message: messages.rejected(prop),
            node: child,
            result,
            ruleName
          });
        }

        decls.push(prop.toLowerCase());
        values.push(value.toLowerCase());
      });
    }
  };
};

rule.ruleName = ruleName;
rule.messages = messages;
module.exports = rule;