Files
crm.clientright.ru/layouts/v7/modules/Quotes/DetailViewHeaderTitle.tpl
Fedor 9245768987 🚀 CRM Files Migration & Real-time Features
 Features:
- Migrated ALL files to new S3 structure (Projects, Contacts, Accounts, HelpDesk, Invoice, etc.)
- Added Nextcloud folder buttons to ALL modules
- Fixed Nextcloud editor integration
- WebSocket server for real-time updates
- Redis Pub/Sub integration
- File path manager for organized storage
- Redis caching for performance (Functions.php)

📁 New Structure:
Documents/Project/ProjectName_ID/file_docID.ext
Documents/Contacts/FirstName_LastName_ID/file_docID.ext
Documents/Accounts/AccountName_ID/file_docID.ext

🔧 Technical:
- FilePathManager for standardized paths
- S3StorageService integration
- WebSocket server (Node.js + Docker)
- Redis cache for getBasicModuleInfo()
- Predis library for Redis connectivity

📝 Scripts:
- Migration scripts for all modules
- Test pages for WebSocket/SSE/Polling
- Documentation (MIGRATION_*.md, REDIS_*.md)

🎯 Result: 15,000+ files migrated successfully!
2025-10-24 19:59:28 +03:00

92 lines
6.4 KiB
Smarty

{*<!--
/*********************************************************************************
** The contents of this file are subject to the vtiger CRM Public License Version 1.0
* ("License"); You may not use this file except in compliance with the License
* The Original Code is: vtiger CRM Open Source
* The Initial Developer of the Original Code is vtiger.
* Portions created by vtiger are Copyright (C) vtiger.
* All Rights Reserved.
*
********************************************************************************/
-->*}
{strip}
<div class="col-sm-6">
<div class="record-header clearfix">
<div class="hidden-sm hidden-xs recordImage bgquotes app-{$SELECTED_MENU_CATEGORY}">
{assign var=IMAGE_DETAILS value=$RECORD->getImageDetails()}
{foreach key=ITER item=IMAGE_INFO from=$IMAGE_DETAILS}
{if !empty($IMAGE_INFO.url)}
<img src="{$IMAGE_INFO.url}" alt="{$IMAGE_INFO.orgname}" title="{$IMAGE_INFO.orgname}" width="100%" height="100%" align="left"><br>
{else}
<img src="{vimage_path('summary_organizations.png')}" class="summaryImg"/>
{/if}
{/foreach}
{if empty($IMAGE_DETAILS)}
<div class="name"><span><strong>{$MODULE_MODEL->getModuleIcon()}</strong></span></div>
{/if}
</div>
<div class="recordBasicInfo">
<div class="info-row">
<h4>
<span class="recordLabel pushDown" title="{$RECORD->getName()}">
{foreach item=NAME_FIELD from=$MODULE_MODEL->getNameFields()}
{assign var=FIELD_MODEL value=$MODULE_MODEL->getField($NAME_FIELD)}
{if $FIELD_MODEL->getPermissions()}
<span class="{$NAME_FIELD}">{trim($RECORD->get($NAME_FIELD))}</span>&nbsp;
{/if}
{/foreach}
</span>
</h4>
</div>
{include file="DetailViewHeaderFieldsView.tpl"|vtemplate_path:$MODULE}
{* Кнопка открытия папки предложения в Nextcloud *}
<div class="info-row" style="margin-top: 10px;">
<button type="button" class="btn btn-info btn-sm" onclick="openRecordFolder('Quotes', '{$RECORD->getId()}', '{$RECORD->get("quote_no")|escape:javascript}')" title="Открыть папку предложения в Nextcloud">
<i class="fa fa-folder-open"></i> Папка в Nextcloud
</button>
</div>
{* Подключаем Nextcloud Editor JS *}
<script type="text/javascript" src="crm_extensions/nextcloud_editor/js/nextcloud-editor.js"></script>
{*
<div class="row info-row">
{assign var=FIELD_MODEL value=$MODULE_MODEL->getField('account_id')}
<div class="col-lg-7 fieldLabel">
<span class="value textOverflowEllipsis" title="{vtranslate($FIELD_MODEL->get('label'),$MODULE)} : {strip_tags($FIELD_MODEL->getDisplayValue($FIELD_MODEL->get('fieldvalue'), $RECORD->getId(), $RECORD))}" {if $FIELD_MODEL->get('uitype') eq '19' or $FIELD_MODEL->get('uitype') eq '20' or $FIELD_MODEL->get('uitype') eq '21'}style="word-wrap: break-word;"{/if}>
{include file=$FIELD_MODEL->getUITypeModel()->getDetailViewTemplateName()|@vtemplate_path FIELD_MODEL=$FIELD_MODEL USER_MODEL=$USER_MODEL MODULE=$MODULE_NAME RECORD=$RECORD}
</span>
</div>
</div>
<div class="row info-row">
{assign var=FIELD_MODEL value=$MODULE_MODEL->getField('contact_id')}
<div class="col-lg-7 fieldLabel">
<span class="value textOverflowEllipsis" title="{vtranslate($FIELD_MODEL->get('label'),$MODULE)} : {strip_tags($FIELD_MODEL->getDisplayValue($FIELD_MODEL->get('fieldvalue'), $RECORD->getId(), $RECORD))}" {if $FIELD_MODEL->get('uitype') eq '19' or $FIELD_MODEL->get('uitype') eq '20' or $FIELD_MODEL->get('uitype') eq '21'}style="word-wrap: break-word;"{/if}>
{include file=$FIELD_MODEL->getUITypeModel()->getDetailViewTemplateName()|@vtemplate_path FIELD_MODEL=$FIELD_MODEL USER_MODEL=$USER_MODEL MODULE=$MODULE_NAME RECORD=$RECORD}
</span>
</div>
</div>
<div class="row info-row">
{assign var=FIELD_MODEL value=$MODULE_MODEL->getField('hdnGrandTotal')}
<div class="col-lg-7 fieldLabel">
<span class="value textOverflowEllipsis" title="{vtranslate($FIELD_MODEL->get('label'),$MODULE)} : {strip_tags($FIELD_MODEL->getDisplayValue($FIELD_MODEL->get('fieldvalue'), $RECORD->getId(), $RECORD))}" {if $FIELD_MODEL->get('uitype') eq '19' or $FIELD_MODEL->get('uitype') eq '20' or $FIELD_MODEL->get('uitype') eq '21'}style="word-wrap: break-word;"{/if}>
{include file=$FIELD_MODEL->getUITypeModel()->getDetailViewTemplateName()|@vtemplate_path FIELD_MODEL=$FIELD_MODEL USER_MODEL=$USER_MODEL MODULE=$MODULE_NAME RECORD=$RECORD}
</span>
</div>
</div>
<div class="row info-row">
{assign var=FIELD_MODEL value=$MODULE_MODEL->getField('quotestage')}
<div class="col-lg-5">{vtranslate($FIELD_MODEL->get('label'),$MODULE)}</div>
<div class="col-lg-7 fieldLabel">
<span class="value textOverflowEllipsis" title="{vtranslate($FIELD_MODEL->get('label'),$MODULE)} : {strip_tags($FIELD_MODEL->getDisplayValue($FIELD_MODEL->get('fieldvalue'), $RECORD->getId(), $RECORD))}" {if $FIELD_MODEL->get('uitype') eq '19' or $FIELD_MODEL->get('uitype') eq '20' or $FIELD_MODEL->get('uitype') eq '21'}style="word-wrap: break-word;"{/if}>
{include file=$FIELD_MODEL->getUITypeModel()->getDetailViewTemplateName()|@vtemplate_path FIELD_MODEL=$FIELD_MODEL USER_MODEL=$USER_MODEL MODULE=$MODULE_NAME RECORD=$RECORD}
</span>
</div>
</div>
*}
</div>
</div>
</div>
{/strip}