mirror of
https://github.com/Part-DB/Part-DB-server.git
synced 2025-06-22 18:03:37 +02:00
Allow to change the permissions for users.
This commit is contained in:
parent
8e61b06abc
commit
7390f2eccd
18 changed files with 933 additions and 8 deletions
|
@ -55,7 +55,10 @@ window.bootbox = require('bootbox')
|
||||||
|
|
||||||
// Includes required for tag input
|
// Includes required for tag input
|
||||||
require('./tagsinput.js');
|
require('./tagsinput.js');
|
||||||
require('../css/tagsinput.css')
|
require('../css/tagsinput.css');
|
||||||
|
|
||||||
|
//Tristate checkbox support
|
||||||
|
require('./jquery.tristate.js');
|
||||||
|
|
||||||
require('../ts_src/ajax_ui');
|
require('../ts_src/ajax_ui');
|
||||||
import {ajaxUI} from "../ts_src/ajax_ui";
|
import {ajaxUI} from "../ts_src/ajax_ui";
|
||||||
|
|
213
assets/js/jquery.tristate.js
Normal file
213
assets/js/jquery.tristate.js
Normal file
|
@ -0,0 +1,213 @@
|
||||||
|
/*jslint devel: true, bitwise: true, regexp: true, browser: true, confusion: true, unparam: true, eqeq: true, white: true, nomen: true, plusplus: true, maxerr: 50, indent: 4 */
|
||||||
|
/*globals jQuery */
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Tristate v1.2.1
|
||||||
|
*
|
||||||
|
* Copyright (c) 2013-2017 Martijn W. van der Lee
|
||||||
|
* Licensed under the MIT.
|
||||||
|
*/
|
||||||
|
/* Based on work by:
|
||||||
|
* Chris Coyier (http://css-tricks.com/indeterminate-checkboxes/)
|
||||||
|
*
|
||||||
|
* Tristate checkbox with support features
|
||||||
|
* pseudo selectors
|
||||||
|
* val() overwrite
|
||||||
|
*/
|
||||||
|
|
||||||
|
;(function($, undefined) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
var pluginName = 'tristate',
|
||||||
|
defaults = {
|
||||||
|
'change': undefined,
|
||||||
|
'checked': undefined,
|
||||||
|
'indeterminate': undefined,
|
||||||
|
'init': undefined,
|
||||||
|
'reverse': false,
|
||||||
|
'state': undefined,
|
||||||
|
'unchecked': undefined,
|
||||||
|
'value': undefined // one-way only!
|
||||||
|
},
|
||||||
|
valFunction = $.fn.val;
|
||||||
|
|
||||||
|
function Plugin(element, options) {
|
||||||
|
if($(element).is(':checkbox')) {
|
||||||
|
this.element = $(element);
|
||||||
|
this.settings = $.extend( {}, defaults, options );
|
||||||
|
this._create();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$.extend(Plugin.prototype, {
|
||||||
|
_create: function() {
|
||||||
|
var that = this,
|
||||||
|
state;
|
||||||
|
|
||||||
|
// Fix for #1
|
||||||
|
if (window.navigator.userAgent.indexOf('Trident') >= 0) {
|
||||||
|
this.element.click(function(e) {
|
||||||
|
that._change.call(that, e);
|
||||||
|
that.element.closest('form').change();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.element.change(function(e) {
|
||||||
|
that._change.call(that, e);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.settings.checked = this.element.attr('checkedvalue') || this.settings.checked;
|
||||||
|
this.settings.unchecked = this.element.attr('uncheckedvalue') || this.settings.unchecked;
|
||||||
|
this.settings.indeterminate = this.element.attr('indeterminatevalue') || this.settings.indeterminate;
|
||||||
|
|
||||||
|
// Initially, set state based on option state or attributes
|
||||||
|
if (typeof this.settings.state === 'undefined') {
|
||||||
|
this.settings.state = typeof this.element.attr('indeterminate') !== 'undefined'? null : this.element.is(':checked');
|
||||||
|
}
|
||||||
|
|
||||||
|
// If value specified, overwrite with value
|
||||||
|
if (typeof this.settings.value !== 'undefined') {
|
||||||
|
state = this._parseValue(this.settings.value);
|
||||||
|
if (typeof state !== 'undefined') {
|
||||||
|
this.settings.state = state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this._refresh(this.settings.init);
|
||||||
|
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
|
||||||
|
_change: function(e) {
|
||||||
|
if (e.isTrigger || !e.hasOwnProperty('which')) {
|
||||||
|
e.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (this.settings.state) {
|
||||||
|
case true: this.settings.state = (this.settings.reverse ? false : null); break;
|
||||||
|
case false: this.settings.state = (this.settings.reverse ? null : true); break;
|
||||||
|
default: this.settings.state = (this.settings.reverse ? true : false); break;
|
||||||
|
}
|
||||||
|
|
||||||
|
this._refresh(this.settings.change);
|
||||||
|
},
|
||||||
|
|
||||||
|
_refresh: function(callback) {
|
||||||
|
var value = this.value();
|
||||||
|
|
||||||
|
this.element.data("vanderlee." + pluginName, value);
|
||||||
|
|
||||||
|
this.element[this.settings.state === null ? 'attr' : 'removeAttr']('indeterminate', 'indeterminate');
|
||||||
|
this.element.prop('indeterminate', this.settings.state === null);
|
||||||
|
this.element.get(0).indeterminate = this.settings.state === null;
|
||||||
|
|
||||||
|
this.element[this.settings.state === true ? 'attr' : 'removeAttr']('checked', true);
|
||||||
|
this.element.prop('checked', this.settings.state === true);
|
||||||
|
|
||||||
|
if ($.isFunction(callback)) {
|
||||||
|
callback.call(this.element, this.settings.state, this.value());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
state: function(value) {
|
||||||
|
if (typeof value === 'undefined') {
|
||||||
|
return this.settings.state;
|
||||||
|
} else if (value === true || value === false || value === null) {
|
||||||
|
this.settings.state = value;
|
||||||
|
|
||||||
|
this._refresh(this.settings.change);
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
|
||||||
|
_parseValue: function(value) {
|
||||||
|
if (value === this.settings.checked) {
|
||||||
|
return true;
|
||||||
|
} else if (value === this.settings.unchecked) {
|
||||||
|
return false;
|
||||||
|
} else if (value === this.settings.indeterminate) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
value: function(value) {
|
||||||
|
if (typeof value === 'undefined') {
|
||||||
|
var value;
|
||||||
|
switch (this.settings.state) {
|
||||||
|
case true:
|
||||||
|
value = this.settings.checked;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case false:
|
||||||
|
value = this.settings.unchecked;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case null:
|
||||||
|
value = this.settings.indeterminate;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return typeof value === 'undefined'? this.element.attr('value') : value;
|
||||||
|
} else {
|
||||||
|
var state = this._parseValue(value);
|
||||||
|
if (typeof state !== 'undefined') {
|
||||||
|
this.settings.state = state;
|
||||||
|
this._refresh(this.settings.change);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$.fn[pluginName] = function (options, value) {
|
||||||
|
var result = this;
|
||||||
|
|
||||||
|
this.each(function() {
|
||||||
|
if (!$.data(this, "plugin.vanderlee." + pluginName)) {
|
||||||
|
$.data(this, "plugin.vanderlee." + pluginName, new Plugin(this, options));
|
||||||
|
} else if (typeof options === 'string') {
|
||||||
|
if (typeof value === 'undefined') {
|
||||||
|
result = $(this).data("plugin.vanderlee." + pluginName)[options]();
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
$(this).data("plugin.vanderlee." + pluginName)[options](value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Overwrite fn.val
|
||||||
|
$.fn.val = function(value) {
|
||||||
|
var data = this.data("vanderlee." + pluginName);
|
||||||
|
if (typeof data === 'undefined') {
|
||||||
|
if (typeof value === 'undefined') {
|
||||||
|
return valFunction.call(this);
|
||||||
|
} else {
|
||||||
|
return valFunction.call(this, value);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (typeof value === 'undefined') {
|
||||||
|
return data;
|
||||||
|
} else {
|
||||||
|
this.data("vanderlee." + pluginName, value);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// :indeterminate pseudo selector
|
||||||
|
$.expr.filters.indeterminate = function(element) {
|
||||||
|
var $element = $(element);
|
||||||
|
return typeof $element.data("vanderlee." + pluginName) !== 'undefined' && $element.prop('indeterminate');
|
||||||
|
};
|
||||||
|
|
||||||
|
// :determinate pseudo selector
|
||||||
|
$.expr.filters.determinate = function(element) {
|
||||||
|
return !($.expr.filters.indeterminate(element));
|
||||||
|
};
|
||||||
|
|
||||||
|
// :tristate selector
|
||||||
|
$.expr.filters.tristate = function(element) {
|
||||||
|
return typeof $(element).data("vanderlee." + pluginName) !== 'undefined';
|
||||||
|
};
|
||||||
|
})(jQuery);
|
|
@ -260,7 +260,7 @@ class AjaxUI {
|
||||||
{
|
{
|
||||||
return {
|
return {
|
||||||
success: this.onAjaxComplete,
|
success: this.onAjaxComplete,
|
||||||
beforeSerialize: function() : boolean {
|
beforeSerialize: function($form, options) : boolean {
|
||||||
|
|
||||||
//Update the content of textarea fields using CKEDITOR before submitting.
|
//Update the content of textarea fields using CKEDITOR before submitting.
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
|
@ -272,6 +272,9 @@ class AjaxUI {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Check every checkbox field, so that it will be submitted (only valid fields are submitted)
|
||||||
|
$form.find("input[type=checkbox].tristate").prop('checked', true);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
beforeSubmit: function (arr, $form, options) : boolean {
|
beforeSubmit: function (arr, $form, options) : boolean {
|
||||||
|
|
|
@ -212,6 +212,15 @@ $(document).on("ajaxUI:start ajaxUI:reload", function() {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
$(document).on("ajaxUI:start ajaxUI:reload", function() {
|
||||||
|
//@ts-ignore
|
||||||
|
$(".tristate").tristate( {
|
||||||
|
checked: "true",
|
||||||
|
unchecked: "false",
|
||||||
|
indeterminate: "indeterminate",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
//Re initialize fileinputs on reload
|
//Re initialize fileinputs on reload
|
||||||
$(document).on("ajaxUI:reload", function () {
|
$(document).on("ajaxUI:reload", function () {
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
|
|
|
@ -2,7 +2,7 @@ twig:
|
||||||
default_path: '%kernel.project_dir%/templates'
|
default_path: '%kernel.project_dir%/templates'
|
||||||
debug: '%kernel.debug%'
|
debug: '%kernel.debug%'
|
||||||
strict_variables: '%kernel.debug%'
|
strict_variables: '%kernel.debug%'
|
||||||
form_themes: ['bootstrap_4_horizontal_layout.html.twig', 'Form/extendedBootstrap4_layout.html.twig' ]
|
form_themes: ['bootstrap_4_horizontal_layout.html.twig', 'Form/extendedBootstrap4_layout.html.twig', 'Form/permissionLayout.html.twig' ]
|
||||||
|
|
||||||
globals:
|
globals:
|
||||||
partdb_title: '%partdb_title%'
|
partdb_title: '%partdb_title%'
|
||||||
|
|
|
@ -1,23 +1,37 @@
|
||||||
# In this file the possible permissions are defined.
|
# In this file the possible permissions are defined.
|
||||||
# This should be compatible with the legacy Part-DB
|
# This should be compatible with the legacy Part-DB
|
||||||
|
|
||||||
|
groups:
|
||||||
|
parts:
|
||||||
|
label: "perm.group.parts"
|
||||||
|
structures:
|
||||||
|
label: "perm.group.structures"
|
||||||
|
system:
|
||||||
|
label: "perm.group.system"
|
||||||
|
|
||||||
|
|
||||||
perms: # Here comes a list with all Permission names (they have a perm_[name] coloumn in DB)
|
perms: # Here comes a list with all Permission names (they have a perm_[name] coloumn in DB)
|
||||||
|
|
||||||
# Part related permissions
|
# Part related permissions
|
||||||
|
|
||||||
parts: # e.g. this maps to perms_parts in User/Group database
|
parts: # e.g. this maps to perms_parts in User/Group database
|
||||||
# label: "perm.parts"
|
group: "parts"
|
||||||
|
label: "perm.parts"
|
||||||
operations: # Here are all possible operations are listed => the op name is mapped to bit value
|
operations: # Here are all possible operations are listed => the op name is mapped to bit value
|
||||||
read:
|
read:
|
||||||
|
label: "perm.read"
|
||||||
bit: 0
|
bit: 0
|
||||||
edit:
|
edit:
|
||||||
# label: "perm.part.edit"
|
label: "perm.edit"
|
||||||
bit: 2
|
bit: 2
|
||||||
create:
|
create:
|
||||||
|
label: "perm.create"
|
||||||
bit: 4
|
bit: 4
|
||||||
move:
|
move:
|
||||||
|
label: "perm.part.move"
|
||||||
bit: 6
|
bit: 6
|
||||||
delete:
|
delete:
|
||||||
|
label: "perm.delete"
|
||||||
bit: 8
|
bit: 8
|
||||||
search:
|
search:
|
||||||
bit: 10
|
bit: 10
|
||||||
|
@ -41,10 +55,13 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 30
|
bit: 30
|
||||||
|
|
||||||
parts_name: &PART_ATTRIBUTE # We define a template here, that we can use for all part attributes.
|
parts_name: &PART_ATTRIBUTE # We define a template here, that we can use for all part attributes.
|
||||||
|
group: "parts"
|
||||||
operations:
|
operations:
|
||||||
read:
|
read:
|
||||||
|
label: "perm.read"
|
||||||
bit: 0
|
bit: 0
|
||||||
edit:
|
edit:
|
||||||
|
label: "perm.edit"
|
||||||
bit: 2
|
bit: 2
|
||||||
|
|
||||||
parts_description:
|
parts_description:
|
||||||
|
@ -81,6 +98,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
*PART_ATTRIBUTE
|
*PART_ATTRIBUTE
|
||||||
|
|
||||||
storelocations: &PART_CONTAINING
|
storelocations: &PART_CONTAINING
|
||||||
|
group: "structures"
|
||||||
operations:
|
operations:
|
||||||
read:
|
read:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
@ -131,6 +149,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 10
|
bit: 10
|
||||||
|
|
||||||
groups:
|
groups:
|
||||||
|
group: "system"
|
||||||
operations:
|
operations:
|
||||||
read:
|
read:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
@ -146,6 +165,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 10
|
bit: 10
|
||||||
|
|
||||||
users:
|
users:
|
||||||
|
group: "system"
|
||||||
operations:
|
operations:
|
||||||
read:
|
read:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
@ -167,6 +187,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 16
|
bit: 16
|
||||||
|
|
||||||
database:
|
database:
|
||||||
|
group: "system"
|
||||||
operations:
|
operations:
|
||||||
see_status:
|
see_status:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
@ -178,6 +199,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 2
|
bit: 2
|
||||||
|
|
||||||
config:
|
config:
|
||||||
|
group: "system"
|
||||||
operations:
|
operations:
|
||||||
read_config:
|
read_config:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
@ -187,6 +209,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 6
|
bit: 6
|
||||||
|
|
||||||
system:
|
system:
|
||||||
|
group: "system"
|
||||||
operations:
|
operations:
|
||||||
use_debug:
|
use_debug:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
@ -196,6 +219,7 @@ perms: # Here comes a list with all Permission names (they have a perm_[name] co
|
||||||
bit: 4
|
bit: 4
|
||||||
|
|
||||||
devices_parts:
|
devices_parts:
|
||||||
|
group: "parts"
|
||||||
operations:
|
operations:
|
||||||
read:
|
read:
|
||||||
bit: 0
|
bit: 0
|
||||||
|
|
|
@ -44,11 +44,18 @@ class PermissionsConfiguration implements ConfigurationInterface
|
||||||
$treeBuilder = new TreeBuilder('permissions');
|
$treeBuilder = new TreeBuilder('permissions');
|
||||||
$rootNode = $treeBuilder->root('permissions');
|
$rootNode = $treeBuilder->root('permissions');
|
||||||
|
|
||||||
|
$rootNode->children()
|
||||||
|
->arrayNode('groups')
|
||||||
|
->arrayPrototype()
|
||||||
|
->children()
|
||||||
|
->scalarNode('label')->end();
|
||||||
|
|
||||||
$rootNode->children()
|
$rootNode->children()
|
||||||
->arrayNode('perms')
|
->arrayNode('perms')
|
||||||
->arrayPrototype()
|
->arrayPrototype()
|
||||||
->children()
|
->children()
|
||||||
->scalarNode('label')->end()
|
->scalarNode('label')->end()
|
||||||
|
->scalarNode('group')->end()
|
||||||
->arrayNode('operations')
|
->arrayNode('operations')
|
||||||
->arrayPrototype()
|
->arrayPrototype()
|
||||||
->children()
|
->children()
|
||||||
|
|
|
@ -64,6 +64,12 @@ class Group extends StructuralDBElement implements HasPermissionsInterface
|
||||||
*/
|
*/
|
||||||
protected $permissions;
|
protected $permissions;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
$this->permissions = new PermissionsEmbed();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the ID as an string, defined by the element class.
|
* Returns the ID as an string, defined by the element class.
|
||||||
* This should have a form like P000014, for a part with ID 14.
|
* This should have a form like P000014, for a part with ID 14.
|
||||||
|
|
|
@ -186,6 +186,11 @@ class User extends NamedDBElement implements UserInterface, HasPermissionsInterf
|
||||||
*/
|
*/
|
||||||
protected $instock_comment_a;
|
protected $instock_comment_a;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
$this->permissions = new PermissionsEmbed();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the current user, is the user which represents the not logged in (anonymous) users.
|
* Checks if the current user, is the user which represents the not logged in (anonymous) users.
|
||||||
*
|
*
|
||||||
|
|
95
src/Form/Permissions/PermissionGroupType.php
Normal file
95
src/Form/Permissions/PermissionGroupType.php
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* part-db version 0.1
|
||||||
|
* Copyright (C) 2005 Christoph Lechner
|
||||||
|
* http://www.cl-projects.de/
|
||||||
|
*
|
||||||
|
* part-db version 0.2+
|
||||||
|
* Copyright (C) 2009 K. Jacobs and others (see authors.php)
|
||||||
|
* http://code.google.com/p/part-db/
|
||||||
|
*
|
||||||
|
* Part-DB Version 0.4+
|
||||||
|
* Copyright (C) 2016 - 2019 Jan Böhmer
|
||||||
|
* https://github.com/jbtronics
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace App\Form\Permissions;
|
||||||
|
|
||||||
|
|
||||||
|
use App\Services\PermissionResolver;
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\OptionsResolver\Options;
|
||||||
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
|
|
||||||
|
class PermissionGroupType extends AbstractType
|
||||||
|
{
|
||||||
|
protected $resolver;
|
||||||
|
protected $perm_structure;
|
||||||
|
|
||||||
|
public function __construct(PermissionResolver $resolver)
|
||||||
|
{
|
||||||
|
$this->resolver = $resolver;
|
||||||
|
$this->perm_structure = $resolver->getPermissionStructure();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options)
|
||||||
|
{
|
||||||
|
$permissions = $this->perm_structure['perms'];
|
||||||
|
|
||||||
|
foreach ($permissions as $key => $permission) {
|
||||||
|
//Check if the permission belongs to our group
|
||||||
|
if (isset($permission['group'])) {
|
||||||
|
if ($permission['group'] !== $options['group_name']) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
//Skip perrmissions without groups unless we have this as blanko group
|
||||||
|
if ($options['group_name'] !== "*") {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$builder->add($key, PermissionType::class, [
|
||||||
|
'perm_name' => $key,
|
||||||
|
'label' => $permission['label'] ?? $key,
|
||||||
|
'mapped' => false,
|
||||||
|
'data' => $builder->getData(),
|
||||||
|
'disabled' => $options['disabled']
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function configureOptions(OptionsResolver $resolver)
|
||||||
|
{
|
||||||
|
parent::configureOptions($resolver);
|
||||||
|
|
||||||
|
$resolver->setDefault('group_name', function (Options $options) {
|
||||||
|
return trim($options['name']);
|
||||||
|
});
|
||||||
|
|
||||||
|
$resolver->setDefault('label', function (Options $options) {
|
||||||
|
if (!empty($this->perm_structure['groups'][$options['group_name']]['label'])) {
|
||||||
|
return $this->perm_structure['groups'][$options['group_name']]['label'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $options['name'];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
157
src/Form/Permissions/PermissionType.php
Normal file
157
src/Form/Permissions/PermissionType.php
Normal file
|
@ -0,0 +1,157 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* part-db version 0.1
|
||||||
|
* Copyright (C) 2005 Christoph Lechner
|
||||||
|
* http://www.cl-projects.de/
|
||||||
|
*
|
||||||
|
* part-db version 0.2+
|
||||||
|
* Copyright (C) 2009 K. Jacobs and others (see authors.php)
|
||||||
|
* http://code.google.com/p/part-db/
|
||||||
|
*
|
||||||
|
* Part-DB Version 0.4+
|
||||||
|
* Copyright (C) 2016 - 2019 Jan Böhmer
|
||||||
|
* https://github.com/jbtronics
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace App\Form\Permissions;
|
||||||
|
|
||||||
|
|
||||||
|
use App\Form\Type\TriStateCheckboxType;
|
||||||
|
use App\Services\PermissionResolver;
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\DataMapperInterface;
|
||||||
|
use Symfony\Component\Form\Exception;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\CheckboxType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\Form\FormInterface;
|
||||||
|
use Symfony\Component\OptionsResolver\Options;
|
||||||
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
|
|
||||||
|
class PermissionType extends AbstractType implements DataMapperInterface
|
||||||
|
{
|
||||||
|
protected $resolver;
|
||||||
|
protected $perm_structure;
|
||||||
|
|
||||||
|
public function __construct(PermissionResolver $resolver)
|
||||||
|
{
|
||||||
|
$this->resolver = $resolver;
|
||||||
|
$this->perm_structure = $resolver->getPermissionStructure();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function configureOptions(OptionsResolver $resolver)
|
||||||
|
{
|
||||||
|
parent::configureOptions($resolver);
|
||||||
|
|
||||||
|
$resolver->setDefault('perm_name', function (Options $options) {
|
||||||
|
return $options['name'];
|
||||||
|
});
|
||||||
|
|
||||||
|
$resolver->setDefault('label', function (Options $options) {
|
||||||
|
if (!empty($this->perm_structure['perms'][$options['perm_name']]['label'])) {
|
||||||
|
return $this->perm_structure['perms'][$options['perm_name']]['label'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $options['name'];
|
||||||
|
});
|
||||||
|
|
||||||
|
$resolver->setDefaults([
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options)
|
||||||
|
{
|
||||||
|
$operations = $this->perm_structure['perms'][$options['perm_name']]['operations'];
|
||||||
|
|
||||||
|
foreach ($operations as $key => $operation) {
|
||||||
|
$builder->add($key, TriStateCheckboxType::class, [
|
||||||
|
'required' => false,
|
||||||
|
'mapped' => false,
|
||||||
|
'label' => $operation['label'] ?? null,
|
||||||
|
'disabled' => $options['disabled']
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$builder->setDataMapper($this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the view data of a compound form to its children.
|
||||||
|
*
|
||||||
|
* The method is responsible for calling {@link FormInterface::setData()}
|
||||||
|
* on the children of compound forms, defining their underlying model data.
|
||||||
|
*
|
||||||
|
* @param mixed $viewData View data of the compound form being initialized
|
||||||
|
* @param FormInterface[]|\Traversable $forms A list of {@link FormInterface} instances
|
||||||
|
*
|
||||||
|
* @throws Exception\UnexpectedTypeException if the type of the data parameter is not supported
|
||||||
|
*/
|
||||||
|
public function mapDataToForms($viewData, $forms)
|
||||||
|
{
|
||||||
|
foreach ($forms as $form) {
|
||||||
|
$value = $this->resolver->dontInherit(
|
||||||
|
$viewData,
|
||||||
|
$form->getParent()->getConfig()->getOption('perm_name'),
|
||||||
|
$form->getName()
|
||||||
|
);
|
||||||
|
$form->setData($value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the model data of a list of children forms into the view data of their parent.
|
||||||
|
*
|
||||||
|
* This is the internal cascade call of FormInterface::submit for compound forms, since they
|
||||||
|
* cannot be bound to any input nor the request as scalar, but their children may:
|
||||||
|
*
|
||||||
|
* $compoundForm->submit($arrayOfChildrenViewData)
|
||||||
|
* // inside:
|
||||||
|
* $childForm->submit($childViewData);
|
||||||
|
* // for each entry, do the same and/or reverse transform
|
||||||
|
* $this->dataMapper->mapFormsToData($compoundForm, $compoundInitialViewData)
|
||||||
|
* // then reverse transform
|
||||||
|
*
|
||||||
|
* When a simple form is submitted the following is happening:
|
||||||
|
*
|
||||||
|
* $simpleForm->submit($submittedViewData)
|
||||||
|
* // inside:
|
||||||
|
* $this->viewData = $submittedViewData
|
||||||
|
* // then reverse transform
|
||||||
|
*
|
||||||
|
* The model data can be an array or an object, so this second argument is always passed
|
||||||
|
* by reference.
|
||||||
|
*
|
||||||
|
* @param FormInterface[]|\Traversable $forms A list of {@link FormInterface} instances
|
||||||
|
* @param mixed $viewData The compound form's view data that get mapped
|
||||||
|
* its children model data
|
||||||
|
*
|
||||||
|
* @throws Exception\UnexpectedTypeException if the type of the data parameter is not supported
|
||||||
|
*/
|
||||||
|
public function mapFormsToData($forms, &$viewData)
|
||||||
|
{
|
||||||
|
foreach ($forms as $form) {
|
||||||
|
$value = $form->getData();
|
||||||
|
$this->resolver->setPermission(
|
||||||
|
$viewData,
|
||||||
|
$form->getParent()->getConfig()->getOption('perm_name'),
|
||||||
|
$form->getName(),
|
||||||
|
$value
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
72
src/Form/Permissions/PermissionsType.php
Normal file
72
src/Form/Permissions/PermissionsType.php
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* part-db version 0.1
|
||||||
|
* Copyright (C) 2005 Christoph Lechner
|
||||||
|
* http://www.cl-projects.de/
|
||||||
|
*
|
||||||
|
* part-db version 0.2+
|
||||||
|
* Copyright (C) 2009 K. Jacobs and others (see authors.php)
|
||||||
|
* http://code.google.com/p/part-db/
|
||||||
|
*
|
||||||
|
* Part-DB Version 0.4+
|
||||||
|
* Copyright (C) 2016 - 2019 Jan Böhmer
|
||||||
|
* https://github.com/jbtronics
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace App\Form\Permissions;
|
||||||
|
|
||||||
|
|
||||||
|
use App\Services\PermissionResolver;
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
|
||||||
|
class PermissionsType extends AbstractType
|
||||||
|
{
|
||||||
|
protected $resolver;
|
||||||
|
protected $perm_structure;
|
||||||
|
|
||||||
|
public function __construct(PermissionResolver $resolver)
|
||||||
|
{
|
||||||
|
$this->resolver = $resolver;
|
||||||
|
$this->perm_structure = $resolver->getPermissionStructure();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options)
|
||||||
|
{
|
||||||
|
$groups = $this->perm_structure['groups'];
|
||||||
|
|
||||||
|
foreach ($groups as $key => $group) {
|
||||||
|
$builder->add($key,PermissionGroupType::class, [
|
||||||
|
'group_name' => $key,
|
||||||
|
'mapped' => false,
|
||||||
|
'data' => $builder->getData(),
|
||||||
|
'disabled' => $options['disabled']
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$builder->add('blanko', PermissionGroupType::class, [
|
||||||
|
'group_name' => '*',
|
||||||
|
'label' => 'perm.group.other',
|
||||||
|
'mapped' => false,
|
||||||
|
'data' => $builder->getData(),
|
||||||
|
'disabled' => $options['disabled']
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
174
src/Form/Type/TriStateCheckboxType.php
Normal file
174
src/Form/Type/TriStateCheckboxType.php
Normal file
|
@ -0,0 +1,174 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* part-db version 0.1
|
||||||
|
* Copyright (C) 2005 Christoph Lechner
|
||||||
|
* http://www.cl-projects.de/
|
||||||
|
*
|
||||||
|
* part-db version 0.2+
|
||||||
|
* Copyright (C) 2009 K. Jacobs and others (see authors.php)
|
||||||
|
* http://code.google.com/p/part-db/
|
||||||
|
*
|
||||||
|
* Part-DB Version 0.4+
|
||||||
|
* Copyright (C) 2016 - 2019 Jan Böhmer
|
||||||
|
* https://github.com/jbtronics
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace App\Form\Type;
|
||||||
|
|
||||||
|
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\DataTransformerInterface;
|
||||||
|
use Symfony\Component\Form\Exception\TransformationFailedException;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\CheckboxType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\Form\FormInterface;
|
||||||
|
use Symfony\Component\Form\FormView;
|
||||||
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
|
|
||||||
|
class TriStateCheckboxType extends AbstractType implements DataTransformerInterface
|
||||||
|
{
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options) : void
|
||||||
|
{
|
||||||
|
$builder->addViewTransformer($this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function configureOptions(OptionsResolver $resolver)
|
||||||
|
{
|
||||||
|
$resolver->setDefaults([
|
||||||
|
'label_attr' => ['class' => 'checkbox-custom checkbox-inline'],
|
||||||
|
'attr' => ['class' => 'tristate'],
|
||||||
|
'compound' => false
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getBlockPrefix()
|
||||||
|
{
|
||||||
|
return 'tristate';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function buildView(FormView $view, FormInterface $form, array $options)
|
||||||
|
{
|
||||||
|
$view->vars = array_replace($view->vars, [
|
||||||
|
'value' => $form->getViewData(),
|
||||||
|
'checked' => true === $form->getData(),
|
||||||
|
'indeterminate' => null === $form->getData()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms a value from the original representation to a transformed representation.
|
||||||
|
*
|
||||||
|
* This method is called when the form field is initialized with its default data, on
|
||||||
|
* two occasions for two types of transformers:
|
||||||
|
*
|
||||||
|
* 1. Model transformers which normalize the model data.
|
||||||
|
* This is mainly useful when the same form type (the same configuration)
|
||||||
|
* has to handle different kind of underlying data, e.g The DateType can
|
||||||
|
* deal with strings or \DateTime objects as input.
|
||||||
|
*
|
||||||
|
* 2. View transformers which adapt the normalized data to the view format.
|
||||||
|
* a/ When the form is simple, the value returned by convention is used
|
||||||
|
* directly in the view and thus can only be a string or an array. In
|
||||||
|
* this case the data class should be null.
|
||||||
|
*
|
||||||
|
* b/ When the form is compound the returned value should be an array or
|
||||||
|
* an object to be mapped to the children. Each property of the compound
|
||||||
|
* data will be used as model data by each child and will be transformed
|
||||||
|
* too. In this case data class should be the class of the object, or null
|
||||||
|
* when it is an array.
|
||||||
|
*
|
||||||
|
* All transformers are called in a configured order from model data to view value.
|
||||||
|
* At the end of this chain the view data will be validated against the data class
|
||||||
|
* setting.
|
||||||
|
*
|
||||||
|
* This method must be able to deal with empty values. Usually this will
|
||||||
|
* be NULL, but depending on your implementation other empty values are
|
||||||
|
* possible as well (such as empty strings). The reasoning behind this is
|
||||||
|
* that data transformers must be chainable. If the transform() method
|
||||||
|
* of the first data transformer outputs NULL, the second must be able to
|
||||||
|
* process that value.
|
||||||
|
*
|
||||||
|
* @param mixed $value The value in the original representation
|
||||||
|
*
|
||||||
|
* @return mixed The value in the transformed representation
|
||||||
|
*
|
||||||
|
* @throws TransformationFailedException when the transformation fails
|
||||||
|
*/
|
||||||
|
public function transform($value)
|
||||||
|
{
|
||||||
|
if ($value === true) {
|
||||||
|
return "true";
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($value === false) {
|
||||||
|
return "false";
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($value === null) {
|
||||||
|
return "indeterminate";
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new \InvalidArgumentException('Invalid value encountered!: ' . $value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms a value from the transformed representation to its original
|
||||||
|
* representation.
|
||||||
|
*
|
||||||
|
* This method is called when {@link Form::submit()} is called to transform the requests tainted data
|
||||||
|
* into an acceptable format.
|
||||||
|
*
|
||||||
|
* The same transformers are called in the reverse order so the responsibility is to
|
||||||
|
* return one of the types that would be expected as input of transform().
|
||||||
|
*
|
||||||
|
* This method must be able to deal with empty values. Usually this will
|
||||||
|
* be an empty string, but depending on your implementation other empty
|
||||||
|
* values are possible as well (such as NULL). The reasoning behind
|
||||||
|
* this is that value transformers must be chainable. If the
|
||||||
|
* reverseTransform() method of the first value transformer outputs an
|
||||||
|
* empty string, the second value transformer must be able to process that
|
||||||
|
* value.
|
||||||
|
*
|
||||||
|
* By convention, reverseTransform() should return NULL if an empty string
|
||||||
|
* is passed.
|
||||||
|
*
|
||||||
|
* @param mixed $value The value in the transformed representation
|
||||||
|
*
|
||||||
|
* @return mixed The value in the original representation
|
||||||
|
*
|
||||||
|
* @throws TransformationFailedException when the transformation fails
|
||||||
|
*/
|
||||||
|
public function reverseTransform($value)
|
||||||
|
{
|
||||||
|
switch ($value) {
|
||||||
|
case "true":
|
||||||
|
return true;
|
||||||
|
case "false":
|
||||||
|
case '':
|
||||||
|
return false;
|
||||||
|
case "indeterminate":
|
||||||
|
return null;
|
||||||
|
default:
|
||||||
|
throw new \InvalidArgumentException('Invalid value encountered!: ' . $value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -35,6 +35,8 @@ namespace App\Form;
|
||||||
use App\Entity\UserSystem\Group;
|
use App\Entity\UserSystem\Group;
|
||||||
use App\Entity\Base\NamedDBElement;
|
use App\Entity\Base\NamedDBElement;
|
||||||
use App\Entity\Base\StructuralDBElement;
|
use App\Entity\Base\StructuralDBElement;
|
||||||
|
use App\Form\Permissions\PermissionsType;
|
||||||
|
use App\Form\Permissions\PermissionType;
|
||||||
use App\Form\Type\StructuralEntityType;
|
use App\Form\Type\StructuralEntityType;
|
||||||
use FOS\CKEditorBundle\Form\Type\CKEditorType;
|
use FOS\CKEditorBundle\Form\Type\CKEditorType;
|
||||||
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
|
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
|
||||||
|
@ -111,6 +113,11 @@ class UserAdminForm extends AbstractType
|
||||||
'disabled' => !$this->security->isGranted('edit_infos', $entity),
|
'disabled' => !$this->security->isGranted('edit_infos', $entity),
|
||||||
])
|
])
|
||||||
|
|
||||||
|
->add('permissions', PermissionsType::class, [
|
||||||
|
'mapped' => false,
|
||||||
|
'data' => $builder->getData(),
|
||||||
|
//'user' => $builder->getData(),
|
||||||
|
])
|
||||||
;
|
;
|
||||||
/*->add('comment', CKEditorType::class, ['required' => false,
|
/*->add('comment', CKEditorType::class, ['required' => false,
|
||||||
'label' => 'comment.label', 'attr' => ['rows' => 4], 'help' => 'bbcode.hint',
|
'label' => 'comment.label', 'attr' => ['rows' => 4], 'help' => 'bbcode.hint',
|
||||||
|
|
|
@ -60,12 +60,17 @@ class PermissionResolver
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
$this->permission_structure = $this->getPermissionStructure();
|
$this->permission_structure = $this->generatePermissionStructure();
|
||||||
|
|
||||||
//dump($this->permission_structure);
|
//dump($this->permission_structure);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getPermissionStructure()
|
public function getPermissionStructure() : array
|
||||||
|
{
|
||||||
|
return $this->permission_structure;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function generatePermissionStructure()
|
||||||
{
|
{
|
||||||
|
|
||||||
$cache = new ConfigCache($this->cache_file, $this->is_debug);
|
$cache = new ConfigCache($this->cache_file, $this->is_debug);
|
||||||
|
@ -166,6 +171,24 @@ class PermissionResolver
|
||||||
return null; //The inherited value is never resolved. Should be treat as false, in Voters.
|
return null; //The inherited value is never resolved. Should be treat as false, in Voters.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the new value for the operation
|
||||||
|
* @param HasPermissionsInterface $user The user or group for which the value should be changed.
|
||||||
|
* @param string $permission The name of the permission that should be changed.
|
||||||
|
* @param string $operation The name of the operation that should be changed.
|
||||||
|
* @param bool|null $new_val The new value for the permission. true = ALLOW, false = DISALLOW, null = INHERIT
|
||||||
|
*/
|
||||||
|
public function setPermission(HasPermissionsInterface $user, string $permission, string $operation, ?bool $new_val) : void
|
||||||
|
{
|
||||||
|
//Get the permissions from the user
|
||||||
|
$perm_list = $user->getPermissions();
|
||||||
|
|
||||||
|
//Determine bit number using our configuration
|
||||||
|
$bit = $this->permission_structure['perms'][$permission]['operations'][$operation]['bit'];
|
||||||
|
|
||||||
|
$perm_list->setPermissionValue($permission, $bit, $new_val);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lists the names of all operations that is supported for the given permission.
|
* Lists the names of all operations that is supported for the given permission.
|
||||||
*
|
*
|
||||||
|
|
|
@ -6,6 +6,11 @@
|
||||||
|
|
||||||
{% block comment %}{% endblock %}
|
{% block comment %}{% endblock %}
|
||||||
|
|
||||||
|
{% block additional_pills %}
|
||||||
|
<li class="nav-item"><a data-toggle="tab" class="nav-link link-anchor" href="#tab_permissions">{% trans %}admin.options{% endtrans %}</a></li>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
{% block additional_controls %}
|
{% block additional_controls %}
|
||||||
{{ form_row(form.group) }}
|
{{ form_row(form.group) }}
|
||||||
{{ form_row(form.first_name) }}
|
{{ form_row(form.first_name) }}
|
||||||
|
@ -13,3 +18,9 @@
|
||||||
{{ form_row(form.email) }}
|
{{ form_row(form.email) }}
|
||||||
{{ form_row(form.department) }}
|
{{ form_row(form.department) }}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block additional_panes %}
|
||||||
|
<div class="tab-pane" id="tab_permissions">
|
||||||
|
{{ form_row(form.permissions) }}
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
|
@ -45,3 +45,67 @@
|
||||||
</div>
|
</div>
|
||||||
{{ form_errors(form.value) }}
|
{{ form_errors(form.value) }}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
{#######################################################################################
|
||||||
|
#
|
||||||
|
# Definitions for Tristate Checkbox Type (mostly based on bootstrap checkbox type)
|
||||||
|
#
|
||||||
|
#######################################################################################}
|
||||||
|
|
||||||
|
{% block tristate_label -%}
|
||||||
|
{#- Do not display the label if widget is not defined in order to prevent double label rendering -#}
|
||||||
|
{%- if widget is defined -%}
|
||||||
|
{% set is_parent_custom = parent_label_class is defined and ('checkbox-custom' in parent_label_class or 'radio-custom' in parent_label_class) %}
|
||||||
|
{% set is_custom = label_attr.class is defined and ('checkbox-custom' in label_attr.class or 'radio-custom' in label_attr.class) %}
|
||||||
|
{%- if is_parent_custom or is_custom -%}
|
||||||
|
{%- set label_attr = label_attr|merge({class: (label_attr.class|default('') ~ ' custom-control-label')|trim}) -%}
|
||||||
|
{%- else %}
|
||||||
|
{%- set label_attr = label_attr|merge({class: (label_attr.class|default('') ~ ' form-check-label')|trim}) -%}
|
||||||
|
{%- endif %}
|
||||||
|
{%- if not compound -%}
|
||||||
|
{% set label_attr = label_attr|merge({'for': id}) %}
|
||||||
|
{%- endif -%}
|
||||||
|
{%- if required -%}
|
||||||
|
{%- set label_attr = label_attr|merge({class: (label_attr.class|default('') ~ ' required')|trim}) -%}
|
||||||
|
{%- endif -%}
|
||||||
|
{%- if parent_label_class is defined -%}
|
||||||
|
{%- set label_attr = label_attr|merge({class: (label_attr.class|default('') ~ ' ' ~ parent_label_class)|replace({'checkbox-inline': '', 'radio-inline': '', 'checkbox-custom': '', 'radio-custom': ''})|trim}) -%}
|
||||||
|
{%- endif -%}
|
||||||
|
{%- if label is not same as(false) and label is empty -%}
|
||||||
|
{%- if label_format is not empty -%}
|
||||||
|
{%- set label = label_format|replace({
|
||||||
|
'%name%': name,
|
||||||
|
'%id%': id,
|
||||||
|
}) -%}
|
||||||
|
{%- else -%}
|
||||||
|
{%- set label = name|humanize -%}
|
||||||
|
{%- endif -%}
|
||||||
|
{%- endif -%}
|
||||||
|
|
||||||
|
{{ widget|raw }}
|
||||||
|
<label{% with { attr: label_attr } %}{{ block('attributes') }}{% endwith %}>
|
||||||
|
{{- label is not same as(false) ? (translation_domain is same as(false) ? label : label|trans(label_translation_parameters, translation_domain)) -}}
|
||||||
|
{{- form_errors(form) -}}
|
||||||
|
</label>
|
||||||
|
{%- endif -%}
|
||||||
|
{%- endblock tristate_label %}
|
||||||
|
|
||||||
|
{%- block tr_parent -%}
|
||||||
|
<input type="checkbox" {{ block('widget_attributes') }}{% if value is defined %} value="{{ value }}"{% endif %}{% if checked %} checked="checked"{% endif %}{% if indeterminate %} indeterminate="indeterminate"{% endif %} />
|
||||||
|
{%- endblock tr_parent -%}
|
||||||
|
|
||||||
|
{% block tristate_widget -%}
|
||||||
|
{%- set parent_label_class = parent_label_class|default(label_attr.class|default('')) -%}
|
||||||
|
{%- if 'checkbox-custom' in parent_label_class -%}
|
||||||
|
{%- set attr = attr|merge({class: (attr.class|default('') ~ ' custom-control-input')|trim}) -%}
|
||||||
|
<div class="custom-control custom-checkbox{{ 'checkbox-inline' in parent_label_class ? ' custom-control-inline' }}">
|
||||||
|
{{- form_label(form, null, { widget: block('tr_parent') }) -}}
|
||||||
|
</div>
|
||||||
|
{%- else -%}
|
||||||
|
{%- set attr = attr|merge({class: (attr.class|default('') ~ ' form-check-input')|trim}) -%}
|
||||||
|
<div class="form-check{{ 'checkbox-inline' in parent_label_class ? ' form-check-inline' }}">
|
||||||
|
{{- form_label(form, null, { widget: block('tr_parent') }) -}}
|
||||||
|
</div>
|
||||||
|
{%- endif -%}
|
||||||
|
{%- endblock tristate_widget %}
|
||||||
|
|
||||||
|
|
52
templates/Form/permissionLayout.html.twig
Normal file
52
templates/Form/permissionLayout.html.twig
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
{% block permission_row %}
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<b>{{ form.vars.label | trans }}</b>
|
||||||
|
{{ form_errors(form) }}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{% for op in form %}
|
||||||
|
{{ form_widget(op) }}
|
||||||
|
{{ form_errors(op) }}
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block permission_group_row %}
|
||||||
|
{{ form_errors(form) }}
|
||||||
|
<table class="table table-bordered table-sm">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>{% trans %}permission.edit.permission{% endtrans %}</th>
|
||||||
|
<th>{% trans %}permission.edit.value{% endtrans %}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
{% for perm in form %}
|
||||||
|
{{ form_row(perm) }}
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block permissions_row %}
|
||||||
|
<ul class="nav nav-tabs">
|
||||||
|
{% for group in form %}
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link {% if loop.first %}active{% endif %}"
|
||||||
|
data-toggle="tab" role="tab" href="#group_row_{{ group.vars.name }}">{{ group.vars.label | trans }}</a>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div class="tab-content mt-2">
|
||||||
|
{% for group in form %}
|
||||||
|
<div class="tab-pane {% if loop.first %}active{% endif %}" id="group_row_{{ group.vars.name }}">
|
||||||
|
{{ form_row(group) }}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% endblock %}
|
Loading…
Add table
Add a link
Reference in a new issue