ancor / yii2-bitmask
Bit mask behavior and validators for models
Installs: 8 416
Dependents: 0
Suggesters: 0
Security: 0
Stars: 5
Watchers: 2
Forks: 0
Type:yii2-extension
Requires
- yiisoft/yii2: ^2.0
This package is not auto-updated.
Last update: 2025-01-04 20:31:15 UTC
README
Описание
Расширение предназначено для работы с битовыми масками.
На пример, вы хотите хранить значения нескольких переключателей в поле типа int.
Возникает ряд вопросов:
- Как в соответствии со сценариями разрешить присваивать\снимать только некоторые биты, а остальные запретить
- Как привязать input[type=checkbox] теги к соответствующим битам
- Где хранить подписи для полей ввода
- Как выводить отмеченные биты
- Как удобно проверить наличие бита в поле у модели
- Ну и наконец как удобно ставить\снимать бит
yii2-bitmask дает решает все эти вопросы
Feel free to let me know what else you want added via:
Installation
The preferred way to install this extension is through composer.
Either run
$ php composer.phar require ancor/yii2-bitmask
or add
"ancor/yii2-bitmask": "dev-master"
to the require
section of your composer.json
file.
Расширение содержит три класса
- BitmaskBehavior - создание полей модели на основе битовой маски
- BitmaskValidator - валидация основанная на битах
- BitmaskFieldsValidator - валидация основанная на названиях полей
BitmaskBehavior
Настройка модели
use ancor\bitmask\BitmaskBehavior; /** * @property integer $options * ... * @property string $spamOption * @property string $deletedOption * ... */ class User extends \yii\db\ActiveRecord { const OPT_SPAM = 1<<0; const OPT_DELETED = 1<<1; public function behaviors() { return [ 'bitmask' => [ 'options' => [ 'spamOption' => static::OPT_SPAM, 'deletedOption' => static::OPT_DELETED, ], // 'bitmaskAttribute' => 'options', // an attribute which is the mask itself ], ]; } public function rules() { return [ [['spamOption', 'deletedOption'], 'safe'], ]; } public function attributeLabels() { return [ ... 'spamOption' => 'This user is spammer', 'emailNotVerified' => 'User is deleted', ... ]; } }
Пример использования
$model = new User(); echo $model->options; // 0 // Назначить бит $model->spamOption = true; // $model->options == User::OPT_SPAM == 1<<0 == 1 // Это эквивалентно следующей строке $model->options = $model->options | User::OPT_SPAM; // Внимание! При присваивании значения - любое значение принудительно преобразуется в boolean // Снять бит $model->spamOption = false; // $model->options == 0 // Это эквивалентно следующей строке $model->options = $model->options & ~User::OPT_SPAM; // Проверить наличие бита if ($model->spamOption) ... // Это эквивалентно следующей строке if ($model->options & User::OPT_SPAM) ...
Групповое присвоение
// Предположим пришла форма $post = [ 'User' => [ ... 'spamOption' => false, 'deletedOption' => true, ... ], ]; /** * Загрузка одной коммандой */ echo $model->options; // 0 $model->load($post); echo $model->options; // $model->options == 1<<1 == 2 var_dump($model->spamOption); // false var_dump($model->deletedOption); // true /** * Пример БЕЗ BitmaskBehavior, при отправке той же самой формы * Предположим что в модели объявлены 2 свойства * public $spamOption; * public $deletedOption; */ echo $model->options; // 0 $model->load($post); // Свойство options, конечно же осталось без изменений echo $model->options; // 0 if ($post['User']['spamOption']) { $model->options |= User::OPT_SPAM; } else { $model->options &= ~User::OPT_SPAM; } if ($post['User']['deletedOption']) { $model->options |= User::OPT_DELETED; } else { $model->options &= ~User::OPT_DELETED; }
Пример шаблона с использованием ActiveForm
$form->field($model, 'spamOption')->checkbox(); $form->field($model, 'deletedOption')->checkbox();
Примечание: Если вы хотите запретить менять некоторые биты - то просто не нужно перечислять их в валидаторе safe
Новые методы и свойства модели
php public integer[] getBitmaskFields(void)
Метод возвращает массив с наименованиями полей и соответствующими им битами. Без значений по умолчанию
Пример ответа(в нашем случае):
[ 'spamOption' => 1, // 1<<0 'deletedOption' => 2, // 1<<1 ]
php public boolean[] getBitmaskValues(void)
Метов возвращает наименования полей с их значениями true\false.
Пример ответа(в нашем случае):
[ 'spamOption' => false, 'deletedOption' => true, ]
php public boolean getOldBit(string $field)
Возвращает значение "старого" атрибута. (Работает через вызов ->getOldAttribute()
)
Собственные статические методы поведения
Добавить бит в маску\Убрать биз из маски
public static integer modifyBitmask(int $mask, int $bit, boolean $exists)
$mask = 0b00100001; $bit = 0b00000100; // Добавить бит $options = BitmaskBehavior::modifyBitmask($mask, $bit, true); // 0b00100101 // Убрать бит $options = BitmaskBehavior::modifyBitmask($mask, $bit, false); // 0b00100001
Получить массив битов на основе битовой маски
public static boolean[] parseBitmask(int $mask, int[] $fields)
$mask = 0b00100000; $fields = [ 'firstOption' => 0b00000001, 'secondOption' => 0b00100000, ]; $values = BitmaskBehavior::parseBitmask($mask, $fields); print_r($values); // ['firstOption' => false, 'secondOption' => true]
Создать битовую маску на основе массива битов
php public static int makeBitmask(boolean[] $values, int[] $fields)
$fields = [ 'firstOption' => 0b00000001, 'secondOption' => 0b00100000, ]; $values = [ 'firstOption' => false, 'secondOption' => true ]; $mask = BitmaskBehavior::makeBitmask($values, $fields); echo $mask; // 0b00100000
BitmaskValidator
Описание: Позволяет указать какие биты можно менять в маске, а какие нельзя.
Примечание: Этот валидатор можно использовать без BitmaskBehavior.
public function rules() { return [ ['options', BitmaskValidator::className(), 'mask' => 1<<3 | 1<<4 | 1<<6], // Или с переводом сообщения ['options', BitmaskValidator::className(), 'mask' => 1<<3 | 1<<4 | 1<<6, 'message' => ...], // Можно так же использовать константы ['options', BitmaskValidator::className(), 'mask' => static::OPT_SPAM], ]; }
BitmaskFieldsValidator
Описание:
- Этот валидатор поход на
BitmaskValidator
. - Предназначен для использования в паре с
BitmaskBehavior
. - Используется вместо
safe
валидатора. Но это не все. - Валидатор точно так же как
BitmaskValidator
блокирует изменение всех битов к кроме тех к которым валидатор применен.
Пример использования
return [ [ ['spamOption', 'deletedOption'], BitmaskFieldsValidator::className(), // 'maskAttribute' => 'options', // По умолчанию ], ];