Новости Joomla

Перевод и публикация интервью с Joomla евангелистом на греческом портале Joomla

Перевод и публикация интервью на греческом портале Joomla 🇬🇷

Утро, просматриваешь входящие письма и изучаешь новости и внезапно обнаруживаешь, что инициатива, которую ты начал, подхватывается другими людьми. 🎉

Недавно я взял интервью у Билла (Василиса) Коциаса - руководителя студии, читающего лекции в университете и популяризатора Joomla в Греции. Это интервью из журнала NorrNext, в оригинале на английском, теперь доступно на греческом языке и опубликовано на портале joomla.gr. 🎉

До чего же приятно… 😇😊 Работа замечена и с ней посчитали необходимым ознакомить аудиторию страны, в которой Билл читает лекции. И это солнечная Греция - страна, страна, с которой Россию многое связывает. 🇬🇷🇷🇺🕊

Смотрю на греческий алфавит и тут же рисуются картины белоснежных зданий в окружении винограда и амфор, красивых женщин в сандалиях и мужественных воинов, охраняющих покой полисов, в которых ученые мужи работают над трудами, позже вошедшими в века. Красиво! 😇Но вернемся к интервью.

Из него вы узнаете, что в Греции доля Joomla среди CMS занимает порядка 30-40%. По моему мнению это - самый высокий показатель во всем мире. Также чтение лекций о Joomla в университетах позволит привести новых пользователей и к тому же молодое поколение. Ну и огромное кол-во сертификтатов Билла на стене (смотрим фото в статье) свидетельствует о том, что Joomla может применяться как профессиональный инструмент.

🌐 Оригинальное интервью (на английском)
🇬🇷 Интервью на греческом портале (joomla.gr)

Что насчет перевода на русский? Увы, времени всего 24 часа в сутках. Я продолжаю готовить новые интервью. Возможно, после завершения выпуска журнала, рассмотрю перевод некоторых интервью на русский. Но я об этом не говорил. 😊 В блоге @eugenius_blog публикую анонсы интересных событий из мира Joomla, интервью, уроки и полезные советы, а также делюсь мыслями:, связанными с разработкой и веб-дизайном.

Обработка HTTP ответа в Joomla 6+. Изменения по сравнению с Joomla 3 - Joomla 5

👩‍💻 Обработка HTTP ответа в Joomla 6+. Изменения по сравнению с Joomla 3 - Joomla 5.В Joomla для выполнения внешних запросов из PHP к сторонним API используется класс Joomla\Http\Http напрямую или же Joomla\Http\HttpFactory, который возвращает для работы преднастроенный по умолчанию класс Http. О работе с HTTP-запросами подробно рассказывалось в статье 2021 года Создание внешних запросов с использованием HttpFactory (Joomla). Некоторые изменения касаются работы с ответами на запросы. Например, наш запрос:
use Joomla\Http\HttpFactory;$http = (new HttpFactory)->getHttp($options, ['curl', 'stream']);$response = $http->get('https://any-url.ru/api/any/endpoint');
Раньше можно было получить код ответа или тело ответа как свойство $response - $response->code или $response->body. Однако, Joomla, начиная с Joomla 4 во многом переходит на стандарты PSR. В частности для работы с HTTP-ответами - на PSR-7. Также хорошая статья на Хабре о PSR-7: PSR-7 в примерах.
Прямое обращение к свойствам code, headers, body объявлено устаревшим в Joomla 6.0.0 и обещают удалить в Joomla 7.0.0.
Вместо этого нужно работать с HTTP-ответом по стандартам PSR-7. Код ответа.Было $response->code. Стало $response->getStatusCode().Заголовки ответа.Было $response->headers. Стало $response->getHeaders().Тело ответа.Было $response->body. Стало (string)$response->getContents().В тело ответа теперь приходит не строка, а поток - объект класса Laminas\Diactoros\Stream. Поэтому его нужно привести к строке (если это json, к примеру): (string)$response->getContents(). Чаще всего в коде Joomla встречается именно такой вариант. Однако, есть и вариант с перемещением указателя чтения на начало потока:
// Получили ответ в виде потока$stream = $response->getBody();// "перемотали" на начало$stream->rewind();// Получили строковый ответ$json = $stream->getContents();
В итоге результат одинаковый.@joomlafeed#joomla #разработка #php

0 Пользователей и 1 Гость просматривают эту тему.
  • 1 Ответов
  • 3795 Просмотров
*

Giacomo

  • Осваиваюсь на форуме
  • 10
  • 0 / 0
  • Знатный Джумлавед
 !  Помогите пожалуйста, совсем запутался, я новичок в этой Joomla.
Проблема такая:
создаю пункт меню, тип "Стандартный шаблон материала", заполняю заголовок, нажимаю "Выберите материал" и открывается список материалов в новой вкладке браузера. Нажимаю на материал - и ничего не происходит. Не могу добавить ссылку на материал в меню.
Насколько я понял, тут должно появляться модальное окошко, а при нажатии оно должно закрываться и передавать адрес статьи.
Модальные окна не работают вообще. Даже в параметрах менеджера материалов.

Что я делал:

- переключал шаблон на milkyway
- отключал некоторые плагины и модули типа 2jTabs и Simple Image Gallery
- в настройках сайта отключал и снова включал flash loader
- пробовал добавлять пункты в других меню и изменять те, что я сам раньше добавлял (тогда всё работало, сейчас ни в какую не хочет)
- переключался с JCE на стандартный тини
- пробовал в Мозилле, ИЕ, Опере
- отключал касперского

Кстати, в JCE модальные окна работают.

Работаю на свежем Денвере. У кого какие соображения?
« Последнее редактирование: 24.07.2010, 15:00:55 от Giacomo »
*

Giacomo

  • Осваиваюсь на форуме
  • 10
  • 0 / 0
  • Знатный Джумлавед
Re: Невозможно выбрать материал
« Ответ #1 : 24.07.2010, 14:58:59 »
Решение найдено.
Проблема неработоспособности Модальных окон (всплывающих) в Joomla заключается в файле:
media/system/js/modal.js
Один из ранее установленных компонентов (предположительно, галерея) подменил этот файл своим.

Правильный modal.js выглядит так:

Код
/**
 * SqueezeBox - Expandable Lightbox
 *
 * Allows to open various content as modal,
 * centered and animated box.
 *
 * Inspired by
 *  ... Lokesh Dhakar - The original Lightbox v2
 *  ... Cody Lindley - ThickBox
 *
 * @version 1.0rc1
 *
 * @license MIT-style license
 * @author Harald Kirschner <mail [at] digitarald.de>
 * @copyright Author
 */
var SqueezeBox = {

presets: {
size: {x: 600, y: 450},
sizeLoading: {x: 200, y: 150},
marginInner: {x: 20, y: 20},
marginImage: {x: 150, y: 200},
handler: false,
adopt: null,
closeWithOverlay: true,
zIndex: 65555,
overlayOpacity: 0.7,
classWindow: '',
classOverlay: '',
disableFx: false,
onOpen: Class.empty,
onClose: Class.empty,
onUpdate: Class.empty,
onResize: Class.empty,
onMove: Class.empty,
onShow: Class.empty,
onHide: Class.empty,
fxOverlayDuration: 250,
fxResizeDuration: 750,
fxContentDuration: 250,
ajaxOptions: {}
},

initialize: function(options) {
if (this.options) return this;
this.presets = $merge(this.presets, options)
this.setOptions(this.presets);
this.build();
this.listeners = {
window: this.reposition.bind(this, [null]),
close: this.close.bind(this),
key: this.onkeypress.bind(this)};
this.isOpen = this.isLoading = false;
this.window.close = this.listeners.close;
return this;
},

build: function() {
this.overlay = new Element('div', {
id: 'sbox-overlay',
styles: {
display: 'none',
zIndex: this.options.zIndex
}
});
this.content = new Element('div', {
id: 'sbox-content'
});
this.btnClose = new Element('a', {
id: 'sbox-btn-close',
href: '#'
});
this.window = new Element('div', {
id: 'sbox-window',
styles: {
display: 'none',
zIndex: this.options.zIndex + 2
}
}).adopt(this.btnClose, this.content);

if (!window.ie6) {
this.overlay.setStyles({
position: 'fixed',
top: 0,
left: 0
});
this.window.setStyles({
position: 'fixed',
top: '50%',
left: '50%'
});
} else {
this.overlay.style.setExpression('marginTop', 'document.documentElement.scrollTop + "px"');
this.window.style.setExpression('marginTop', '0 - parseInt(this.offsetHeight / 2) + document.documentElement.scrollTop + "px"');

this.overlay.setStyles({
position: 'absolute',
top: '0%',
left: '0%'
//,marginTop: "expression(document.documentElement.scrollTop + 'px')"
});

this.window.setStyles({
position: 'absolute',
top: '0%',
left: '0%'
//,marginTop: "(expression(0 - parseInt(this.offsetHeight / 2) + document.documentElement.scrollTop + 'px')"
});
}

$(document.body).adopt(this.overlay, this.window);

this.fx = {
overlay: this.overlay.effect('opacity', {
duration: this.options.fxOverlayDuration,
wait: false}).set(0),
window: this.window.effects({
duration: this.options.fxResizeDuration,
wait: false}),
content: this.content.effect('opacity', {
duration: this.options.fxContentDuration,
wait: false}).set(0)
};
},

addClick: function(el) {
return el.addEvent('click', function() {
if (this.fromElement(el)) return false;
}.bind(this));
},

fromElement: function(el, options) {
this.initialize();
this.element = $(el);
if (this.element && this.element.rel) options = $merge(options || {}, Json.evaluate(this.element.rel));
this.setOptions(this.presets, options);
this.assignOptions();
this.url = (this.element ? (this.options.url || this.element.href) : el) || '';

if (this.options.handler) {
var handler = this.options.handler;
return this.setContent(handler, this.parsers[handler].call(this, true));
}
var res = false;
for (var key in this.parsers) {
if ((res = this.parsers[key].call(this))) return this.setContent(key, res);
}
return this;
},

assignOptions: function() {
this.overlay.setProperty('class', this.options.classOverlay);
this.window.setProperty('class', this.options.classWindow);
},

close: function(e) {
if (e) new Event(e).stop();
if (!this.isOpen) return this;
this.fx.overlay.start(0).chain(this.toggleOverlay.bind(this));
this.window.setStyle('display', 'none');
this.trashImage();
this.toggleListeners();
this.isOpen = null;
this.fireEvent('onClose', [this.content]).removeEvents();
this.options = {};
this.setOptions(this.presets).callChain();
return this;
},

onError: function() {
if (this.image) this.trashImage();
this.setContent('Error during loading');
},

trashImage: function() {
if (this.image) this.image = this.image.onload = this.image.onerror = this.image.onabort = null;
},

setContent: function(handler, content) {
this.content.setProperty('class', 'sbox-content-' + handler);
this.applyTimer = this.applyContent.delay(this.fx.overlay.options.duration, this, [this.handlers[handler].call(this, content)]);
if (this.overlay.opacity) return this;
this.toggleOverlay(true);
this.fx.overlay.start(this.options.overlayOpacity);
this.reposition();
return this;
},

applyContent: function(content, size) {
this.applyTimer = $clear(this.applyTimer);
this.hideContent();
if (!content) this.toggleLoading(true);
else {
if (this.isLoading) this.toggleLoading(false);
this.fireEvent('onUpdate', [this.content], 20);
}
this.content.empty()[['string', 'array', false].contains($type(content))? 'setHTML' : 'adopt'](content || '');
this.callChain();
if (!this.isOpen) {
this.toggleListeners(true);
this.resize(size, true);
this.isOpen = true;
this.fireEvent('onOpen', [this.content]);
} else this.resize(size);
},

resize: function(size, instantly) {
var sizes = window.getSize();
this.size = $merge(this.isLoading ? this.options.sizeLoading : this.options.size, size);
var to = {
width: this.size.x,
height: this.size.y,
marginLeft: - this.size.x / 2,
marginTop: - this.size.y / 2
//left: (sizes.scroll.x + (sizes.size.x - this.size.x - this.options.marginInner.x) / 2).toInt(),
//top: (sizes.scroll.y + (sizes.size.y - this.size.y - this.options.marginInner.y) / 2).toInt()
};
$clear(this.showTimer || null);
this.hideContent();
if (!instantly) this.fx.window.start(to).chain(this.showContent.bind(this));
else {
this.window.setStyles(to).setStyle('display', '');
this.showTimer = this.showContent.delay(50, this);
}
this.reposition(sizes);
},

toggleListeners: function(state) {
var task = state ? 'addEvent' : 'removeEvent';
this.btnClose[task]('click', this.listeners.close);
if (this.options.closeWithOverlay) this.overlay[task]('click', this.listeners.close);
document[task]('keydown', this.listeners.key);
window[task]('resize', this.listeners.window);
window[task]('scroll', this.listeners.window);
},

toggleLoading: function(state) {
this.isLoading = state;
this.window[state ? 'addClass' : 'removeClass']('sbox-loading');
if (state) this.fireEvent('onLoading', [this.window]);
},

toggleOverlay: function(state) {
this.overlay.setStyle('display', state ? '' : 'none');
$(document.body)[state ? 'addClass' : 'removeClass']('body-overlayed');
},

showContent: function() {
if (this.content.opacity) this.fireEvent('onShow', [this.window]);
this.fx.content.start(1);
},

hideContent: function() {
if (!this.content.opacity) this.fireEvent('onHide', [this.window]);
this.fx.content.stop().set(0);
},

onkeypress: function(e) {
switch (e.key) {
case 'esc':
case 'x':
this.close();
break;
}
},

reposition: function(sizes) {
sizes = sizes || window.getSize();
this.overlay.setStyles({
//'left': sizes.scroll.x, 'top': sizes.scroll.y,
width: sizes.size.x,
height: sizes.size.y
});
/*
this.window.setStyles({
left: (sizes.scroll.x + (sizes.size.x - this.window.offsetWidth) / 2).toInt(),
top: (sizes.scroll.y + (sizes.size.y - this.window.offsetHeight) / 2).toInt()
});
*/
this.fireEvent('onMove', [this.overlay, this.window, sizes]);
},

removeEvents: function(type){
if (!this.$events) return this;
if (!type) this.$events = null;
else if (this.$events[type]) this.$events[type] = null;
return this;
},

parsers: {
'image': function(preset) {
return (preset || this.url.test(/\.(jpg|jpeg|png|gif|bmp)$/i))? this.url : false;
},
'adopt': function(preset) {
if ($(this.options.adopt)) return $(this.options.adopt);
if (preset || ($(this.element) && !this.element.parentNode)) return $(this.element);
var bits = this.url.match(/#([\w-]+)$/);
return bits ? $(bits[1]) : false;
},
'url': function(preset) {
return (preset || (this.url && !this.url.test(/^javascript:/i)))? this.url: false;
},
'iframe': function(preset) {
return (preset || this.url)? this.url: false;
},
'string': function(preset) {
return true;
}
},

handlers: {
'image': function(url) {
this.image = new Image();
var events = {
loaded: function() {
var win = {x: window.getWidth() - this.options.marginImage.x, y: window.getHeight() - this.options.marginImage.y};
var size = {x: this.image.width, y: this.image.height};
for (var i = 0; i < 2; i++)
if (size.x > win.x) {
size.y *= win.x / size.x;
size.x = win.x;
} else if (size.y > win.y) {
size.x *= win.y / size.y;
size.y = win.y;
}
size = {x: parseInt(size.x), y: parseInt(size.y)};
if (window.webkit419) this.image = new Element('img', {'src': this.image.src});
else $(this.image);
this.image.setProperties({
'width': size.x,
'height': size.y});
this.applyContent(this.image, size);
}.bind(this),
failed: this.onError.bind(this)
};
(function() {
this.src = url;
}).delay(10, this.image);
this.image.onload = events.loaded;
this.image.onerror = this.image.onabort = events.failed;
},
'adopt': function(el) {
return el.clone();
},
'url': function(url) {
this.ajax = new Ajax(url, this.options.ajaxOptions);
this.ajax.addEvent('onSuccess', function(resp) {
this.applyContent(resp);
this.ajax = null;
}.bind(this));
this.ajax.addEvent('onFailure', this.onError.bind(this));
this.ajax.request.delay(10, this.ajax);
},
'iframe': function(url) {
return new Element('iframe', {
'src': url,
'frameBorder': 0,
'width': this.options.size.x,
'height': this.options.size.y
});
},
'string': function(str) {
return str;
}
},

extend: $extend
};

SqueezeBox.extend(Events.prototype);
SqueezeBox.extend(Options.prototype);
SqueezeBox.extend(Chain.prototype);
Чтобы оставить сообщение,
Вам необходимо Войти или Зарегистрироваться
 

Отключение всплывающих окон

Автор 8gksp

Ответов: 2
Просмотров: 3238
Последний ответ 27.04.2022, 13:28:22
от 8gksp
Проблема со входом в Админ панель после переноса сайта

Автор Vlad Ermolaev

Ответов: 11
Просмотров: 4177
Последний ответ 12.12.2019, 15:48:25
от SeBun
[Решено] Проблема с авторизацией Invalid Token

Автор treX

Ответов: 100
Просмотров: 97112
Последний ответ 06.10.2019, 16:55:45
от PaLyCH
ссылка на неопубликованный материал

Автор snoopik

Ответов: 9
Просмотров: 7178
Последний ответ 21.03.2019, 04:50:20
от vasmed
Проблема c MobileJoomla

Автор alexmishka

Ответов: 0
Просмотров: 1774
Последний ответ 25.02.2018, 20:32:10
от alexmishka