Overview

Namespaces

  • Ecailles
    • DomClassName
      • Bem

Classes

  • Ecailles\DomClassName\Bem\Bem
  • Ecailles\DomClassName\Bem\Contexts
  • Overview
  • Namespace
  • Class
  • Tree
  • Deprecated
  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:  84:  85:  86:  87:  88:  89:  90:  91:  92:  93:  94:  95:  96:  97:  98:  99: 100: 101: 102: 103: 104: 105: 106: 107: 108: 109: 110: 111: 112: 113: 114: 115: 116: 117: 118: 119: 120: 121: 122: 123: 124: 125: 126: 127: 128: 129: 130: 131: 132: 133: 134: 135: 136: 137: 138: 139: 140: 141: 142: 143: 144: 145: 146: 147: 148: 149: 150: 151: 152: 153: 154: 155: 156: 157: 158: 159: 160: 161: 162: 163: 164: 165: 166: 167: 168: 169: 170: 171: 172: 173: 174: 175: 176: 177: 178: 179: 180: 181: 182: 183: 184: 185: 186: 187: 188: 189: 190: 191: 192: 193: 194: 195: 196: 197: 198: 199: 200: 201: 202: 203: 204: 205: 206: 207: 208: 209: 210: 211: 212: 213: 214: 215: 216: 217: 218: 219: 220: 221: 222: 223: 224: 225: 226: 227: 228: 229: 230: 231: 232: 233: 234: 235: 236: 237: 238: 239: 240: 241: 242: 243: 244: 245: 246: 247: 248: 249: 250: 251: 252: 253: 254: 255: 256: 257: 258: 259: 260: 261: 262: 263: 264: 265: 266: 267: 268: 269: 270: 271: 272: 273: 274: 275: 276: 277: 278: 279: 280: 281: 282: 283: 284: 285: 286: 287: 288: 289: 290: 291: 292: 293: 294: 295: 296: 297: 298: 299: 300: 301: 302: 303: 304: 305: 306: 307: 308: 309: 310: 311: 312: 313: 314: 315: 316: 317: 318: 319: 320: 321: 322: 323: 324: 325: 326: 327: 328: 329: 330: 331: 332: 333: 334: 335: 336: 337: 338: 339: 340: 341: 342: 343: 344: 345: 346: 347: 348: 349: 350: 351: 352: 353: 354: 355: 356: 357: 358: 359: 360: 361: 362: 363: 364: 365: 366: 367: 368: 369: 370: 371: 372: 373: 374: 375: 376: 377: 378: 379: 380: 381: 382: 383: 384: 385: 386: 387: 388: 389: 390: 391: 392: 393: 394: 395: 396: 397: 398: 399: 400: 401: 402: 403: 404: 405: 406: 407: 408: 409: 410: 411: 412: 413: 414: 415: 416: 417: 418: 419: 420: 421: 422: 423: 424: 425: 426: 427: 428: 429: 430: 431: 432: 433: 434: 435: 436: 437: 438: 439: 440: 441: 442: 443: 444: 445: 446: 447: 448: 449: 450: 451: 452: 453: 454: 455: 456: 457: 458: 459: 460: 461: 462: 463: 464: 465: 466: 467: 468: 469: 470: 471: 472: 473: 474: 475: 476: 477: 478: 479: 480: 481: 482: 483: 484: 485: 486: 487: 488: 489: 490: 491: 492: 493: 494: 495: 496: 497: 498: 499: 500: 501: 502: 503: 504: 505: 506: 507: 508: 509: 510: 511: 512: 513: 514: 515: 516: 517: 518: 519: 520: 521: 522: 523: 524: 525: 526: 527: 
<?php
/**
 * Bem class name builder.
 *
 * @author Whizark <contact@whizark.com>
 * @see http://whizark.com
 * @copyright Copyright (C) 2015 Whizark.
 * @license MIT
 */
namespace Ecailles\DomClassName\Bem;

/**
 * Class Bem
 *
 * @package Ecailles\DomClassName\Bem
 *
 * @method Bem class() An alias of `classname()`.
 * @method Bem clone() Clones the current instance of Bem.
 */
class Bem
{
    /**
     * @var string The current context.
     */
    protected $context = Contexts::NORMAL;

    /**
     * @var string|null The prefix.
     */
    protected $prefix;

    /**
     * @var string The prefix separator.
     */
    protected $prefixSeparator;

    /**
     * @var string[] The Block names.
     */
    protected $blocks = [];

    /**
     * @var string[] The Element names.
     */
    protected $elements = [];

    /**
     * @var string The Element separator.
     */
    protected $elementSeparator;

    /**
     * @var string[] The contexts and Modifier names.
     */
    protected $modifiers = [];

    /**
     * @var string The Modifier separator.
     */
    protected $modifierSeparator;

    /**
     * @var string The plain class names.
     */
    protected $classes = [];

    /**
     * Bem constructor.
     *
     * @param string $prefix The prefix.
     * @param string $prefixSeparator The prefix separator.
     * @param string $elementSeparator The Element separator.
     * @param string $modifierSeparator The Modifier separator.
     */
    public function __construct(
        $prefix = null,
        $prefixSeparator = null,
        $elementSeparator = null,
        $modifierSeparator = null
    ) {
        $this->prefix            = $prefix;
        $this->prefixSeparator   = ($prefixSeparator !== null) ? $prefixSeparator : '-';
        $this->elementSeparator  = ($elementSeparator !== null) ? $elementSeparator : '__';
        $this->modifierSeparator = ($modifierSeparator !== null) ? $modifierSeparator : '--';
        $this->modifiers         = [
            Contexts::NORMAL  => [],
            Contexts::BLOCK   => [],
            Contexts::ELEMENT => [],
        ];
    }

    /**
     * Returns the string representation of the instance.
     *
     * @return string The string representation.
     */
    public function __toString()
    {
        return $this->value();
    }

    /**
     * The magic method that is triggered when invoking an inaccessible method in an object context.
     *
     * @param string $name The name of the method being called.
     * @param mixed[] $arguments The arguments of the method being called.
     *
     * @return Bem|void The current instance of Bem when the method name is `class`.
     *                  A cloned instance of Bem when the method name is `clone`.
     */
    public function __call($name, $arguments)
    {
        switch ($name) {
            case 'class':
                return call_user_func_array([$this, 'classname'], $arguments);

                break;

            case 'clone':
                return clone $this;

                break;

            default:
                break;
        }
    }

    /**
     * Changes the prefix.
     *
     * @param string $name The prefix to change.
     *
     * @return Bem The current instance of Bem.
     */
    public function prefix($name)
    {
        $this->prefix = $name;

        return $this;
    }

    /**
     * Changes the prefix separator.
     *
     * @param string $separator The prefix separator to change.
     *
     * @return Bem The current instance of Bem.
     */
    public function prefixSeparator($separator)
    {
        $this->prefixSeparator = $separator;

        return $this;
    }

    /**
     * Adds Block name(s).
     *
     * @param string|string[] $names The name(s) to add.
     *
     * @return Bem The current instance of Bem.
     */
    public function block($names)
    {
        $this->context = Contexts::BLOCK;
        $names         = is_array($names) ? $names : [$names];
        $this->blocks  = array_unique(array_merge($this->blocks, $names));

        return $this;
    }

    /**
     * Adds Block Modifier name(s).
     *
     * @param string|string[] $names The Block Modifier name(s) to add.
     *
     * @return Bem The current instance of Bem.
     */
    public function blockModifier($names)
    {
        $names                            = is_array($names) ? $names : [$names];
        $this->modifiers[Contexts::BLOCK] = array_unique(array_merge($this->modifiers[Contexts::BLOCK], $names));

        return $this;
    }

    /**
     * Adds Element name(s).
     *
     * @param string|string[] $names The Element name(s) to add.
     *
     * @return Bem The current instance of Bem.
     */
    public function element($names)
    {
        $this->context  = Contexts::ELEMENT;
        $names          = is_array($names) ? $names : [$names];
        $this->elements = array_unique(array_merge($this->elements, $names));

        return $this;
    }

    /**
     * Changes the Element separator.
     *
     * @param string $separator The Element separator to change.
     *
     * @return Bem The current instance of Bem.
     */
    public function elementSeparator($separator)
    {
        $this->elementSeparator = $separator;

        return $this;
    }

    /**
     * Adds Element Modifier name(s).
     *
     * @param string|string[] $names The Element Modifier name(s) to add.
     *
     * @return Bem The current instance of Bem.
     */
    public function elementModifier($names)
    {
        $names                              = is_array($names) ? $names : [$names];
        $this->modifiers[Contexts::ELEMENT] = array_unique(array_merge($this->modifiers[Contexts::ELEMENT], $names));

        return $this;
    }

    /**
     * Adds contextual Modifier name(s).
     *
     * @param string|string[] $names The contextual Modifier name(s) to add.
     *
     * @return Bem The current instance of Bem.
     */
    public function modifier($names)
    {
        $names                           = is_array($names) ? $names : [$names];
        $this->modifiers[$this->context] = array_unique(array_merge($this->modifiers[$this->context], $names));

        return $this;
    }

    /**
     * Changes the Modifier separator.
     *
     * @param string $separator The Modifier separator to change.
     *
     * @return Bem The current instance of Bem.
     */
    public function modifierSeparator($separator)
    {
        $this->modifierSeparator = $separator;

        return $this;
    }

    /**
     * Adds plain class name(s).
     *
     * @param string|string[] $classes The class name(s) to add.
     *
     * @return Bem The current instance of Bem.
     */
    public function classname($classes)
    {
        $classes       = is_array($classes) ? $classes : [$classes];
        $this->classes = array_unique(array_merge($this->classes, $classes));

        return $this;
    }

    /**
     * Builds and gets the class name(s) as an array.
     *
     * @return string[] An array of the class name(s).
     */
    public function get()
    {
        $this->mergeContextualModifiers();

        $classes = [];
        $sets    = $this->createSets();

        if (count($sets) > 0) {
            $product = $this->createCartesianProduct($sets);
            $classes = array_map(function ($value) {
                return implode('', $value);
            }, $product);
        }

        $classes = array_merge($classes, $this->classes);

        return array_unique($classes);
    }

    /**
     * Builds and gets the class name(s) as a string.
     *
     * @return string The class name(s) string.
     */
    public function value()
    {
        return implode(' ', $this->get());
    }

    /**
     * Returns whether Block name(s) has/have been added.
     *
     * @return bool True if Block name(s) has/have been added, false otherwise.
     */
    protected function hasBlock()
    {
        return (count($this->blocks) > 0);
    }

    /**
     * Gets the Block Modifier name(s) that has/have been added.
     *
     * @return string[] The Block Modifier name(s).
     */
    protected function getBlockModifiers()
    {
        return $this->modifiers[Contexts::BLOCK];
    }

    /**
     * Returns whether Block Modifier name(s) has/have been added.
     *
     * @return bool True if Block Modifier name(s) has/have been added, false otherwise.
     */
    protected function hasBlockModifier()
    {
        return (count($this->getBlockModifiers()) > 0);
    }

    /**
     * Returns whether Element name(s) has/have been added.
     *
     * @return bool True if Element name(s) has/have been added, false otherwise.
     */
    protected function hasElement()
    {
        return (count($this->elements) > 0);
    }

    /**
     * Gets the Block Modifier name(s) that has/have been added.
     *
     * @return string[] The Block Modifier name(s).
     */
    protected function getElementModifiers()
    {
        return $this->modifiers[Contexts::ELEMENT];
    }

    /**
     * Returns whether Element Modifier name(s) has/have been added.
     *
     * @return bool True if Element Modifier name(s) has/have been added, false otherwise.
     */
    protected function hasElementModifier()
    {
        return (count($this->getElementModifiers()) > 0);
    }

    /**
     * Gets the contextual Modifier name(s).
     *
     * @return string[] The contextual Modifier name(s).
     */
    protected function getContextualModifiers()
    {
        return $this->modifiers[Contexts::NORMAL];
    }

    /**
     * Returns whether contextual Modifier name(s) has/have been added.
     *
     * @return bool True if contextual Modifier name(s) has/have been added, false otherwise.
     */
    protected function hasContextualModifier()
    {
        return (count($this->getContextualModifiers()) > 0);
    }

    /**
     * Merges contextual Modifier name(s) with Block/Element Modifier name(s) based on the current context.
     */
    protected function mergeContextualModifiers()
    {
        if ($this->context === Contexts::NORMAL || !$this->hasContextualModifier()) {
            return;
        }

        if ($this->context === Contexts::BLOCK) {
            $this->blockModifier($this->getContextualModifiers());
        } elseif ($this->context === Contexts::ELEMENT) {
            $this->elementModifier($this->getContextualModifiers());
        }

        $this->modifiers[Contexts::NORMAL] = [];
    }

    /**
     * Creates a set of the Block name(s).
     *
     * @return string[] The created set of the Block name(s).
     */
    protected function createBlockSet()
    {
        $prefix = ($this->prefix !== null && $this->prefix !== '') ? ($this->prefix . $this->prefixSeparator) : '';
        $set    = array_map(function ($blockName) use ($prefix) {
            return $prefix . $blockName;
        }, $this->blocks);

        return $set;
    }

    /**
     * Creates a set of the Block Modifier name(s).
     *
     * @return string[] The created set of the Block Modifier name(s).
     */
    protected function createBlockModifierSet()
    {
        $set = array_map(function ($modifiers) {
            return $this->modifierSeparator . $modifiers;
        }, $this->getBlockModifiers());

        return $set;
    }

    /**
     * Creates a set of the Element name(s).
     *
     * @return string[] The created set of the Element name(s).
     */
    protected function createElementSet()
    {
        $prefix = ($this->prefix !== null && $this->prefix !== '') ? ($this->prefix . $this->prefixSeparator) : '';
        $prefix = $this->hasBlock() ? $this->elementSeparator : $prefix;

        $set = array_map(function ($elementName) use ($prefix) {
            return $prefix . $elementName;
        }, $this->elements);

        return $set;
    }

    /**
     * Creates a set of the Element Modifier name(s).
     *
     * @return string[] The created set of the Element Modifier name(s).
     */
    protected function createElementModifierSet()
    {
        $set = array_map(function ($modifiers) {
            return $this->modifierSeparator . $modifiers;
        }, $this->getElementModifiers());

        return $set;
    }

    /**
     * Creates sets of the Block, Element, Modifier name(s) based on the state.
     *
     * @return string[] The sets of the items.
     */
    protected function createSets()
    {
        $sets = [];

        if ($this->hasBlock()) {
            $sets[] = $this->createBlockSet();

            if ($this->hasBlockModifier()) {
                $sets[] = $this->createBlockModifierSet();
            }
        }

        if (!$this->hasElement()) {
            return $sets;
        }

        $sets[] = $this->createElementSet();

        if ($this->hasElementModifier()) {
            $sets[] = $this->createElementModifierSet();
        }

        return $sets;
    }

    /**
     * Creates Cartesian product with given sets of name(s).
     *
     * @param array[] $sets The sets of the name(s).
     *
     * @return array[] The created Cartesian product.
     */
    protected function createCartesianProduct(array $sets = [])
    {
        if (count($sets) === 0) {
            return [[]];
        }

        $root     = array_shift($sets);
        $subPairs = $this->createCartesianProduct($sets);
        $product  = [];

        foreach ($root as $rootValue) {
            foreach ($subPairs as $subPairValue) {
                array_unshift($subPairValue, $rootValue);

                $product[] = $subPairValue;
            }
        }

        return $product;
    }
}
DomClassName API Documentation API documentation generated by ApiGen