2017-11-24 07:50:27 +00:00
|
|
|
// (C) Copyright 2015 Martin Dougiamas
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2018-01-31 10:37:42 +00:00
|
|
|
import { Directive, ElementRef, Input, Output, EventEmitter, OnChanges, SimpleChange, Optional } from '@angular/core';
|
|
|
|
import { Platform, NavController, Content } from 'ionic-angular';
|
2017-11-24 07:50:27 +00:00
|
|
|
import { TranslateService } from '@ngx-translate/core';
|
2018-03-01 15:55:49 +00:00
|
|
|
import { CoreAppProvider } from '@providers/app';
|
2018-08-27 10:50:56 +00:00
|
|
|
import { CoreEventsProvider } from '@providers/events';
|
2018-03-01 15:55:49 +00:00
|
|
|
import { CoreFilepoolProvider } from '@providers/filepool';
|
|
|
|
import { CoreLoggerProvider } from '@providers/logger';
|
|
|
|
import { CoreSitesProvider } from '@providers/sites';
|
|
|
|
import { CoreDomUtilsProvider } from '@providers/utils/dom';
|
2018-07-27 10:43:57 +00:00
|
|
|
import { CoreIframeUtilsProvider } from '@providers/utils/iframe';
|
2018-03-01 15:55:49 +00:00
|
|
|
import { CoreTextUtilsProvider } from '@providers/utils/text';
|
|
|
|
import { CoreUrlUtilsProvider } from '@providers/utils/url';
|
|
|
|
import { CoreUtilsProvider } from '@providers/utils/utils';
|
|
|
|
import { CoreSite } from '@classes/site';
|
2017-11-24 07:50:27 +00:00
|
|
|
import { CoreLinkDirective } from '../directives/link';
|
|
|
|
import { CoreExternalContentDirective } from '../directives/external-content';
|
2018-03-01 15:55:49 +00:00
|
|
|
import { CoreContentLinksHelperProvider } from '@core/contentlinks/providers/helper';
|
2018-07-31 13:02:44 +00:00
|
|
|
import { CoreSplitViewComponent } from '@components/split-view/split-view';
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Directive to format text rendered. It renders the HTML and treats all links and media, using CoreLinkDirective
|
|
|
|
* and CoreExternalContentDirective.
|
|
|
|
*
|
|
|
|
* Example usage:
|
|
|
|
* <core-format-text [text]="myText" [component]="component" [componentId]="componentId"></core-format-text>
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
@Directive({
|
|
|
|
selector: 'core-format-text'
|
|
|
|
})
|
2017-12-11 13:59:44 +00:00
|
|
|
export class CoreFormatTextDirective implements OnChanges {
|
2017-11-24 07:50:27 +00:00
|
|
|
@Input() text: string; // The text to format.
|
|
|
|
@Input() siteId?: string; // Site ID to use.
|
|
|
|
@Input() component?: string; // Component for CoreExternalContentDirective.
|
2018-01-29 09:05:20 +00:00
|
|
|
@Input() componentId?: string | number; // Component ID to use in conjunction with the component.
|
|
|
|
@Input() adaptImg?: boolean | string = true; // Whether to adapt images to screen width.
|
|
|
|
@Input() clean?: boolean | string; // Whether all the HTML tags should be removed.
|
|
|
|
@Input() singleLine?: boolean | string; // Whether new lines should be removed (all text in single line). Only if clean=true.
|
2017-11-24 07:50:27 +00:00
|
|
|
@Input() maxHeight?: number; // Max height in pixels to render the content box. It should be 50 at least to make sense.
|
2018-01-29 09:05:20 +00:00
|
|
|
// Using this parameter will force display: block to calculate height better.
|
|
|
|
// If you want to avoid this use class="inline" at the same time to use display: inline-block.
|
|
|
|
@Input() fullOnClick?: boolean | string; // Whether it should open a new page with the full contents on click.
|
2017-11-24 07:50:27 +00:00
|
|
|
@Input() fullTitle?: string; // Title to use in full view. Defaults to "Description".
|
|
|
|
@Output() afterRender?: EventEmitter<any>; // Called when the data is rendered.
|
|
|
|
|
|
|
|
protected element: HTMLElement;
|
2018-08-22 11:44:39 +00:00
|
|
|
protected showMoreDisplayed: boolean;
|
2018-08-27 10:50:56 +00:00
|
|
|
protected loadingChangedListener;
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
constructor(element: ElementRef, private sitesProvider: CoreSitesProvider, private domUtils: CoreDomUtilsProvider,
|
|
|
|
private textUtils: CoreTextUtilsProvider, private translate: TranslateService, private platform: Platform,
|
|
|
|
private utils: CoreUtilsProvider, private urlUtils: CoreUrlUtilsProvider, private loggerProvider: CoreLoggerProvider,
|
2018-01-23 10:26:21 +00:00
|
|
|
private filepoolProvider: CoreFilepoolProvider, private appProvider: CoreAppProvider,
|
2018-03-02 14:25:00 +00:00
|
|
|
private contentLinksHelper: CoreContentLinksHelperProvider, @Optional() private navCtrl: NavController,
|
2018-07-27 10:43:57 +00:00
|
|
|
@Optional() private content: Content, @Optional() private svComponent: CoreSplitViewComponent,
|
2018-08-27 10:50:56 +00:00
|
|
|
private iframeUtils: CoreIframeUtilsProvider, private eventsProvider: CoreEventsProvider) {
|
2017-11-24 07:50:27 +00:00
|
|
|
this.element = element.nativeElement;
|
|
|
|
this.element.classList.add('opacity-hide'); // Hide contents until they're treated.
|
|
|
|
this.afterRender = new EventEmitter();
|
2018-08-22 11:44:39 +00:00
|
|
|
|
|
|
|
this.element.addEventListener('click', this.elementClicked.bind(this));
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-11 13:59:44 +00:00
|
|
|
* Detect changes on input properties.
|
2017-11-24 07:50:27 +00:00
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
ngOnChanges(changes: { [name: string]: SimpleChange }): void {
|
2017-12-11 13:59:44 +00:00
|
|
|
if (changes.text) {
|
|
|
|
this.formatAndRenderContents();
|
|
|
|
}
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply CoreExternalContentDirective to a certain element.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element to add the attributes to.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected addExternalContent(element: HTMLElement): void {
|
2017-11-24 07:50:27 +00:00
|
|
|
// Angular 2 doesn't let adding directives dynamically. Create the CoreExternalContentDirective manually.
|
2018-01-29 09:05:20 +00:00
|
|
|
const extContent = new CoreExternalContentDirective(<any> element, this.loggerProvider, this.filepoolProvider,
|
2018-08-13 14:31:24 +00:00
|
|
|
this.platform, this.sitesProvider, this.domUtils, this.urlUtils, this.appProvider, this.utils);
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
extContent.component = this.component;
|
|
|
|
extContent.componentId = this.componentId;
|
|
|
|
extContent.siteId = this.siteId;
|
|
|
|
|
2017-11-30 09:21:03 +00:00
|
|
|
extContent.ngAfterViewInit();
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add class to adapt media to a certain element.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element to add the class to.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected addMediaAdaptClass(element: HTMLElement): void {
|
2017-12-29 17:05:52 +00:00
|
|
|
element.classList.add('core-media-adapt-width');
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-01-02 07:47:21 +00:00
|
|
|
* Wrap an image with a container to adapt its width and, if needed, add an anchor to view it in full size.
|
2017-11-24 07:50:27 +00:00
|
|
|
*
|
|
|
|
* @param {number} elWidth Width of the directive's element.
|
|
|
|
* @param {HTMLElement} img Image to adapt.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected adaptImage(elWidth: number, img: HTMLElement): void {
|
|
|
|
const imgWidth = this.getElementWidth(img),
|
2018-01-02 07:47:21 +00:00
|
|
|
// Element to wrap the image.
|
2018-08-24 11:14:10 +00:00
|
|
|
container = document.createElement('span'),
|
|
|
|
originalWidth = img.attributes.getNamedItem('width');
|
2017-11-24 07:50:27 +00:00
|
|
|
|
2018-08-24 11:14:10 +00:00
|
|
|
const forcedWidth = parseInt(originalWidth && originalWidth.value);
|
|
|
|
if (!isNaN(forcedWidth)) {
|
|
|
|
if (originalWidth.value.indexOf('%') < 0) {
|
|
|
|
img.style.width = forcedWidth + 'px';
|
|
|
|
} else {
|
|
|
|
img.style.width = forcedWidth + '%';
|
|
|
|
}
|
2018-08-06 13:58:25 +00:00
|
|
|
}
|
|
|
|
|
2017-12-29 17:05:52 +00:00
|
|
|
container.classList.add('core-adapted-img-container');
|
2017-11-24 07:50:27 +00:00
|
|
|
container.style.cssFloat = img.style.cssFloat; // Copy the float to correctly position the search icon.
|
|
|
|
if (img.classList.contains('atto_image_button_right')) {
|
|
|
|
container.classList.add('atto_image_button_right');
|
|
|
|
} else if (img.classList.contains('atto_image_button_left')) {
|
|
|
|
container.classList.add('atto_image_button_left');
|
|
|
|
}
|
2018-01-02 07:47:21 +00:00
|
|
|
|
|
|
|
this.domUtils.wrapElement(img, container);
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
if (imgWidth > elWidth) {
|
2018-01-02 07:47:21 +00:00
|
|
|
// The image has been adapted, add an anchor to view it in full size.
|
2018-01-03 09:36:07 +00:00
|
|
|
this.addMagnifyingGlass(container, img);
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-03 09:36:07 +00:00
|
|
|
/**
|
|
|
|
* Add a magnifying glass icon to view an image at full size.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} container The container of the image.
|
|
|
|
* @param {HTMLElement} img The image.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
addMagnifyingGlass(container: HTMLElement, img: HTMLElement): void {
|
|
|
|
const imgSrc = this.textUtils.escapeHTML(img.getAttribute('src')),
|
2018-01-03 09:36:07 +00:00
|
|
|
label = this.textUtils.escapeHTML(this.translate.instant('core.openfullimage')),
|
|
|
|
anchor = document.createElement('a');
|
|
|
|
|
|
|
|
anchor.classList.add('core-image-viewer-icon');
|
|
|
|
anchor.setAttribute('aria-label', label);
|
|
|
|
// Add an ion-icon item to apply the right styles, but the ion-icon component won't be executed.
|
|
|
|
anchor.innerHTML = '<ion-icon name="search" class="icon icon-md ion-md-search"></ion-icon>';
|
|
|
|
|
|
|
|
anchor.addEventListener('click', (e: Event) => {
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2018-01-23 12:00:00 +00:00
|
|
|
this.domUtils.viewImage(imgSrc, img.getAttribute('alt'), this.component, this.componentId);
|
2018-01-03 09:36:07 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
container.appendChild(anchor);
|
|
|
|
}
|
|
|
|
|
2018-06-19 07:22:52 +00:00
|
|
|
/**
|
|
|
|
* Calculate the height and check if we need to display show more or not.
|
|
|
|
*/
|
|
|
|
protected calculateHeight(): void {
|
|
|
|
// @todo: Work on calculate this height better.
|
2018-08-27 10:50:56 +00:00
|
|
|
|
|
|
|
// Remove max-height (if any) to calculate the real height.
|
|
|
|
const initialMaxHeight = this.element.style.maxHeight;
|
|
|
|
this.element.style.maxHeight = null;
|
|
|
|
|
|
|
|
const height = this.getElementHeight(this.element);
|
|
|
|
|
|
|
|
// Restore the max height now.
|
|
|
|
this.element.style.maxHeight = initialMaxHeight;
|
2018-06-19 07:22:52 +00:00
|
|
|
|
|
|
|
// If cannot calculate height, shorten always.
|
|
|
|
if (!height || height > this.maxHeight) {
|
2018-08-22 11:44:39 +00:00
|
|
|
if (!this.showMoreDisplayed) {
|
2018-06-19 07:22:52 +00:00
|
|
|
this.displayShowMore();
|
|
|
|
}
|
2018-08-22 11:44:39 +00:00
|
|
|
} else if (this.showMoreDisplayed) {
|
2018-06-19 07:22:52 +00:00
|
|
|
this.hideShowMore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Display the "Show more" in the element.
|
|
|
|
*/
|
|
|
|
protected displayShowMore(): void {
|
|
|
|
const expandInFullview = this.utils.isTrueOrOne(this.fullOnClick) || false,
|
|
|
|
showMoreDiv = document.createElement('div');
|
|
|
|
|
|
|
|
showMoreDiv.classList.add('core-show-more');
|
|
|
|
showMoreDiv.innerHTML = this.translate.instant('core.showmore');
|
|
|
|
this.element.appendChild(showMoreDiv);
|
|
|
|
|
|
|
|
if (expandInFullview) {
|
|
|
|
this.element.classList.add('core-expand-in-fullview');
|
|
|
|
}
|
|
|
|
this.element.classList.add('core-text-formatted');
|
|
|
|
this.element.classList.add('core-shortened');
|
|
|
|
this.element.style.maxHeight = this.maxHeight + 'px';
|
|
|
|
|
2018-08-22 11:44:39 +00:00
|
|
|
this.showMoreDisplayed = true;
|
2018-06-19 07:22:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Listener to call when the element is clicked.
|
|
|
|
*
|
|
|
|
* @param {MouseEvent} e Click event.
|
|
|
|
*/
|
2018-08-22 11:44:39 +00:00
|
|
|
protected elementClicked(e: MouseEvent): void {
|
2018-06-19 07:22:52 +00:00
|
|
|
if (e.defaultPrevented) {
|
|
|
|
// Ignore it if the event was prevented by some other listener.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-08-22 11:44:39 +00:00
|
|
|
const expandInFullview = this.utils.isTrueOrOne(this.fullOnClick) || false;
|
|
|
|
|
|
|
|
if (!expandInFullview && !this.showMoreDisplayed) {
|
|
|
|
// Nothing to do on click, just stop.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-19 07:22:52 +00:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
|
2018-08-21 10:41:24 +00:00
|
|
|
if (!expandInFullview) {
|
|
|
|
// Change class.
|
|
|
|
this.element.classList.toggle('core-shortened');
|
2018-06-19 07:22:52 +00:00
|
|
|
|
2018-08-21 10:41:24 +00:00
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
// Open a new state with the contents.
|
|
|
|
this.textUtils.expandText(this.fullTitle || this.translate.instant('core.description'), this.text,
|
|
|
|
this.component, this.componentId);
|
2018-06-19 07:22:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
/**
|
|
|
|
* Finish the rendering, displaying the element again and calling afterRender.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected finishRender(): void {
|
2017-11-24 07:50:27 +00:00
|
|
|
// Show the element again.
|
|
|
|
this.element.classList.remove('opacity-hide');
|
|
|
|
// Emit the afterRender output.
|
|
|
|
this.afterRender.emit();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format contents and render.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected formatAndRenderContents(): void {
|
2017-11-24 07:50:27 +00:00
|
|
|
if (!this.text) {
|
2017-12-15 07:03:33 +00:00
|
|
|
this.element.innerHTML = ''; // Remove current contents.
|
2017-11-24 07:50:27 +00:00
|
|
|
this.finishRender();
|
2018-01-29 09:05:20 +00:00
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-08-28 11:45:28 +00:00
|
|
|
// In AOT the inputs and ng-reflect aren't in the DOM sometimes. Add them so styles are applied.
|
|
|
|
if (this.maxHeight && !this.element.getAttribute('maxHeight')) {
|
|
|
|
this.element.setAttribute('maxHeight', String(this.maxHeight));
|
|
|
|
}
|
|
|
|
if (!this.element.getAttribute('singleLine')) {
|
|
|
|
this.element.setAttribute('singleLine', String(this.utils.isTrueOrOne(this.singleLine)));
|
|
|
|
}
|
|
|
|
|
2018-03-06 11:47:51 +00:00
|
|
|
this.text = this.text ? this.text.trim() : '';
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
this.formatContents().then((div: HTMLElement) => {
|
2018-01-23 12:00:00 +00:00
|
|
|
// Disable media adapt to correctly calculate the height.
|
|
|
|
this.element.classList.add('core-disable-media-adapt');
|
2017-12-15 07:03:33 +00:00
|
|
|
|
2018-01-23 12:00:00 +00:00
|
|
|
this.element.innerHTML = ''; // Remove current contents.
|
2018-01-29 09:05:20 +00:00
|
|
|
if (this.maxHeight && div.innerHTML != '') {
|
2018-07-06 15:15:08 +00:00
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
// Move the children to the current element to be able to calculate the height.
|
|
|
|
this.domUtils.moveChildren(div, this.element);
|
|
|
|
|
2018-06-19 07:22:52 +00:00
|
|
|
// Calculate the height now.
|
|
|
|
this.calculateHeight();
|
2017-11-24 07:50:27 +00:00
|
|
|
|
2018-06-19 07:22:52 +00:00
|
|
|
// Wait for images to load and calculate the height again if needed.
|
2018-09-13 15:19:07 +00:00
|
|
|
this.domUtils.waitForImages(this.element).then((hasImgToLoad) => {
|
2018-06-19 07:22:52 +00:00
|
|
|
if (hasImgToLoad) {
|
|
|
|
this.calculateHeight();
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
2018-06-19 07:22:52 +00:00
|
|
|
});
|
2018-08-27 10:50:56 +00:00
|
|
|
|
|
|
|
if (!this.loadingChangedListener) {
|
|
|
|
// Recalculate the height if a parent core-loading displays the content.
|
|
|
|
this.loadingChangedListener = this.eventsProvider.on(CoreEventsProvider.CORE_LOADING_CHANGED, (data) => {
|
|
|
|
if (data.loaded && this.domUtils.closest(this.element.parentElement, '#' + data.uniqueId)) {
|
|
|
|
// The format-text is inside the loading, re-calculate the height.
|
|
|
|
this.calculateHeight();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2017-11-24 07:50:27 +00:00
|
|
|
} else {
|
|
|
|
this.domUtils.moveChildren(div, this.element);
|
|
|
|
}
|
|
|
|
|
2018-01-23 12:00:00 +00:00
|
|
|
this.element.classList.remove('core-disable-media-adapt');
|
2017-11-24 07:50:27 +00:00
|
|
|
this.finishRender();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply formatText and set sub-directives.
|
|
|
|
*
|
|
|
|
* @return {Promise<HTMLElement>} Promise resolved with a div element containing the code.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected formatContents(): Promise<HTMLElement> {
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
let site: CoreSite;
|
|
|
|
|
|
|
|
// Retrieve the site since it might be needed later.
|
|
|
|
return this.sitesProvider.getSite(this.siteId).catch(() => {
|
|
|
|
// Error getting the site. This probably means that there is no current site and no siteId was supplied.
|
|
|
|
}).then((siteInstance: CoreSite) => {
|
|
|
|
site = siteInstance;
|
|
|
|
|
|
|
|
// Apply format text function.
|
2018-01-25 12:19:11 +00:00
|
|
|
return this.textUtils.formatText(this.text, this.utils.isTrueOrOne(this.clean),
|
|
|
|
this.utils.isTrueOrOne(this.singleLine));
|
2017-11-24 07:50:27 +00:00
|
|
|
}).then((formatted) => {
|
2018-01-29 09:05:20 +00:00
|
|
|
const div = document.createElement('div'),
|
|
|
|
canTreatVimeo = site && site.isVersionGreaterEqualThan(['3.3.4', '3.4']);
|
|
|
|
let images,
|
2017-11-24 07:50:27 +00:00
|
|
|
anchors,
|
|
|
|
audios,
|
|
|
|
videos,
|
|
|
|
iframes,
|
2018-08-13 14:31:24 +00:00
|
|
|
buttons,
|
2018-08-21 10:41:24 +00:00
|
|
|
elementsWithInlineStyles,
|
2018-07-27 10:43:57 +00:00
|
|
|
stopClicksElements,
|
|
|
|
frames;
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
div.innerHTML = formatted;
|
|
|
|
images = Array.from(div.querySelectorAll('img'));
|
|
|
|
anchors = Array.from(div.querySelectorAll('a'));
|
|
|
|
audios = Array.from(div.querySelectorAll('audio'));
|
|
|
|
videos = Array.from(div.querySelectorAll('video'));
|
|
|
|
iframes = Array.from(div.querySelectorAll('iframe'));
|
|
|
|
buttons = Array.from(div.querySelectorAll('.button'));
|
2018-08-13 14:31:24 +00:00
|
|
|
elementsWithInlineStyles = Array.from(div.querySelectorAll('*[style]'));
|
2018-08-21 10:41:24 +00:00
|
|
|
stopClicksElements = Array.from(div.querySelectorAll('button,input,select,textarea'));
|
2018-08-20 07:38:31 +00:00
|
|
|
frames = Array.from(div.querySelectorAll(CoreIframeUtilsProvider.FRAME_TAGS.join(',').replace(/iframe,?/, '')));
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
// Walk through the content to find the links and add our directive to it.
|
2017-12-29 17:05:52 +00:00
|
|
|
// Important: We need to look for links first because in 'img' we add new links without core-link.
|
2017-11-24 07:50:27 +00:00
|
|
|
anchors.forEach((anchor) => {
|
|
|
|
// Angular 2 doesn't let adding directives dynamically. Create the CoreLinkDirective manually.
|
2018-01-29 09:05:20 +00:00
|
|
|
const linkDir = new CoreLinkDirective(anchor, this.domUtils, this.utils, this.sitesProvider, this.urlUtils,
|
2018-07-31 13:02:44 +00:00
|
|
|
this.contentLinksHelper, this.navCtrl, this.content, this.svComponent);
|
2017-11-24 07:50:27 +00:00
|
|
|
linkDir.capture = true;
|
|
|
|
linkDir.ngOnInit();
|
|
|
|
|
|
|
|
this.addExternalContent(anchor);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (images && images.length > 0) {
|
|
|
|
// If cannot calculate element's width, use a medium number to avoid false adapt image icons appearing.
|
2018-01-29 09:05:20 +00:00
|
|
|
const elWidth = this.getElementWidth(this.element) || 100;
|
2017-11-24 07:50:27 +00:00
|
|
|
|
|
|
|
// Walk through the content to find images, and add our directive.
|
|
|
|
images.forEach((img: HTMLElement) => {
|
|
|
|
this.addMediaAdaptClass(img);
|
|
|
|
this.addExternalContent(img);
|
2017-11-30 15:27:10 +00:00
|
|
|
if (this.utils.isTrueOrOne(this.adaptImg)) {
|
2018-01-02 07:47:21 +00:00
|
|
|
this.adaptImage(elWidth, img);
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
audios.forEach((audio) => {
|
|
|
|
this.treatMedia(audio);
|
|
|
|
});
|
|
|
|
|
|
|
|
videos.forEach((video) => {
|
|
|
|
this.treatVideoFilters(video);
|
|
|
|
this.treatMedia(video);
|
|
|
|
});
|
|
|
|
|
|
|
|
iframes.forEach((iframe) => {
|
|
|
|
this.treatIframe(iframe, site, canTreatVimeo);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Handle buttons with inner links.
|
|
|
|
buttons.forEach((button: HTMLElement) => {
|
|
|
|
// Check if it has a link inside.
|
|
|
|
if (button.querySelector('a')) {
|
2017-12-29 17:05:52 +00:00
|
|
|
button.classList.add('core-button-with-inner-link');
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-08-13 14:31:24 +00:00
|
|
|
// Handle inline styles.
|
|
|
|
elementsWithInlineStyles.forEach((el: HTMLElement) => {
|
|
|
|
// Only add external content for tags that haven't been treated already.
|
|
|
|
if (el.tagName != 'A' && el.tagName != 'IMG' && el.tagName != 'AUDIO' && el.tagName != 'VIDEO'
|
|
|
|
&& el.tagName != 'SOURCE' && el.tagName != 'TRACK') {
|
|
|
|
this.addExternalContent(el);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-08-21 10:41:24 +00:00
|
|
|
// Stop propagating click events.
|
|
|
|
stopClicksElements.forEach((element: HTMLElement) => {
|
|
|
|
element.addEventListener('click', (e) => {
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-07-27 10:43:57 +00:00
|
|
|
// Handle all kind of frames.
|
|
|
|
frames.forEach((frame: any) => {
|
|
|
|
this.iframeUtils.treatFrame(frame);
|
|
|
|
});
|
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
return div;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the element width in pixels.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element to get width from.
|
|
|
|
* @return {number} The width of the element in pixels. When 0 is returned it means the element is not visible.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected getElementWidth(element: HTMLElement): number {
|
2017-11-24 07:50:27 +00:00
|
|
|
let width = this.domUtils.getElementWidth(element);
|
|
|
|
|
|
|
|
if (!width) {
|
|
|
|
// All elements inside are floating or inline. Change display mode to allow calculate the width.
|
2018-01-29 09:05:20 +00:00
|
|
|
const parentWidth = this.domUtils.getElementWidth(element.parentNode, true, false, false, true),
|
2017-11-24 07:50:27 +00:00
|
|
|
previousDisplay = getComputedStyle(element, null).display;
|
|
|
|
|
|
|
|
element.style.display = 'inline-block';
|
|
|
|
|
|
|
|
width = this.domUtils.getElementWidth(element);
|
|
|
|
|
|
|
|
// If width is incorrectly calculated use parent width instead.
|
|
|
|
if (parentWidth > 0 && (!width || width > parentWidth)) {
|
|
|
|
width = parentWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
element.style.display = previousDisplay;
|
|
|
|
}
|
|
|
|
|
|
|
|
return width;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the element height in pixels.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} elementAng Element to get height from.
|
|
|
|
* @return {number} The height of the element in pixels. When 0 is returned it means the element is not visible.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected getElementHeight(element: HTMLElement): number {
|
2018-01-23 12:00:00 +00:00
|
|
|
return this.domUtils.getElementHeight(element) || 0;
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
2018-06-19 07:22:52 +00:00
|
|
|
/**
|
|
|
|
* "Hide" the "Show more" in the element if it's shown.
|
|
|
|
*/
|
|
|
|
protected hideShowMore(): void {
|
|
|
|
const showMoreDiv = this.element.querySelector('div.core-show-more');
|
|
|
|
|
|
|
|
if (showMoreDiv) {
|
|
|
|
showMoreDiv.remove();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.element.classList.remove('core-expand-in-fullview');
|
|
|
|
this.element.classList.remove('core-text-formatted');
|
|
|
|
this.element.classList.remove('core-shortened');
|
|
|
|
this.element.style.maxHeight = null;
|
2018-08-22 11:44:39 +00:00
|
|
|
this.showMoreDisplayed = false;
|
2018-06-19 07:22:52 +00:00
|
|
|
}
|
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
/**
|
|
|
|
* Treat video filters. Currently only treating youtube video using video JS.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} el Video element.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected treatVideoFilters(video: HTMLElement): void {
|
2017-11-24 07:50:27 +00:00
|
|
|
// Treat Video JS Youtube video links and translate them to iframes.
|
|
|
|
if (!video.classList.contains('video-js')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-03-13 08:12:30 +00:00
|
|
|
const data = this.textUtils.parseJSON(video.getAttribute('data-setup') || video.getAttribute('data-setup-lazy') || '{}'),
|
2017-11-24 07:50:27 +00:00
|
|
|
youtubeId = data.techOrder && data.techOrder[0] && data.techOrder[0] == 'youtube' && data.sources && data.sources[0] &&
|
|
|
|
data.sources[0].src && this.youtubeGetId(data.sources[0].src);
|
|
|
|
|
|
|
|
if (!youtubeId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-29 09:05:20 +00:00
|
|
|
const iframe = document.createElement('iframe');
|
2017-11-24 07:50:27 +00:00
|
|
|
iframe.id = video.id;
|
|
|
|
iframe.src = 'https://www.youtube.com/embed/' + youtubeId;
|
|
|
|
iframe.setAttribute('frameborder', '0');
|
2018-07-16 10:22:20 +00:00
|
|
|
iframe.setAttribute('allowfullscreen', '1');
|
2017-11-24 07:50:27 +00:00
|
|
|
iframe.width = '100%';
|
|
|
|
iframe.height = '300';
|
|
|
|
|
|
|
|
// Replace video tag by the iframe.
|
|
|
|
video.parentNode.replaceChild(iframe, video);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add media adapt class and apply CoreExternalContentDirective to the media element and its sources and tracks.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Video or audio to treat.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected treatMedia(element: HTMLElement): void {
|
2017-11-24 07:50:27 +00:00
|
|
|
this.addMediaAdaptClass(element);
|
|
|
|
this.addExternalContent(element);
|
|
|
|
|
2018-01-29 09:05:20 +00:00
|
|
|
const sources = Array.from(element.querySelectorAll('source')),
|
2017-11-24 07:50:27 +00:00
|
|
|
tracks = Array.from(element.querySelectorAll('track'));
|
|
|
|
|
|
|
|
sources.forEach((source) => {
|
|
|
|
source.setAttribute('target-src', source.getAttribute('src'));
|
|
|
|
source.removeAttribute('src');
|
|
|
|
this.addExternalContent(source);
|
|
|
|
});
|
|
|
|
|
|
|
|
tracks.forEach((track) => {
|
|
|
|
this.addExternalContent(track);
|
|
|
|
});
|
2018-08-21 10:41:24 +00:00
|
|
|
|
|
|
|
// Stop propagating click events.
|
|
|
|
element.addEventListener('click', (e) => {
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add media adapt class and treat the iframe source.
|
|
|
|
*
|
|
|
|
* @param {HTMLIFrameElement} iframe Iframe to treat.
|
|
|
|
* @param {CoreSite} site Site instance.
|
|
|
|
* @param {Boolean} canTreatVimeo Whether Vimeo videos can be treated in the site.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected treatIframe(iframe: HTMLIFrameElement, site: CoreSite, canTreatVimeo: boolean): void {
|
2018-08-20 07:38:31 +00:00
|
|
|
const src = iframe.src,
|
|
|
|
currentSite = this.sitesProvider.getCurrentSite();
|
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
this.addMediaAdaptClass(iframe);
|
|
|
|
|
2018-08-20 07:38:31 +00:00
|
|
|
if (currentSite && currentSite.containsUrl(src)) {
|
|
|
|
// URL points to current site, try to use auto-login.
|
|
|
|
currentSite.getAutoLoginUrl(src, false).then((finalUrl) => {
|
|
|
|
iframe.src = finalUrl;
|
|
|
|
|
|
|
|
this.iframeUtils.treatFrame(iframe);
|
|
|
|
});
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
} else if (src && canTreatVimeo) {
|
2017-11-24 07:50:27 +00:00
|
|
|
// Check if it's a Vimeo video. If it is, use the wsplayer script instead to make restricted videos work.
|
2018-05-18 07:32:25 +00:00
|
|
|
const matches = iframe.src.match(/https?:\/\/player\.vimeo\.com\/video\/([0-9]+)/);
|
2017-11-24 07:50:27 +00:00
|
|
|
if (matches && matches[1]) {
|
2018-05-18 07:32:25 +00:00
|
|
|
const newUrl = this.textUtils.concatenatePaths(site.getURL(), '/media/player/vimeo/wsplayer.php?video=') +
|
2018-01-29 09:05:20 +00:00
|
|
|
matches[1] + '&token=' + site.getToken();
|
2018-05-18 07:32:25 +00:00
|
|
|
|
|
|
|
// Width and height are mandatory, we need to calculate them.
|
|
|
|
let width, height;
|
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
if (iframe.width) {
|
2018-05-18 07:32:25 +00:00
|
|
|
width = iframe.width;
|
|
|
|
} else {
|
|
|
|
width = this.getElementWidth(iframe);
|
|
|
|
if (!width) {
|
|
|
|
width = window.innerWidth;
|
|
|
|
}
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
2018-05-18 07:32:25 +00:00
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
if (iframe.height) {
|
2018-05-18 07:32:25 +00:00
|
|
|
height = iframe.height;
|
|
|
|
} else {
|
|
|
|
height = this.getElementHeight(iframe);
|
|
|
|
if (!height) {
|
|
|
|
height = width;
|
|
|
|
}
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
2018-05-18 07:32:25 +00:00
|
|
|
// Always include the width and height in the URL.
|
|
|
|
iframe.src = newUrl + '&width=' + width + '&height=' + height;
|
|
|
|
if (!iframe.width) {
|
|
|
|
iframe.width = width;
|
|
|
|
}
|
|
|
|
if (!iframe.height) {
|
|
|
|
iframe.height = height;
|
|
|
|
}
|
2018-06-08 09:12:31 +00:00
|
|
|
|
|
|
|
// Do the iframe responsive.
|
|
|
|
if (iframe.parentElement.classList.contains('embed-responsive')) {
|
|
|
|
iframe.addEventListener('load', () => {
|
2018-08-01 12:54:46 +00:00
|
|
|
if (iframe.contentDocument) {
|
|
|
|
const css = document.createElement('style');
|
|
|
|
css.setAttribute('type', 'text/css');
|
|
|
|
css.innerHTML = 'iframe {width: 100%;height: 100%;}';
|
|
|
|
iframe.contentDocument.head.appendChild(css);
|
|
|
|
}
|
2018-06-08 09:12:31 +00:00
|
|
|
});
|
|
|
|
}
|
2018-08-20 07:38:31 +00:00
|
|
|
|
|
|
|
return;
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
}
|
2018-08-20 07:38:31 +00:00
|
|
|
|
|
|
|
this.iframeUtils.treatFrame(iframe);
|
2017-11-24 07:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convenience function to extract YouTube Id to translate to embedded video.
|
|
|
|
* Based on http://stackoverflow.com/questions/3452546/javascript-regex-how-to-get-youtube-video-id-from-url
|
|
|
|
*
|
|
|
|
* @param {string} url URL of the video.
|
|
|
|
*/
|
2018-01-29 09:05:20 +00:00
|
|
|
protected youtubeGetId(url: string): string {
|
|
|
|
const regExp = /^.*(?:(?:youtu.be\/)|(?:v\/)|(?:\/u\/\w\/)|(?:embed\/)|(?:watch\?))\??v?=?([^#\&\?]*).*/,
|
2017-11-24 07:50:27 +00:00
|
|
|
match = url.match(regExp);
|
2018-01-29 09:05:20 +00:00
|
|
|
|
2017-11-24 07:50:27 +00:00
|
|
|
return (match && match[1].length == 11) ? match[1] : '';
|
|
|
|
}
|
|
|
|
}
|