Merge branch 'stable'

This commit is contained in:
Yuri Kuznetsov
2025-10-21 11:37:06 +03:00
53 changed files with 766 additions and 595 deletions

View File

@@ -73,12 +73,18 @@ class UserColumnsLoader implements Loader
return;
}
$entity->set([
$values = [
Email::USERS_COLUMN_IS_READ => $emailUser->get(Email::USERS_COLUMN_IS_READ),
Email::USERS_COLUMN_IS_IMPORTANT => $emailUser->get(Email::USERS_COLUMN_IS_IMPORTANT),
Email::USERS_COLUMN_IN_TRASH => $emailUser->get(Email::USERS_COLUMN_IN_TRASH),
Email::USERS_COLUMN_IN_ARCHIVE => $emailUser->get(Email::USERS_COLUMN_IN_ARCHIVE),
'isUsersSent' => $entity->getSentBy()?->getId() === $this->user->getId(),
]);
];
$entity->setMultiple($values);
foreach ($values as $key => $value) {
$entity->setFetched($key, $value);
}
}
}

View File

@@ -31,8 +31,10 @@ namespace Espo\Classes\RecordHooks\Email;
use Espo\Core\Mail\EmailSender;
use Espo\Core\Name\Field;
use Espo\Core\ORM\Type\FieldType;
use Espo\Core\Record\Hook\SaveHook;
use Espo\Core\Utils\FieldUtil;
use Espo\Core\Utils\Metadata;
use Espo\Core\Utils\SystemUser;
use Espo\Entities\Email;
use Espo\Entities\User;
@@ -54,7 +56,8 @@ class BeforeUpdate implements SaveHook
public function __construct(
private User $user,
private EntityManager $entityManager,
private FieldUtil $fieldUtil
private FieldUtil $fieldUtil,
private Metadata $metadata,
) {}
public function process(Entity $entity): void
@@ -125,18 +128,24 @@ class BeforeUpdate implements SaveHook
private function clearEntityForUpdate(Email $email): void
{
$fieldDefsList = $this->entityManager
$entityDefs = $this->entityManager
->getDefs()
->getEntity(Email::ENTITY_TYPE)
->getFieldList();
->getEntity(Email::ENTITY_TYPE);
foreach ($fieldDefsList as $fieldDefs) {
foreach ($entityDefs->getFieldList() as $fieldDefs) {
$field = $fieldDefs->getName();
if ($fieldDefs->getParam('isCustom')) {
continue;
}
if (
$fieldDefs->getType() === FieldType::LINK_MULTIPLE &&
$this->metadata->get("entityDefs.Email.links.$field.isCustom")
) {
continue;
}
if (in_array($field, $this->allowedForUpdateFieldList)) {
continue;
}
@@ -144,7 +153,9 @@ class BeforeUpdate implements SaveHook
$attributeList = $this->fieldUtil->getAttributeList(Email::ENTITY_TYPE, $field);
foreach ($attributeList as $attribute) {
$email->clear($attribute);
if ($email->isAttributeChanged($attribute) && $email->isAttributeWritten($attribute)) {
$email->set($attribute, $email->getFetched($attribute));
}
}
}
}

View File

@@ -36,6 +36,7 @@ use Espo\Core\Formula\Exceptions\Error;
use Espo\Core\Formula\Functions\BaseFunction;
use Espo\Core\Formula\Functions\RecordGroup\Util\FindQueryUtil;
use Espo\Core\Di;
use Espo\Core\Select\Primary\Filters\All;
use Espo\Core\Select\SelectBuilderFactory;
/**
@@ -68,6 +69,7 @@ class CountType extends BaseFunction implements
$builder = $this->injectableFactory->create(SelectBuilderFactory::class)
->create()
->forUser($this->user)
->withPrimaryFilter(All::NAME)
->from($entityType);
(new FindQueryUtil())->applyFilter($builder, $filter, 2);

View File

@@ -36,6 +36,7 @@ use Espo\Core\Formula\Exceptions\Error;
use Espo\Core\Formula\Functions\BaseFunction;
use Espo\Core\Di;
use Espo\Core\Formula\Functions\RecordGroup\Util\FindQueryUtil;
use Espo\Core\Select\Primary\Filters\All;
use Espo\Core\Select\SelectBuilderFactory;
/**
@@ -68,6 +69,7 @@ class ExistsType extends BaseFunction implements
$builder = $this->injectableFactory->create(SelectBuilderFactory::class)
->create()
->forUser($this->user)
->withPrimaryFilter(All::NAME)
->from($entityType);
(new FindQueryUtil())->applyFilter($builder, $filter, 2);

View File

@@ -38,6 +38,7 @@ use Espo\Core\Formula\Exceptions\Error as FormulaError;
use Espo\Core\Formula\Exceptions\TooFewArguments;
use Espo\Core\Formula\Func;
use Espo\Core\Formula\Functions\RecordGroup\Util\FindQueryUtil;
use Espo\Core\Select\Primary\Filters\All;
use Espo\Core\Select\SelectBuilderFactory;
use Espo\ORM\Entity;
use Espo\ORM\EntityManager;
@@ -92,6 +93,7 @@ class FindManyType implements Func
$builder = $this->selectBuilderFactory
->create()
->withPrimaryFilter(All::NAME)
->from($entityType);
$whereClause = [];

View File

@@ -36,6 +36,7 @@ use Espo\Core\Formula\Exceptions\Error as FormulaError;
use Espo\Core\Formula\Functions\BaseFunction;
use Espo\Core\Di;
use Espo\Core\Formula\Functions\RecordGroup\Util\FindQueryUtil;
use Espo\Core\Select\Primary\Filters\All;
use Espo\Core\Select\SelectBuilderFactory;
use Espo\ORM\Name\Attribute;
use Espo\ORM\Query\Part\Order;
@@ -65,6 +66,7 @@ class FindOneType extends BaseFunction implements
$builder = $this->injectableFactory->create(SelectBuilderFactory::class)
->create()
->forUser($this->user)
->withPrimaryFilter(All::NAME)
->from($entityType);
$whereClause = [];

View File

@@ -41,6 +41,7 @@ use Espo\Core\Formula\ArgumentList;
use Espo\Core\Formula\Functions\BaseFunction;
use Espo\Core\Di;
use Espo\Core\Select\Helpers\RandomStringGenerator;
use Espo\Core\Select\Primary\Filters\All;
use Espo\Core\Select\SelectBuilderFactory;
use Espo\ORM\Defs\Params\RelationParam;
use Espo\ORM\Name\Attribute;
@@ -165,6 +166,7 @@ class FindRelatedManyType extends BaseFunction implements
$builder = $this->injectableFactory->create(SelectBuilderFactory::class)
->create()
->forUser($this->user)
->withPrimaryFilter(All::NAME)
->from($foreignEntityType);
$whereClause = [];

View File

@@ -32,6 +32,7 @@ namespace Espo\Core\Formula\Functions\RecordGroup;
use Espo\Core\Exceptions\BadRequest;
use Espo\Core\Exceptions\Forbidden;
use Espo\Core\ORM\Entity as CoreEntity;
use Espo\Core\Select\Primary\Filters\All;
use Espo\Core\Select\SelectBuilderFactory;
use Espo\ORM\Defs\Params\RelationParam;
use Espo\ORM\Name\Attribute;
@@ -151,6 +152,7 @@ class FindRelatedOneType extends BaseFunction implements
$builder = $this->injectableFactory->create(SelectBuilderFactory::class)
->create()
->forUser($this->user)
->withPrimaryFilter(All::NAME)
->from($foreignEntityType);
$whereClause = [];

View File

@@ -95,7 +95,7 @@ class Entity extends BaseEntity
throw new LogicException("No entity-manager.");
}
$toSetFetched = !$this->isNew() && !$this->hasFetched($idAttribute);
$toSetFetched = !$this->isNew() && !$this->isAttributeChanged($idAttribute);
if (!$parentId || !$parentType) {
/** @noinspection PhpRedundantOptionalArgumentInspection */

View File

@@ -129,6 +129,12 @@ class LogicDefsBc implements AdditionalBuilder
$item = $subDefs->$subKey;
$logicDefs->$key ??= (object) [];
// Fix if corrupted.
if (is_array($logicDefs->$key)) {
$logicDefs->$key = (object) [];
}
$logicDefs->$key->$name ??= (object) [];
$logicDefs->$key->$name->$subKey = $item !== null ?

View File

@@ -126,6 +126,24 @@ class Email extends Entity
return parent::has($attribute);
}
public function getFetched(string $attribute): mixed
{
if ($attribute === 'subject') {
return $this->getFetched(Field::NAME);
}
return parent::getFetched($attribute);
}
public function isAttributeChanged(string $name): bool
{
if ($name === 'subject') {
$name = Field::NAME;
}
return parent::isAttributeChanged($name);
}
/** @noinspection PhpUnused */
protected function _setSubject(?string $value): void
{

View File

@@ -1,15 +1,14 @@
[
{
"label":"",
"rows":[
[{"name":"name", "fullWidth": true}],
[{"name":"name"}],
[{"name":"status"}, {"name":"direction"}],
[{"name":"dateStart", "fullWidth": true}],
[{"name":"duration", "fullWidth": true}],
[{"name":"dateEnd", "fullWidth": true}],
[{"name":"parent", "fullWidth": true}],
[{"name":"reminders", "fullWidth": true}],
[{"name":"description", "fullWidth": true}]
[{"name":"dateStart"}],
[{"name":"duration"}, false],
[{"name":"dateEnd"}],
[{"name":"parent"}],
[{"name":"reminders"}],
[{"name":"description"}]
]
}
]
]

View File

@@ -1,14 +1,12 @@
[
{
"label":"",
"rows":[
[{"name":"name", "fullWidth": true}],
[{"name":"status"}, {"name":"priority"}],
[{"name":"type"}, {"name":"number"}],
[{"name":"account", "fullWidth": true}],
[{"name":"contacts", "fullWidth": true}],
[{"name":"description", "fullWidth": true}],
[{"name":"attachments", "fullWidth": true}]
[{"name": "name"}],
[{"name": "status"}, {"name": "priority"}],
[{"name": "type"}, {"name": "number"}],
[{"name": "account"}, {"name": "contacts"}],
[{"name": "description"}],
[{"name": "attachments"}]
]
}
]
]

View File

@@ -1,15 +1,14 @@
[
{
"label":"",
"rows":[
[{"name":"name", "fullWidth": true}],
[{"name":"status", "fullWidth": true}],
[{"name":"dateStart", "fullWidth": true}],
[{"name":"duration", "fullWidth": true}],
[{"name":"dateEnd", "fullWidth": true}],
[{"name":"parent", "fullWidth": true}],
[{"name":"reminders", "fullWidth": true}],
[{"name":"description", "fullWidth": true}]
[{"name":"name"}],
[{"name":"status"}, false],
[{"name":"dateStart"}],
[{"name":"duration"}, false],
[{"name":"dateEnd"}],
[{"name":"parent"}],
[{"name":"reminders"}],
[{"name":"description"}]
]
}
]
]

View File

@@ -186,13 +186,21 @@ class Email extends Database implements
return;
}
$setFetched = !$entity->isAttributeChanged($type . 'EmailAddressesNames');
$addresses = [];
foreach (get_object_vars($names) as $address) {
$addresses[] = $address;
}
$entity->set($type, implode(';', $addresses));
$value = implode(';', $addresses);
$entity->set($type, $value);
if ($setFetched) {
$entity->setFetched($type, $value);
}
}
/**
@@ -289,6 +297,10 @@ class Email extends Database implements
$entity->set('nameHash', $nameHash);
$entity->set('typeHash', $typeHash);
$entity->set('idHash', $idHash);
$entity->setFetched('nameHash', $nameHash);
$entity->setFetched('typeHash', $typeHash);
$entity->setFetched('idHash', $idHash);
}
/**

View File

@@ -89,6 +89,10 @@
{
"type": "isTrue",
"attribute": "ldapAuth"
}, {
"type": "equals",
"attribute": "authenticationMethod",
"value": "LDAP"
}
]
}

View File

@@ -2,21 +2,25 @@
"fields": {
"name": {
"type": "varchar",
"required": true
"required": true,
"audited": true
},
"subject": {
"type": "varchar"
"type": "varchar",
"audited": true
},
"body": {
"type": "wysiwyg",
"view": "views/email-template/fields/body",
"useIframe": true,
"attachmentField": "attachments"
"attachmentField": "attachments",
"audited": true
},
"isHtml": {
"type": "bool",
"default": true,
"inlineEditDisabled": true
"inlineEditDisabled": true,
"audited": true
},
"status": {
"type": "enum",
@@ -28,7 +32,8 @@
"style": {
"Inactive": "info"
},
"maxLength": 8
"maxLength": 8,
"audited": true
},
"oneOff": {
"type": "bool",
@@ -36,7 +41,8 @@
"tooltip": true
},
"attachments": {
"type": "attachmentMultiple"
"type": "attachmentMultiple",
"audited": true
},
"category": {
"type": "link",
@@ -47,7 +53,8 @@
"view": "views/fields/assigned-user"
},
"teams": {
"type": "linkMultiple"
"type": "linkMultiple",
"audited": true
},
"createdAt": {
"type": "datetime",

View File

@@ -1,7 +1,7 @@
{
"stylesheet": "client/css/espo/hazyblue.css",
"stylesheetIframe": "client/css/espo/hazyblue-iframe.css",
"logo": "client/img/logo-hazy.svg",
"logo": "client/img/logo-light.svg",
"textColor": "#333",
"chartGridColor": "#ddd",
"chartTickColor": "#e8eced",

View File

@@ -1,17 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<svg width="318" height="78" enable-background="new 0 0 307.813 75" overflow="visible" version="1.1" viewBox="0 0 318 78" xml:space="preserve" xmlns="http://www.w3.org/2000/svg">
<switch transform="matrix(1.089 0 0 1.089 -.89733 -.52658)">
<foreignObject width="1" height="1" requiredExtensions="http://ns.adobe.com/AdobeIllustrator/10.0/">
</foreignObject>
<g transform="matrix(.96767 0 0 .96767 3.9659 -1.2011)">
<path d="m169.53 21.864c-7.453 2.972-9.569 11.987-9.005 19.212 1.587 2.982 3.845 5.562 5.783 8.312l4.262-1.083c-1.796-4.447-1.689-9.424-0.806-14.066 0.585-3.001 2.309-6.476 5.634-7.032 5.307-0.847 10.733-0.271 16.088-0.369 0.091-2.196 0.115-4.392 0.107-6.585-7.333 0.387-15.043-1.038-22.063 1.611zm52.714-1.294c-8.12-0.952-16.332-0.149-24.492-0.387-0.021 6.43-3e-3 12.854 0.078 19.274 2.625-0.849 5.251-1.739 7.909-2.532 0.042-3.272 0.028-6.527-0.071-9.789 4.869-0.029 9.874-0.757 14.639 0.451 1.838 0.298 2.051 2.25 2.687 3.641 2.541-0.891 5.111-1.717 7.672-2.574-0.703-4.246-4.129-7.633-8.422-8.084zm23.522-0.593c-3.954 0.072-7.912 0.064-11.864 0.047 0.051 2.544 0.063 5.074 0.072 7.617 4.263-1.482 8.553-2.889 12.848-4.268-0.35-1.128-0.706-2.268-1.056-3.396z" fill="#75797c"/>
<path d="m161.96 69.125c7.886-3.717 15.757-7.463 23.72-11.018 5.563 0.359 11.146 0.021 16.722 0.193 1.14-0.036 2.292-0.061 3.432-0.088-0.011-3.195-0.025-6.38-0.082-9.564 3.428-1.502 10.227-4.623 10.227-4.623l15.215 13.941 11.096 0.106-0.715-26.236 0.803-0.211 9.005 26.344 8.834-0.066 8.99-28.394-0.308 28.434 8.074-0.021-0.231-37.932-9.279 0.071 30.625-14.141s-37.593 14.279-56.404 21.385c-2.996 1.022-5.878 2.315-8.853 3.394-2.278 0.867-4.558 1.713-6.834 2.58-20.071 7.526-39.945 15.604-60.126 22.803-6.777-10.522-15.314-19.854-21.768-30.585zm72.116-17.961c-0.108 0.154-0.324 0.458-0.429 0.611-3.448-3.018-6.765-6.189-10.21-9.205 1.745-1.096 3.47-2.242 5.026-3.597 1.625-1.386 3.479-2.469 5.345-3.499 0.293 5.227 0.258 10.452 0.268 15.69zm23.942-9.67c-0.857 2.578-1.825 5.137-2.793 7.682-1.644-6.217-3.94-12.238-5.856-18.383-0.119-0.52-0.366-1.574-0.487-2.093 3.428-1.709 10.585-4.854 15.229-6.815-1.647 5.969-4.306 14.029-6.093 19.609z" fill="#ddaf28"/>
<g fill="#75797c">
<path d="m45.672 58.148h-18.526c-2.861 0-5.614-0.651-8.257-1.953-2.861-1.409-5.043-3.651-6.547-6.725-1.503-3.074-2.254-6.455-2.254-10.145 0-3.652 0.724-6.961 2.173-9.926 1.594-3.219 3.803-5.569 6.628-7.052 1.557-0.795 3.052-1.355 4.482-1.682 1.43-0.325 3.07-0.488 4.917-0.488h17.168v6.789h-15.886c-1.415 0-2.602 0.187-3.563 0.558-0.961 0.372-1.912 1.037-2.855 1.994s-1.597 1.887-1.959 2.791c-0.363 0.902-0.543 2.027-0.543 3.375h25.023v6.789h-25.025c0 1.24 0.164 2.325 0.491 3.256 0.327 0.93 0.919 1.887 1.776 2.871 0.856 0.985 1.749 1.732 2.677 2.242 0.929 0.512 2.03 0.767 3.306 0.767h16.774z"/>
<path d="m76.499 49.519c0 2.397-0.771 4.449-2.312 6.154-1.541 1.706-3.49 2.56-5.846 2.56h-18.653v-5.113h15.326c1.087 0 2.001-0.272 2.744-0.817s1.115-1.327 1.115-2.345c0-2.362-1.595-3.543-4.783-3.543h-7.825c-1.666 0-3.278-0.79-4.836-2.369-1.559-1.58-2.336-3.287-2.336-5.119 0-2.585 0.579-4.667 1.738-6.248 1.34-1.794 3.313-2.692 5.922-2.692h17.928v5.364h-15.938c-0.614 0-1.147 0.289-1.599 0.868s-0.677 1.235-0.677 1.972c0 0.807 0.298 1.498 0.896 2.076 0.597 0.579 1.311 0.867 2.144 0.867h8.415c2.643 0 4.733 0.79 6.271 2.369 1.536 1.579 2.306 3.584 2.306 6.016z"/>
<path d="m109.29 43.414c0 4.495-1.166 8.074-3.497 10.738s-5.395 3.996-9.188 3.996h-8.186v10.309h-7.627v-38.472h15.09c4.27 0 7.6 1.269 9.989 3.806 2.279 2.428 3.419 5.637 3.419 9.623zm-7.627 0.405c0-2.356-0.754-4.286-2.262-5.793-1.509-1.505-3.388-2.258-5.641-2.258h-5.341v16.429h5.886c2.179 0 3.951-0.771 5.313-2.313 1.363-1.54 2.045-3.562 2.045-6.065z"/>
<path d="m145.1 43.967c0 4.896-1.557 8.65-4.669 11.261-2.86 2.394-6.751 3.591-11.673 3.591-4.923 0-8.742-1.087-11.456-3.264-3.15-2.502-4.724-6.401-4.724-11.696 0-4.424 1.701-7.906 5.104-10.446 3.04-2.283 6.786-3.427 11.238-3.427 4.887 0 8.805 1.225 11.754 3.673s4.426 5.884 4.426 10.308zm-8.382-0.065c0-2.285-0.716-4.197-2.146-5.738-1.432-1.54-3.379-2.312-5.841-2.312-2.246 0-4.103 0.79-5.57 2.366-1.467 1.577-2.2 3.563-2.2 5.955 0 2.756 0.743 4.949 2.228 6.581s3.405 2.448 5.76 2.448c2.679 0 4.673-0.852 5.977-2.557 1.193-1.557 1.792-3.805 1.792-6.743z"/>
</g>
</g>
</switch>
</svg>

Before

Width:  |  Height:  |  Size: 4.3 KiB

View File

@@ -26,12 +26,8 @@
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
************************************************************************/
define('crm:views/knowledge-base-article/record/detail-quick', ['views/record/detail-small'], function (Dep) {
import DetailRecordView from 'views/record/detail';
return Dep.extend({
isWide: true,
sideView: false,
});
});
export default class extends DetailRecordView {
}

View File

@@ -26,11 +26,8 @@
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
************************************************************************/
define('crm:views/knowledge-base-article/record/edit-quick', ['views/record/edit-small'], function (Dep) {
import EditRecordView from 'views/record/edit';
return Dep.extend({
export default class extends EditRecordView {
isWide: true,
sideView: false,
});
});
}

View File

@@ -1,30 +1,40 @@
<div class="panel panel-default no-side-margin">
<div class="panel-body">
<div class="row">
<div class="cell form-group col-md-6" data-name="currentPassword">
<label
class="control-label"
data-name="currentPassword"
>{{translate 'currentPassword' scope='User' category='fields'}}</label>
<div class="field" data-name="currentPassword">{{{currentPassword}}}</div>
</div>
</div>
<div class="row">
<div class="cell form-group col-md-6" data-name="password">
<label
class="control-label"
data-name="password"
>{{translate 'newPassword' scope='User' category='fields'}}</label>
<div class="field" data-name="password">{{{password}}}</div>
</div>
</div>
<div class="row">
<div class="cell form-group col-md-6" data-name="passwordConfirm">
<label
class="control-label"
data-name="passwordConfirm"
>{{translate 'passwordConfirm' scope='User' category='fields'}}</label>
<div class="field" data-name="passwordConfirm">{{{passwordConfirm}}}</div>
<div class="no-side-margin record">
<div>
<div class="record-grid-wide">
<div class="left">
<div class="middle">
<div class="panel panel-default first last">
<div class="panel-body panel-body-form">
<div class="row">
<div class="cell form-group col-md-6" data-name="currentPassword">
<label
class="control-label"
data-name="currentPassword"
>{{translate 'currentPassword' scope='User' category='fields'}}</label>
<div class="field" data-name="currentPassword">{{{currentPassword}}}</div>
</div>
</div>
<div class="row">
<div class="cell form-group col-md-6" data-name="password">
<label
class="control-label"
data-name="password"
>{{translate 'newPassword' scope='User' category='fields'}}</label>
<div class="field" data-name="password">{{{password}}}</div>
</div>
</div>
<div class="row">
<div class="cell form-group col-md-6" data-name="passwordConfirm">
<label
class="control-label"
data-name="passwordConfirm"
>{{translate 'passwordConfirm' scope='User' category='fields'}}</label>
<div class="field" data-name="passwordConfirm">{{{passwordConfirm}}}</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>

View File

@@ -1,26 +1,37 @@
<div class="panel panel-default no-side-margin">
<div class="panel-body panel-body-form">
<div class="row">
<div class="cell form-group col-md-6" data-name="dashboardTabList">
<label
class="control-label"
data-name="dashboardTabList"
>{{translate 'dashboardTabList' category='fields' scope="Preferences"}}</label>
<div class="field" data-name="dashboardTabList">
{{{dashboardTabList}}}
</div>
</div>
{{#if hasLocked}}
<div class="cell form-group col-md-6" data-name="dashboardLocked">
<label
class="control-label"
data-name="dashboardLocked"
>{{translate 'dashboardLocked' category='fields' scope="Preferences"}}</label>
<div class="field" data-name="dashboardLocked">
{{{dashboardLocked}}}
<div class="no-side-margin record">
<div>
<div class="record-grid-wide">
<div class="left">
<div class="middle">
<div class="panel panel-default first last">
<div class="panel-body panel-body-form">
<div class="row">
<div class="cell form-group col-md-6" data-name="dashboardTabList">
<label
class="control-label"
data-name="dashboardTabList"
>{{translate 'dashboardTabList' category='fields' scope="Preferences"}}</label>
<div class="field" data-name="dashboardTabList">
{{{dashboardTabList}}}
</div>
</div>
{{#if hasLocked}}
<div class="cell form-group col-md-6" data-name="dashboardLocked">
<label
class="control-label"
data-name="dashboardLocked"
>{{translate 'dashboardLocked' category='fields' scope="Preferences"}}</label>
<div class="field" data-name="dashboardLocked">
{{{dashboardLocked}}}
</div>
</div>
{{/if}}
</div>
</div>
</div>
</div>
{{/if}}
</div>
</div>
</div>
</div>

View File

@@ -25,9 +25,11 @@
role="button"
tabindex="0"
data-action="expandDetails"
class="text-soft"
><span class="fas fa-chevron-down"></span></a><span style="user-select: none"> </span>
<span class="fields text-muted small">{{fieldsString}}</span>
class="text-muted no-underline"
><span class="fas fa-chevron-down text-soft" data-role="icon"></span>
<span style="user-select: none"> </span>
<span class="fields small">{{fieldsString}}</span>
</a>
</div>
{{/if}}

View File

@@ -3,7 +3,7 @@
<div class="buttons-panel margin hide floated-row clearfix">
<div>
<button class="btn btn-primary btn-xs-wide post">{{translate 'Post'}}</button>
{{#if allowInternalNotes}}
{{~#if allowInternalNotes~}}
<span
style="cursor: pointer;"
class="internal-mode-switcher{{#if isInternalNoteMode}} enabled{{/if}} action"
@@ -12,7 +12,7 @@
>
<span class="fas fa-lock"></span>
</span>
{{/if}}
{{~/if~}}
</div>
<div class="attachments-container">
{{{attachments}}}

View File

@@ -30,6 +30,12 @@
export default class {
/**
* @private
* @type {string}
*/
entityType
/**
* @param {module:views/fields/base} view A field view.
*/
@@ -48,6 +54,8 @@ export default class {
const entityType = metadata.get(['entityDefs', model.entityType, 'links', link, 'entity']) ||
model.entityType;
this.entityType = entityType;
const fieldDefs = metadata.get(['entityDefs', entityType, 'fields', field]) || {};
const type = fieldDefs.type;
@@ -78,4 +86,11 @@ export default class {
getForeignParams() {
return Espo.Utils.cloneDeep(this.foreignParams);
}
/**
* @return {string}
*/
getEntityType() {
return this.entityType;
}
}

View File

@@ -121,6 +121,11 @@ Espo.Utils = {
});
}
else if (typeof view[method] === 'function') {
if (view?.events[`click [data-action="${action}"]`]) {
// Prevents from firing if a handler is already assigned. Important.
return false;
}
view[method].call(view, data, event);
event.preventDefault();

View File

@@ -108,6 +108,7 @@ class View extends BullView {
* @param {module:view~actionHandlerCallback} handler A handler.
*/
addActionHandler(action, handler) {
// The key should be in sync with one in Utils.handleAction.
const fullAction = `click [data-action="${action}"]`;
this.events[fullAction] = e => {

View File

@@ -31,9 +31,6 @@ import Detail from 'views/email-template/record/detail';
export default class extends EditRecordView {
isWide = true
sideView = false
setup() {
super.setup();
Detail.prototype.listenToInsertField.call(this);

View File

@@ -28,11 +28,31 @@
import ArrayFieldView from 'views/fields/array';
import ForeignEnumFieldView from 'views/fields/foreign-enum';
import Helper from 'helpers/misc/foreign-field';
class ForeignArrayFieldView extends ArrayFieldView {
type = 'foreign'
/**
* @private
* @type {string}
*/
foreignEntityType
setup() {
const helper = new Helper(this);
const foreignParams = helper.getForeignParams();
for (const param in foreignParams) {
this.params[param] = foreignParams[param];
}
this.foreignEntityType = helper.getEntityType();
super.setup();
}
setupOptions() {
ForeignEnumFieldView.prototype.setupOptions.call(this);
}

View File

@@ -27,37 +27,34 @@
************************************************************************/
import ChecklistFieldView from 'views/fields/checklist';
import Helper from 'helpers/misc/foreign-field';
import ForeignArrayFieldView from 'views/fields/foreign-array';
class ForeignChecklistFieldView extends ChecklistFieldView {
type = 'foreign'
/**
* @private
* @type {string}
*/
foreignEntityType
setup() {
const helper = new Helper(this);
const foreignParams = helper.getForeignParams();
for (const param in foreignParams) {
this.params[param] = foreignParams[param];
}
this.foreignEntityType = helper.getEntityType();
super.setup();
}
setupOptions() {
this.params.options = [];
if (!this.params.field || !this.params.link) {
return;
}
const scope = this.getMetadata()
.get(['entityDefs', this.model.entityType, 'links', this.params.link, 'entity']);
if (!scope) {
return;
}
this.params.isSorted = this.getMetadata()
.get(['entityDefs', scope, 'fields', this.params.field, 'isSorted']) || false;
this.params.options = this.getMetadata()
.get(['entityDefs', scope, 'fields', this.params.field, 'options']) || [];
this.translatedOptions = {};
this.params.options.forEach(item => {
this.translatedOptions[item] = this.getLanguage()
.translateOption(item, this.params.field, scope);
});
ForeignArrayFieldView.prototype.setupOptions.call(this);
}
}

View File

@@ -27,14 +27,32 @@
************************************************************************/
import EnumFieldView from 'views/fields/enum';
import Helper from 'helpers/misc/foreign-field';
class ForeignEnumFieldView extends EnumFieldView {
type = 'foreign'
setupOptions() {
this.params.options = [];
/**
* @private
* @type {string}
*/
foreignEntityType
setup() {
const helper = new Helper(this);
const foreignParams = helper.getForeignParams();
for (const param in foreignParams) {
this.params[param] = foreignParams[param];
}
this.foreignEntityType = helper.getEntityType();
super.setup();
}
setupOptions() {
const field = this.params.field;
const link = this.params.link;
@@ -42,47 +60,15 @@ class ForeignEnumFieldView extends EnumFieldView {
return;
}
const entityType = this.getMetadata().get(`entityDefs.${this.model.entityType}.links.${link}.entity`);
if (!entityType) {
return;
}
/**
* @type {{
* optionsPath?: string|null,
* optionsReference?: string|null,
* translation?: string|null,
* options?: string[],
* isSorted?: boolean,
* displayAsLabel?: boolean,
* style?: Record,
* labelType?: string,
* }}
*/
const fieldDefs = this.getMetadata().get(`entityDefs.${entityType}.fields.${field}`);
if (!fieldDefs) {
return;
}
let {
optionsPath,
optionsReference,
translation,
options,
isSorted,
displayAsLabel,
style,
labelType,
} = fieldDefs;
let optionsPath = this.params.optionsPath;
const optionsReference = this.params.optionsReference;
let options = this.params.options;
const style = this.params.style;
if (!optionsPath && optionsReference) {
const [refEntityType, refField] = optionsReference.split('.');
optionsPath = `entityDefs.${refEntityType}.fields.${refField}.options`;
style = this.getMetadata().get(`entityDefs.${refEntityType}.fields.${refField}.style`) ?? {};
}
if (optionsPath) {
@@ -90,14 +76,10 @@ class ForeignEnumFieldView extends EnumFieldView {
}
this.params.options = Espo.Utils.clone(options) ?? [];
this.params.translation = translation;
this.params.isSorted = isSorted ?? false;
this.params.displayAsLabel = displayAsLabel ?? false;
this.params.labelType = labelType;
this.styleMap = style ?? {};
const pairs = this.params.options
.map(item => [item, this.getLanguage().translateOption(item, field, entityType)])
.map(item => [item, this.getLanguage().translateOption(item, field, this.foreignEntityType)])
this.translatedOptions = Object.fromEntries(pairs);
}

View File

@@ -28,11 +28,31 @@
import MultiEnumFieldView from 'views/fields/multi-enum';
import ForeignArrayFieldView from 'views/fields/foreign-array';
import Helper from 'helpers/misc/foreign-field';
class ForeignMultiEnumFieldView extends MultiEnumFieldView {
type = 'foreign'
/**
* @private
* @type {string}
*/
foreignEntityType
setup() {
const helper = new Helper(this);
const foreignParams = helper.getForeignParams();
for (const param in foreignParams) {
this.params[param] = foreignParams[param];
}
this.foreignEntityType = helper.getEntityType();
super.setup();
}
setupOptions() {
ForeignArrayFieldView.prototype.setupOptions.call(this);
}

View File

@@ -119,6 +119,7 @@ class MasterSiteView extends View {
}
body.dataset.isDark = this.getThemeManager().getParam('isDark') ?? false;
body.dataset.themeName = this.getThemeManager().getName();
const footerView = this.getView('footer');

View File

@@ -92,7 +92,7 @@ class UpdateNoteStreamView extends NoteStreamView {
const statusValue = data.value;
this.statusStyle = this.getMetadata()
.get(`entityDefs.${parentType}.fields.${statusField}.style.${statusValue}`) ||
.get(`entityDefs.${parentType}.fields.${statusField}.style.${statusValue}`) ||
'default';
this.statusText = this.getLanguage()
@@ -101,7 +101,7 @@ class UpdateNoteStreamView extends NoteStreamView {
this.wait(true);
this.getModelFactory().create(parentType, model => {
this.getModelFactory().create(parentType).then(model => {
const modelWas = model;
const modelBecame = model.clone();
@@ -116,7 +116,9 @@ class UpdateNoteStreamView extends NoteStreamView {
fields.forEach(field => {
const type = model.getFieldType(field) || 'base';
const viewName = this.getMetadata().get(['entityDefs', model.entityType, 'fields', field, 'view']) ||
const viewName = model.getFieldParam(field, 'auditView') ??
model.getFieldParam(field, 'view') ??
this.getFieldManager().getViewName(type);
const attributeList = this.getFieldManager().getEntityTypeFieldAttributeList(model.entityType, field);
@@ -143,24 +145,26 @@ class UpdateNoteStreamView extends NoteStreamView {
this.createView(field + 'Was', viewName, {
model: modelWas,
name: field,
readOnly: true,
defs: {
name: field
},
mode: 'detail',
inlineEditDisabled: true,
selector: `.row[data-name="${field}"] .cell-was`,
auditData: {
type: 'was',
},
});
this.createView(field + 'Became', viewName, {
model: modelBecame,
name: field,
readOnly: true,
defs: {
name: field,
},
mode: 'detail',
inlineEditDisabled: true,
selector: `.row[data-name="${field}"] .cell-became`,
auditData: {
type: 'became',
},
});
this.fieldDataList.push({
@@ -175,16 +179,17 @@ class UpdateNoteStreamView extends NoteStreamView {
});
}
toggleDetails() {
const target = this.element.querySelector('[data-action="expandDetails"]');
const $details = this.$el.find('> .details');
const $fields = this.$el.find('> .stream-details-container > .fields');
const detailsElement = this.element.querySelector(':scope > .details');
const fieldElement = this.element.querySelector(':scope > .stream-details-container > .fields');
const iconElement = target.querySelector('[data-role="icon"]');
if (!this.isExpanded) {
$details.removeClass('hidden');
$fields.addClass('hidden');
detailsElement.classList.remove('hidden');
fieldElement?.classList.add('hidden');
this.fieldList.forEach(field => {
const wasField = this.getView(field + 'Was');
@@ -196,21 +201,19 @@ class UpdateNoteStreamView extends NoteStreamView {
}
});
$(target).find('span')
.removeClass('fa-chevron-down')
.addClass('fa-chevron-up');
iconElement.classList.remove('fa-chevron-down');
iconElement.classList.add('fa-chevron-up');
this.isExpanded = true;
return;
}
$details.addClass('hidden');
$fields.removeClass('hidden');
detailsElement.classList.add('hidden');
fieldElement?.classList.remove('hidden');
$(target).find('span')
.addClass('fa-chevron-down')
.removeClass('fa-chevron-up');
iconElement.classList.remove('fa-chevron-up');
iconElement.classList.add('fa-chevron-down');
this.isExpanded = false;
}

View File

@@ -63,26 +63,26 @@ class UserGeneratePasswordFieldView extends BaseFieldView {
this.strengthParams = this.options.strengthParams || {};
this.passwordStrengthLength = this.strengthParams.passwordStrengthLength ||
this.getConfig().get('passwordStrengthLength');
this.passwordStrengthLength = this.strengthParams.passwordStrengthLength ??
this.getConfig().get('passwordStrengthLength') ?? null;
this.passwordStrengthLetterCount = this.strengthParams.passwordStrengthLetterCount ||
this.getConfig().get('passwordStrengthLetterCount');
this.passwordStrengthLetterCount = this.strengthParams.passwordStrengthLetterCount ??
this.getConfig().get('passwordStrengthLetterCount') ?? null;
this.passwordStrengthNumberCount = this.strengthParams.passwordStrengthNumberCount ||
this.getConfig().get('passwordStrengthNumberCount');
this.passwordStrengthNumberCount = this.strengthParams.passwordStrengthNumberCount ??
this.getConfig().get('passwordStrengthNumberCount') ?? null;
this.passwordStrengthSpecialCharacterCount = this.strengthParams.passwordStrengthSpecialCharacterCount ||
this.getConfig().get('passwordStrengthSpecialCharacterCount');
this.passwordStrengthSpecialCharacterCount = this.strengthParams.passwordStrengthSpecialCharacterCount ??
this.getConfig().get('passwordStrengthSpecialCharacterCount') ?? null;
this.passwordGenerateLength = this.strengthParams.passwordGenerateLength ||
this.getConfig().get('passwordGenerateLength');
this.passwordGenerateLength = this.strengthParams.passwordGenerateLength ??
this.getConfig().get('passwordGenerateLength') ?? null;
this.passwordGenerateLetterCount = this.strengthParams.passwordGenerateLetterCount ||
this.getConfig().get('passwordGenerateLetterCount');
this.passwordGenerateLetterCount = this.strengthParams.passwordGenerateLetterCount ??
this.getConfig().get('passwordGenerateLetterCount') ?? null;
this.passwordGenerateNumberCount = this.strengthParams.passwordGenerateNumberCount ||
this.getConfig().get('passwordGenerateNumberCount');
this.passwordGenerateNumberCount = this.strengthParams.passwordGenerateNumberCount ??
this.getConfig().get('passwordGenerateNumberCount') ?? null;
}
fetch() {

View File

@@ -106,6 +106,7 @@
@dropdown-link-hover-bg-value: @navbar-inverse-link-hover-bg-value;
@dropdown-divider-bg-value: @default-border-color-value;
@dropdown-border-value: @panel-default-border-value;
@dropdown-link-active-bg-value: lighten(@navbar-inverse-link-hover-bg-value, 3%);
@select-item-bg-value: @table-bg-accent-value;
@select-item-border-value: @btn-default-border-value;

View File

@@ -448,10 +448,6 @@ input.numeric-text {
font-variant-numeric: tabular-nums;
}
.filter .selectize-input {
min-height: var(--input-height-small);
}
.btn.active {
box-shadow: none;
}
@@ -642,60 +638,6 @@ input.global-search-input {
margin-right: var(--panel-padding);
}
.cell > .field {
overflow-wrap: break-word;
word-wrap: break-word;
}
.field .row {
margin-left: calc(var(--3px) * -1) !important;
margin-right: calc(var(--3px) * -1) !important;
> div {
padding-left: var(--3px) !important;
padding-right: var(--3px) !important;
}
}
.field .form-control,
.field .btn {
margin-bottom: var(--3px);
}
.field .input-group .form-control,
.field .input-group .btn {
margin-bottom: 0;
}
.field .input-group {
margin-bottom: var(--3px);
}
.field .link-container {
margin-bottom: 0;
&:not(.no-input) {
> .list-group-item:last-child {
border-bottom-width: 0;
}
border-bottom-left-radius: 0;
border-bottom-right-radius: 0;
> .list-group-item:last-child {
border-bottom-left-radius: 0;
border-bottom-right-radius: 0;
border-bottom-width: 0;
}
}
> .list-group-item {
.text:empty:before {
content: "\200b";
}
}
}
.list-group {
border-radius: var(--border-radius);
@@ -798,20 +740,6 @@ input.global-search-input {
}
}
.field {
.list-group .list-group-item {
background-color: var(--panel-bg);
}
.list-group {
background-color: var(--default-border-color);
}
}
.panel-body .field > .link-container > .list-group-item {
background-color: @panel-bg;
}
.panel-body .list-group-item {
background-color: @panel-bg;
@@ -820,214 +748,6 @@ input.global-search-input {
}
}
.field {
.link-container {
.list-group-item {
> div {
margin: calc(var(--6px) * -1) 0 calc(var(--6px) * -1);
}
> div > div {
margin: var(--6px) 0;
}
.form-control,
.btn {
margin-top: var(--2px);
margin-bottom: 0;
}
> span.text {
width: calc(100% - var(--18px));
display: inline-block;
}
> span.drag-handle {
display: inline-block;
width: var(--18px);
color: var(--text-muted-color);
cursor: grab;
vertical-align: top;
&:active {
cursor: grabbing;
}
}
> span.item-button {
display: inline-block;
width: var(--18px);
vertical-align: top;
}
&:has(> .item-button) {
> span.text {
width: calc(100% - var(--36px));
}
}
&:has(> .drag-handle) {
> span.text {
width: calc(100% - var(--36px));
}
}
&:has(> .item-button):has(> .drag-handle) {
> span.text {
width: calc(100% - var(--36px) - var(--18px));
}
}
> a[role="button"] {
margin-top: var(--1px);
margin-left: var(--2px);
}
}
.link-group-item-with-columns {
> div > .btn-group {
margin-top: 0;
margin-bottom: 0;
.caret {
border-top-color: var(--gray-light);
}
}
}
}
}
.field, .cell {
.checklist-label {
color: @text-color;
margin-bottom: 0;
padding-left: var(--7px);
}
.checklist-item-container {
margin-bottom: var(--2px);
&:last-child {
margin-bottom: 0;
}
> input[type=checkbox] {
float: left;
}
> input[type=checkbox]:not(:disabled) + label {
cursor: pointer;
}
user-select: none;
}
.multi-enum-item-label-container {
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
padding-bottom: var(--4px);
&:last-child {
padding-bottom: var(--1px);
}
.label-md {
line-height: 1.5;
}
}
}
.field {
> .label {
white-space: normal;
}
}
.filter > .form-group .field {
.link-container {
font-size: var(--font-size-small);
.fa-times {
font-size: var(--12px);
}
> .list-group-item {
padding-top: var(--4px);
padding-bottom: var(--4px);
line-height: var(--20px);
}
}
.input-group {
input {
font-size: var(--font-size-small);
height: var(--input-height-small);
min-height: var(--input-height-small);
}
button.btn {
font-size: var(--font-size-small);
height: var(--input-height-small);
}
}
}
.link-container {
> .list-group-item {
.link-item-column {
float: right;
display: inline-block;
width: 40%;
> .selectize-control {
width: 100%;
}
}
> a.pull-right {
margin-left: var(--4px);
}
&,
> .text {
> img.avatar {
margin-right: var(--7px);
top: var(--2px);
}
}
.link-item-name {
> img.avatar {
margin-right: var(--7px);
top: var(--2px);
}
}
}
}
.filter .field .link-container .list-group-item a[role="button"] {
margin-top: 0;
margin-left: var(--2px);
}
.field .link-container > .list-group-item.link-with-role > a[role="button"],
.field .link-container > .list-group-item.link-with-role > div > a[role="button"] {
margin-top: var(--7px);
}
.field .link-container > .list-group-item.link-group-item-with-columns,
.field .link-container > .list-group-item.link-group-item-with-primary {
> div:nth-child(1) {
display: inline-block;
width: calc(~"100% - 23px");
float: left;
input.form-control {
width: 100%;
}
}
> div:nth-child(2) {
display: inline-block;
width: var(--23px);
float: right;
vertical-align: top;
}
}
#main > .list-container {
> .no-data {
@@ -1249,14 +969,6 @@ input.global-search-input {
}
}
.filter a.remove-filter {
display: none;
}
.filter:hover a.remove-filter {
display: block;
}
optgroup {
font-weight: 600;
}
@@ -3415,7 +3127,7 @@ table.table td.cell .html-container {
.post-container {
.internal-mode-switcher {
color: var(--text-muted-color);
margin-left: var(--10px);
margin-left: var(--14px);
display: inline-block;
position: relative;
top: var(--2px);
@@ -4557,3 +4269,17 @@ body > .autocomplete-suggestions.text-search-suggestions {
@import "elements/popup-notification.less";
@import "elements/grid.less";
@import "elements/animation.less";
.compact-form {
--input-height-base: var(--input-height-small);
--line-height-computed: var(--line-height-small);
--line-height-base: var(--line-height-small);
--font-size-base: var(--font-size-small);
--padding-base-vertical: var(--padding-small-vertical);
--padding-base-horizontal: var(--padding-small-horizontal);
--icon-size-base: var(--icon-size-small);
--icon-line-height-base: var(--icon-line-height-small);
--btn-icon-width: var(--btn-icon-width-small);
}

View File

@@ -56,10 +56,6 @@ a.btn {
}
}
.panel-heading .btn-sm {
height: auto;
}
.input-group {
.input-group-btn > {
.btn {
@@ -102,7 +98,7 @@ a.btn {
}
.btn-icon {
width: var(--36px);
width: var(--btn-icon-width);
padding-left: 0;
padding-right: 0;
@@ -121,7 +117,7 @@ a.btn {
}
.btn-icon.btn-sm {
width: var(--34px);
width: var(--btn-icon-width-small);
.fa, .fas {
font-size: var(--12px);

View File

@@ -100,7 +100,7 @@ ul.dropdown-menu {
line-height: var(--line-height-computed);
&.active {
background-color: @gray-lighter;
background-color: var(--dropdown-link-active-bg);
text-decoration: none;
outline: 0;
}

View File

@@ -124,21 +124,6 @@ input.form-control {
}
}
.link-container:not(:empty) + div,
.link-container:not(:empty) + div > div.input-group {
> .form-control,
> .btn,
> .input-group-btn > .btn {
border-top-left-radius: 0 !important;
border-top-right-radius: 0 !important;
}
}
.link-container:not(:empty) + .form-control {
border-top-left-radius: 0 !important;
border-top-right-radius: 0 !important;
}
.input-group > .input-group-btn:first-child > select.form-control:first-child {
border-top-left-radius: var(--border-radius);
border-bottom-left-radius: var(--border-radius);
@@ -510,6 +495,8 @@ input[type="radio"].form-radio {
.control-label {
user-select: none;
font-size: var(--font-size-base);
}
.form-group.hidden-cell {
@@ -663,46 +650,316 @@ select.form-control.native-select {
}
}
/*
.link-container {
a[data-action="clearLink"] {
color: var(--gray-soft);
&:hover {
color: var(--btn-text-color);
.field {
.link-container {
.list-group-item {
> div {
margin: calc(var(--6px) * -1) 0 calc(var(--6px) * -1);
}
> div > div {
margin: var(--6px) 0;
}
.form-control,
.btn {
margin-top: var(--2px);
margin-bottom: 0;
}
> span.text {
width: calc(100% - var(--18px));
display: inline-block;
}
> span.drag-handle {
display: inline-block;
width: var(--18px);
color: var(--text-muted-color);
cursor: grab;
vertical-align: top;
&:active {
cursor: grabbing;
}
}
> span.item-button {
display: inline-block;
width: var(--18px);
vertical-align: top;
}
&:has(> .item-button) {
> span.text {
width: calc(100% - var(--36px));
}
}
&:has(> .drag-handle) {
> span.text {
width: calc(100% - var(--36px));
}
}
&:has(> .item-button):has(> .drag-handle) {
> span.text {
width: calc(100% - var(--36px) - var(--18px));
}
}
> a[role="button"] {
margin-top: var(--1px);
margin-left: var(--2px);
}
}
.link-group-item-with-columns {
> div > .btn-group {
margin-top: 0;
margin-bottom: 0;
.caret {
border-top-color: var(--gray-light);
}
}
}
> .list-group-item.link-with-role {
> a[role="button"],
> div > a[role="button"] {
margin-top: var(--7px);
}
}
> .list-group-item.link-group-item-with-columns,
> .list-group-item.link-group-item-with-primary {
> div:nth-child(1) {
display: inline-block;
width: calc(~"100% - 23px");
float: left;
input.form-control {
width: 100%;
}
}
> div:nth-child(2) {
display: inline-block;
width: var(--23px);
float: right;
vertical-align: top;
}
}
}
> .label {
white-space: normal;
}
.list-group .list-group-item {
background-color: var(--panel-bg);
}
.list-group {
background-color: var(--default-border-color);
}
}
.field,
.cell {
.checklist-label {
color: var(--text-color);
margin-bottom: 0;
padding-left: var(--7px);
}
.checklist-item-container {
margin-bottom: var(--2px);
&:last-child {
margin-bottom: 0;
}
> input[type=checkbox] {
float: left;
}
> input[type=checkbox]:not(:disabled) + label {
cursor: pointer;
}
user-select: none;
}
.multi-enum-item-label-container {
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
padding-bottom: var(--4px);
&:last-child {
padding-bottom: var(--1px);
}
.label-md {
line-height: 1.5;
}
}
}
.panel-body {
.field {
> .link-container {
> .list-group-item {
background-color: var(--panel-bg);
}
}
}
}
.filter {
> .form-group .field {
.link-container {
font-size: var(--font-size-small);
.fa-times {
font-size: var(--12px);
}
> .list-group-item {
padding-top: var(--4px);
padding-bottom: var(--4px);
line-height: var(--20px);
}
}
.input-group {
input {
font-size: var(--font-size-small);
height: var(--input-height-small);
min-height: var(--input-height-small);
}
button.btn {
font-size: var(--font-size-small);
height: var(--input-height-small);
}
}
}
.link-container {
.list-group-item a[role="button"] {
margin-top: 0;
margin-left: var(--2px);
}
}
.selectize-input {
min-height: var(--input-height-small);
}
}
.link-container {
> .list-group-item {
a[role="button"] {
color: var(--gray-soft);
&:hover {
color: var(--btn-text-color);
.link-item-column {
float: right;
display: inline-block;
width: 40%;
> .selectize-control {
width: 100%;
}
}
> a.pull-right {
margin-left: var(--4px);
}
&,
> .text {
> img.avatar {
margin-right: var(--7px);
top: var(--2px);
}
}
.link-item-name {
> img.avatar {
margin-right: var(--7px);
top: var(--2px);
}
}
}
&:not(:empty) {
+ div,
+ div > div.input-group {
> .form-control,
> .btn,
> .input-group-btn > .btn {
border-top-left-radius: 0 !important;
border-top-right-radius: 0 !important;
}
}
+ .form-control {
border-top-left-radius: 0 !important;
border-top-right-radius: 0 !important;
}
}
}
.cell > .field {
overflow-wrap: break-word;
word-wrap: break-word;
}
.field {
.add-item-container {
a.add-item {
color: var(--gray-soft);
&:hover {
color: var(--btn-text-color);
}
.row {
margin-left: calc(var(--3px) * -1) !important;
margin-right: calc(var(--3px) * -1) !important;
> div {
padding-left: var(--3px) !important;
padding-right: var(--3px) !important;
}
}
.item-list {
.item {
a.remove-item {
color: var(--gray-soft);
&:hover {
color: var(--btn-text-color);
}
.form-control,
.btn {
margin-bottom: var(--3px);
}
.input-group .form-control,
.input-group .btn {
margin-bottom: 0;
}
.input-group {
margin-bottom: var(--3px);
}
.link-container {
margin-bottom: 0;
&:not(.no-input) {
> .list-group-item:last-child {
border-bottom-width: 0;
}
border-bottom-left-radius: 0;
border-bottom-right-radius: 0;
> .list-group-item:last-child {
border-bottom-left-radius: 0;
border-bottom-right-radius: 0;
border-bottom-width: 0;
}
}
> .list-group-item {
.text:empty:before {
content: "\200b";
}
}
}
}
*/

View File

@@ -1,8 +1,8 @@
.fas,
.far,
.fa {
font-size: var(--16px);
line-height: var(--21px);
font-size: var(--icon-size-base);
line-height: var(--icon-line-height-base);
}
.btn {
@@ -11,7 +11,7 @@
line-height: var(--line-height-computed);
&.fa-sm {
line-height: var(--18px);
line-height: var(--icon-line-height-small);
}
}
}
@@ -24,14 +24,8 @@
.small .fas,
.fas.small,
.far.small {
font-size: var(--12px);
line-height: var(--18px);
}
.btn:not(.btn-sm) {
> .fas.fa-ellipsis-h {
//line-height: var(--23px);
}
font-size: var(--icon-size-small);
line-height: var(--icon-line-height-small);
}
.btn:not(.btn-sm),

View File

@@ -271,7 +271,7 @@
@media screen and (min-width: @screen-md-min) {
.modal-dialog {
width: ~"min(var(--900px), 90vw)";
width: ~"min(calc(var(--900px) + var(--30px)), 90vw)";
}
.dialog-confirm > .modal-dialog,
@@ -348,8 +348,12 @@
}
.modal.dialog-record {
.modal-body {
scrollbar-gutter: stable;
.modal {
&.dialog-record,
&:has(> .modal-dialog > .modal-content > .modal-body > .record.no-side-margin),
&:has(> .modal-dialog > .modal-content > .modal-body > .record-container.no-side-margin) {
.modal-body {
scrollbar-gutter: stable;
}
}
}

View File

@@ -618,6 +618,10 @@ body {
}
}
.panel-heading .btn-sm {
height: auto;
}
.panel.highlighted {
outline: var(--2px) solid var(--brand-info);
outline-offset: var(--minus-1px);

View File

@@ -13,7 +13,7 @@
}
&.record-grid-small {
grid-template-columns: minmax(auto, 60%) minmax(auto, 40%);
grid-template-columns: minmax(auto, 61%) minmax(auto, 39%);
}
max-width: var(--record-grid-max-width);
@@ -35,6 +35,38 @@
.record .record-grid {
grid-column-gap: var(--padding-base-horizontal);
}
@media screen and (min-width: @screen-sm-min) {
.record,
.record-container,
.edit-container {
&.no-side-margin {
> div {
> .record-grid-wide {
margin-right: var(--panel-padding);
> .left {
> .middle {
> .panel {
&:not(.middle):not(.last),
&.first {
border-top-right-radius: var(--panel-border-radius);
}
&:not(.middle):not(.first),
&.last {
border-bottom-right-radius: var(--panel-border-radius);
}
}
}
}
}
}
}
}
}
}
@media screen and (max-width: @screen-sm-max) {

View File

@@ -152,7 +152,7 @@ label {
.label-md {
font-weight: normal;
font-size: 100%;
font-size: var(--font-size-base);
padding: var(--1px) var(--7px) var(--2px);
top: 0;
line-height: 1.8;
@@ -270,6 +270,13 @@ a.text-default {
}
}
a.no-underline {
&:hover,
&:focus {
text-decoration: none;
}
}
.nowrap {
white-space: nowrap;
}

View File

@@ -481,6 +481,7 @@
--dropdown-border: @dropdown-border-value;
--dropdown-divider-bg: @dropdown-divider-bg-value;
--dropdown-box-shadow: @dropdown-box-shadow-value;
--dropdown-link-active-bg: @dropdown-link-active-bg-value;
--calendar-today-bg: @calendar-today-bg-value;
--calendar-border: @calendar-border-value;
@@ -510,4 +511,15 @@
--top-bar-box-shadow: @top-bar-box-shadow-value;
--vertical-gap: @vertical-gap-value;
//
--icon-size-base: var(--16px);
--icon-line-height-base: var(--21px);
--icon-size-small: var(--12px);
--icon-line-height-small: var(--18px);
--btn-icon-width: var(--36px);
--btn-icon-width-small: var(--32px);
}

View File

@@ -208,6 +208,7 @@
@dropdown-border-width-value: var(--1px);
@dropdown-divider-bg-value: #e5e5e5;
@dropdown-box-shadow-value: 0 4px 6px rgba(0, 0, 0, 0.11);
@dropdown-link-active-bg-value: @gray-lighter-value;
@login-panel-heading-bg-value: @panel-bg-value;

View File

@@ -123,6 +123,7 @@
@dropdown-divider-bg-value: @default-border-color-value;
@dropdown-border-value: transparent;
@dropdown-border-width-value: 0;
@dropdown-link-active-bg-value: #4D4950C7;
@select-item-bg-value: #1e253054;
@select-item-border-value: @btn-default-border-value;

View File

@@ -1,12 +1,12 @@
@body-bg-value: #d3dee3;
@body-bg-value: #e1e9ed;
@white-color-value: #FFF;
@panel-bg-value: @white-color-value;
@link-color-value: #5b8fc8;
@navbar-inverse-bg-value: @body-bg-value;
@navbar-inverse-link-active-bg-value: #bbc6cb;
@navbar-inverse-link-hover-bg-value: #c8d4d9;
@navbar-inverse-bg-value: #edf1f3;
@navbar-inverse-link-active-bg-value: #d9e0e3;
@navbar-inverse-link-hover-bg-value: #e5e9eb;
@navbar-inverse-color-value: #8d8f93;
@navbar-box-shadow-value: @default-box-shadow-value;

4
package-lock.json generated
View File

@@ -1,12 +1,12 @@
{
"name": "espocrm",
"version": "9.2.2",
"version": "9.2.3",
"lockfileVersion": 2,
"requires": true,
"packages": {
"": {
"name": "espocrm",
"version": "9.2.2",
"version": "9.2.3",
"hasInstallScript": true,
"license": "AGPL-3.0-or-later",
"dependencies": {

View File

@@ -1,6 +1,6 @@
{
"name": "espocrm",
"version": "9.2.2",
"version": "9.2.3",
"description": "Open-source CRM.",
"repository": {
"type": "git",

View File

@@ -993,6 +993,26 @@
},
"description": "Parameters available in the Entity Manager tool when editing the field. Specify an empty array to disable all parameters."
},
"fieldManagerAdditionalParamList": {
"type": "array",
"items": {
"allOf": [
{
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "A parameter name."
}
}
},
{
"$ref": "entityDefs.json#/definitions/fieldDefs"
}
]
},
"description": "Additional field parameters available in the Entity Manager."
},
"layoutAvailabilityList": {
"type": "array",
"items": {
@@ -1259,6 +1279,10 @@
"type": "boolean",
"description": "Linking and unlinking will be logged in the Stream. Available for hasMany."
},
"auditView": {
"type": "string",
"description": "A view used for audit. If not specified, the ordinary view is used. As of v9.2.3."
},
"readOnly": {
"type": "boolean",
"description": "Read-only links cannot be edited via link and unlink requests. But they can be edited via link and link-multiple fields."