@atlaskit/editor-plugin-media
Version:
Media plugin for @atlaskit/editor-core
401 lines (395 loc) • 22.1 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _typeof = require("@babel/runtime/helpers/typeof");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.MediaNode = exports.MEDIA_HEIGHT = exports.FILE_WIDTH = void 0;
var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _bindEventListener = require("bind-event-listener");
var _memoizeOne = _interopRequireDefault(require("memoize-one"));
var _analyticsNamespacedContext = require("@atlaskit/analytics-namespaced-context");
var _analyticsNext = require("@atlaskit/analytics-next");
var _toolbarFlagCheck = require("@atlaskit/editor-common/toolbar-flag-check");
var _utils = require("@atlaskit/editor-common/utils");
var _utils2 = require("@atlaskit/editor-prosemirror/utils");
var _cellSelection = require("@atlaskit/editor-tables/cell-selection");
var _mediaCard = require("@atlaskit/media-card");
var _platformFeatureFlags = require("@atlaskit/platform-feature-flags");
var _expValEquals = require("@atlaskit/tmp-editor-statsig/exp-val-equals");
var _experiments = require("@atlaskit/tmp-editor-statsig/experiments");
var _pluginKey = require("../../pm-plugins/plugin-key");
var _styles = require("../styles");
function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function _interopRequireWildcard(e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != _typeof(e) && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (var _t in e) "default" !== _t && {}.hasOwnProperty.call(e, _t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, _t)) && (i.get || i.set) ? o(f, _t, i) : f[_t] = e[_t]); return f; })(e, t); }
function _callSuper(t, o, e) { return o = (0, _getPrototypeOf2.default)(o), (0, _possibleConstructorReturn2.default)(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], (0, _getPrototypeOf2.default)(t).constructor) : o.apply(t, e)); }
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
// This is being used by DropPlaceholder now
var MEDIA_HEIGHT = exports.MEDIA_HEIGHT = 125;
var FILE_WIDTH = exports.FILE_WIDTH = 156;
// eslint-disable-next-line @repo/internal/react/no-class-components
var MediaNode = exports.MediaNode = /*#__PURE__*/function (_Component) {
function MediaNode(_props) {
var _this;
(0, _classCallCheck2.default)(this, MediaNode);
_this = _callSuper(this, MediaNode, [_props]);
(0, _defineProperty2.default)(_this, "state", {});
(0, _defineProperty2.default)(_this, "videoControlsWrapperRef", /*#__PURE__*/_react.default.createRef());
(0, _defineProperty2.default)(_this, "unbindKeyDown", null);
(0, _defineProperty2.default)(_this, "setViewMediaClientConfig", /*#__PURE__*/(0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee() {
var mediaProvider, viewMediaClientConfig, viewAndUploadMediaClientConfig;
return _regenerator.default.wrap(function _callee$(_context) {
while (1) switch (_context.prev = _context.next) {
case 0:
_context.next = 2;
return _this.props.mediaProvider;
case 2:
mediaProvider = _context.sent;
if (mediaProvider) {
viewMediaClientConfig = mediaProvider.viewMediaClientConfig;
viewAndUploadMediaClientConfig = mediaProvider.viewAndUploadMediaClientConfig;
if ((0, _expValEquals.expValEquals)('platform_editor_media_vc_fixes', 'isEnabled', true)) {
// Only update state if new configs are available and different from current state
if (viewMediaClientConfig && _this.state.viewMediaClientConfig !== viewMediaClientConfig || viewAndUploadMediaClientConfig && _this.state.viewAndUploadMediaClientConfig !== viewAndUploadMediaClientConfig) {
_this.setState({
viewMediaClientConfig: viewMediaClientConfig,
viewAndUploadMediaClientConfig: viewAndUploadMediaClientConfig
});
}
} else {
_this.setState({
viewMediaClientConfig: viewMediaClientConfig,
viewAndUploadMediaClientConfig: viewAndUploadMediaClientConfig
});
}
}
case 4:
case "end":
return _context.stop();
}
}, _callee);
})));
(0, _defineProperty2.default)(_this, "selectMediaSingleFromCard", function (_ref2) {
var _this$props$pluginInj;
var event = _ref2.event;
_this.selectMediaSingle(event);
// In edit mode (node content wrapper has contenteditable set to true), link redirection is disabled by default
// We need to call "stopPropagation" here in order to prevent in editor view mode, the browser from navigating to
// another URL if the media node is wrapped in a link mark.
if (_this.props.isViewOnly && (0, _toolbarFlagCheck.areToolbarFlagsEnabled)(Boolean((_this$props$pluginInj = _this.props.pluginInjectionApi) === null || _this$props$pluginInj === void 0 ? void 0 : _this$props$pluginInj.toolbar))) {
event.preventDefault();
}
});
(0, _defineProperty2.default)(_this, "selectMediaSingle", function (event) {
var propPos = _this.props.getPos();
if (typeof propPos !== 'number') {
return;
}
// NOTE: This does not prevent the link navigation in the editor view mode, .preventDefault is needed (see selectMediaSingleFromCard)
// Hence it should be removed
// We need to call "stopPropagation" here in order to prevent the browser from navigating to
// another URL if the media node is wrapped in a link mark.
if ((0, _experiments.editorExperiment)('platform_editor_controls', 'control')) {
event.stopPropagation();
}
var state = _this.props.view.state;
if (event.shiftKey) {
// don't select text if there is current selection in a table (as this would override selected cells)
if (state.selection instanceof _cellSelection.CellSelection) {
return;
}
(0, _utils.setTextSelection)(_this.props.view, state.selection.from < propPos ? state.selection.from : propPos - 1,
// + 3 needed for offset of the media inside mediaSingle and cursor to make whole mediaSingle selected
state.selection.to > propPos ? state.selection.to : propPos + 2);
} else {
(0, _utils.setNodeSelection)(_this.props.view, propPos - 1);
}
});
(0, _defineProperty2.default)(_this, "getMediaSettings", (0, _memoizeOne.default)(function (viewAndUploadMediaClientConfig) {
return {
canUpdateVideoCaptions: (0, _platformFeatureFlags.fg)('platform_media_video_captions') ? !!viewAndUploadMediaClientConfig : false
};
}));
(0, _defineProperty2.default)(_this, "onError", function (reason) {
var _this$props$api;
var nestedUnder = _this.getNestedUnder();
(_this$props$api = _this.props.api) === null || _this$props$api === void 0 || _this$props$api.media.actions.handleMediaNodeRenderError(_this.props.node, reason, nestedUnder);
});
/**
* This function checks if the media node is nested under a certain nodes, and if so,
* returns the name of the parent node type. This is used for providing more context in media render errors.
* @returns
*/
(0, _defineProperty2.default)(_this, "getNestedUnder", function () {
var pos = _this.props.getPos();
if (typeof pos !== 'number') {
return undefined;
}
var _this$props$view$stat = _this.props.view.state,
doc = _this$props$view$stat.doc,
schema = _this$props$view$stat.schema;
var bodiedSyncBlock = schema.nodes.bodiedSyncBlock;
if (!bodiedSyncBlock) {
return undefined;
}
var resolvedPos = doc.resolve(pos);
var bodiedSyncBlockNode = (0, _utils2.findParentNodeClosestToPos)(resolvedPos, function (currentNode) {
return currentNode.type === bodiedSyncBlock;
});
return bodiedSyncBlockNode === null || bodiedSyncBlockNode === void 0 ? void 0 : bodiedSyncBlockNode.node.type.name;
});
(0, _defineProperty2.default)(_this, "onFullscreenChange", function (fullscreen) {
var _this$mediaPluginStat;
(_this$mediaPluginStat = _this.mediaPluginState) === null || _this$mediaPluginStat === void 0 || _this$mediaPluginStat.updateAndDispatch({
isFullscreen: fullscreen
});
});
(0, _defineProperty2.default)(_this, "handleNewNode", function (props) {
var _this$mediaPluginStat2;
var node = props.node;
(_this$mediaPluginStat2 = _this.mediaPluginState) === null || _this$mediaPluginStat2 === void 0 || _this$mediaPluginStat2.handleMediaNodeMount(node, function () {
return _this.props.getPos();
});
});
var _this$props = _this.props,
view = _this$props.view,
syncProvider = _this$props.syncProvider;
_this.mediaPluginState = _pluginKey.stateKey.getState(view.state);
// Initialize state from syncProvider (available on both server and client for SSR)
if ((0, _expValEquals.expValEquals)('platform_editor_media_vc_fixes', 'isEnabled', true) && syncProvider) {
_this.state = {
viewMediaClientConfig: syncProvider.viewMediaClientConfig,
viewAndUploadMediaClientConfig: syncProvider.viewAndUploadMediaClientConfig
};
}
return _this;
}
(0, _inherits2.default)(MediaNode, _Component);
return (0, _createClass2.default)(MediaNode, [{
key: "shouldComponentUpdate",
value: function shouldComponentUpdate(nextProps, nextState) {
var hasNewViewMediaClientConfig = !this.state.viewMediaClientConfig && nextState.viewMediaClientConfig;
var hasNewViewAndUploadMediaClientConfig = !this.state.viewAndUploadMediaClientConfig && nextState.viewAndUploadMediaClientConfig;
if (this.props.selected !== nextProps.selected || this.props.node.attrs.id !== nextProps.node.attrs.id || this.props.node.attrs.collection !== nextProps.node.attrs.collection || this.props.isAIGenerating !== nextProps.isAIGenerating || this.props.maxDimensions.height !== nextProps.maxDimensions.height || this.props.maxDimensions.width !== nextProps.maxDimensions.width || this.props.contextIdentifierProvider !== nextProps.contextIdentifierProvider || this.props.isLoading !== nextProps.isLoading || this.props.mediaProvider !== nextProps.mediaProvider || (0, _expValEquals.expValEquals)('platform_editor_media_vc_fixes', 'isEnabled', true) && this.props.syncProvider !== nextProps.syncProvider || hasNewViewMediaClientConfig || hasNewViewAndUploadMediaClientConfig) {
return true;
}
return false;
}
}, {
key: "componentDidMount",
value: function () {
var _componentDidMount = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee2() {
var contextIdentifierProvider;
return _regenerator.default.wrap(function _callee2$(_context2) {
while (1) switch (_context2.prev = _context2.next) {
case 0:
this.handleNewNode(this.props);
contextIdentifierProvider = this.props.contextIdentifierProvider;
_context2.t0 = this;
_context2.next = 5;
return contextIdentifierProvider;
case 5:
_context2.t1 = _context2.sent;
_context2.t2 = {
contextIdentifierProvider: _context2.t1
};
_context2.t0.setState.call(_context2.t0, _context2.t2);
_context2.next = 10;
return this.setViewMediaClientConfig();
case 10:
case "end":
return _context2.stop();
}
}, _callee2, this);
}));
function componentDidMount() {
return _componentDidMount.apply(this, arguments);
}
return componentDidMount;
}()
}, {
key: "componentWillUnmount",
value: function componentWillUnmount() {
var _this$mediaPluginStat3;
var node = this.props.node;
(_this$mediaPluginStat3 = this.mediaPluginState) === null || _this$mediaPluginStat3 === void 0 || _this$mediaPluginStat3.handleMediaNodeUnmount(node);
if (this.unbindKeyDown && typeof this.unbindKeyDown === 'function') {
this.unbindKeyDown();
}
}
}, {
key: "componentDidUpdate",
value: function componentDidUpdate(prevProps) {
var _this$mediaPluginStat5;
if (prevProps.node.attrs.id !== this.props.node.attrs.id) {
var _this$mediaPluginStat4;
(_this$mediaPluginStat4 = this.mediaPluginState) === null || _this$mediaPluginStat4 === void 0 || _this$mediaPluginStat4.handleMediaNodeUnmount(prevProps.node);
this.handleNewNode(this.props);
}
(_this$mediaPluginStat5 = this.mediaPluginState) === null || _this$mediaPluginStat5 === void 0 || _this$mediaPluginStat5.updateElement();
this.setViewMediaClientConfig();
// this.videoControlsWrapperRef is null on componentDidMount. We need to wait until it has value
if (this.videoControlsWrapperRef && this.videoControlsWrapperRef.current) {
var _this$mediaPluginStat6;
if (!((_this$mediaPluginStat6 = this.mediaPluginState) !== null && _this$mediaPluginStat6 !== void 0 && _this$mediaPluginStat6.videoControlsWrapperRef)) {
var _this$mediaPluginStat7;
this.bindKeydown();
(_this$mediaPluginStat7 = this.mediaPluginState) === null || _this$mediaPluginStat7 === void 0 || _this$mediaPluginStat7.updateAndDispatch({
videoControlsWrapperRef: this.videoControlsWrapperRef.current
});
}
}
}
}, {
key: "bindKeydown",
value: function bindKeydown() {
var _this2 = this,
_this$videoControlsWr;
var onKeydown = function onKeydown(event) {
if (event.key === 'Tab') {
var _this2$videoControlsW;
// Add focus trap for controls panel
var firstElement;
var lastElement;
var focusableElements = (_this2$videoControlsW = _this2.videoControlsWrapperRef) === null || _this2$videoControlsW === void 0 || (_this2$videoControlsW = _this2$videoControlsW.current) === null || _this2$videoControlsW === void 0 ? void 0 : _this2$videoControlsW.querySelectorAll('button, input, [tabindex]:not([tabindex="-1"])');
if (focusableElements && focusableElements.length) {
// Ignored via go/ees005
// eslint-disable-next-line @atlaskit/editor/no-as-casting
firstElement = focusableElements[0];
// Ignored via go/ees005
// eslint-disable-next-line @atlaskit/editor/no-as-casting
lastElement = focusableElements[focusableElements.length - 1];
if (event.shiftKey && document.activeElement === firstElement) {
event.preventDefault();
lastElement.focus();
} else if (!event.shiftKey && document.activeElement === lastElement) {
var _firstElement;
event.preventDefault();
(_firstElement = firstElement) === null || _firstElement === void 0 || _firstElement.focus();
}
}
}
};
if ((_this$videoControlsWr = this.videoControlsWrapperRef) !== null && _this$videoControlsWr !== void 0 && _this$videoControlsWr.current) {
this.unbindKeyDown = (0, _bindEventListener.bind)(this.videoControlsWrapperRef.current, {
type: 'keydown',
listener: onKeydown,
options: {
capture: true,
passive: false
}
});
}
}
}, {
key: "render",
value: function render() {
var _this$props2 = this.props,
node = _this$props2.node,
selected = _this$props2.selected,
originalDimensions = _this$props2.originalDimensions,
isLoading = _this$props2.isLoading,
maxDimensions = _this$props2.maxDimensions,
mediaOptions = _this$props2.mediaOptions;
var borderMark = node.marks.find(function (m) {
return m.type.name === 'border';
});
var _this$state = this.state,
viewMediaClientConfig = _this$state.viewMediaClientConfig,
viewAndUploadMediaClientConfig = _this$state.viewAndUploadMediaClientConfig,
contextIdentifierProvider = _this$state.contextIdentifierProvider;
var _node$attrs = node.attrs,
id = _node$attrs.id,
type = _node$attrs.type,
collection = _node$attrs.collection,
url = _node$attrs.url,
alt = _node$attrs.alt;
// Check if we have any media client config available (syncProvider, state, or upload config)
var hasNoMediaClientConfig = !viewMediaClientConfig && ((0, _platformFeatureFlags.fg)('platform_media_video_captions') ? !viewAndUploadMediaClientConfig : true);
if (isLoading || type !== 'external' && hasNoMediaClientConfig) {
if ((0, _expValEquals.expValEquals)('platform_editor_media_vc_fixes', 'isEnabled', true)) {
return /*#__PURE__*/_react.default.createElement(_styles.MediaCardWrapper, {
dimensions: originalDimensions,
borderWidth: borderMark === null || borderMark === void 0 ? void 0 : borderMark.attrs.size,
selected: selected
}, /*#__PURE__*/_react.default.createElement(_mediaCard.CardLoading, {
interactionName: "editor-media-card-loading"
}));
}
return /*#__PURE__*/_react.default.createElement(_styles.MediaCardWrapper, {
dimensions: originalDimensions
}, /*#__PURE__*/_react.default.createElement(_mediaCard.CardLoading, {
interactionName: "editor-media-card-loading"
}));
}
var contextId = contextIdentifierProvider && contextIdentifierProvider.objectId;
var identifier = type === 'external' ? {
// Ignored via go/ees005
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
dataURI: url,
name: url,
mediaItemType: 'external-image'
} : {
id: id,
mediaItemType: 'file',
// Ignored via go/ees005
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
collectionName: collection
};
var resolvedViewAndUploadMediaClientConfig = (0, _platformFeatureFlags.fg)('platform_media_video_captions') ? viewAndUploadMediaClientConfig : undefined;
// mediaClientConfig is not needed for "external" case. So we have to cheat here.
// there is a possibility mediaClientConfig will be part of a identifier,
// so this might be not an issue
var mediaClientConfig = resolvedViewAndUploadMediaClientConfig || viewMediaClientConfig || {
// Ignored via go/ees005
// eslint-disable-next-line @typescript-eslint/no-explicit-any
authProvider: function authProvider() {
return {};
}
};
var ssr = process.env.REACT_SSR ? 'server' : 'client';
return /*#__PURE__*/_react.default.createElement(_styles.MediaCardWrapper, {
dimensions: originalDimensions,
onContextMenu: this.selectMediaSingle,
borderWidth: borderMark === null || borderMark === void 0 ? void 0 : borderMark.attrs.size,
selected: selected
}, /*#__PURE__*/_react.default.createElement(_analyticsNext.AnalyticsContext
// eslint-disable-next-line @atlassian/perf-linting/no-unstable-inline-props -- Ignored via go/ees017 (to be fixed)
, {
data: (0, _defineProperty2.default)({}, _analyticsNamespacedContext.MEDIA_CONTEXT, {
border: !!borderMark
})
}, /*#__PURE__*/_react.default.createElement(_mediaCard.Card, {
mediaClientConfig: mediaClientConfig,
resizeMode: "stretchy-fit",
dimensions: maxDimensions,
originalDimensions: originalDimensions,
identifier: identifier,
selectable: true,
selected: selected,
disableOverlay: true,
onFullscreenChange: this.onFullscreenChange,
onClick: this.selectMediaSingleFromCard,
useInlinePlayer: mediaOptions && mediaOptions.allowLazyLoading,
isLazy: mediaOptions && mediaOptions.allowLazyLoading,
featureFlags: mediaOptions && mediaOptions.featureFlags,
contextId: contextId,
alt: alt,
videoControlsWrapperRef: this.videoControlsWrapperRef,
ssr: ssr,
mediaSettings: this.getMediaSettings(viewAndUploadMediaClientConfig),
isAIGenerating: !!this.props.isAIGenerating,
onError: (0, _expValEquals.expValEquals)('platform_editor_media_error_analytics', 'isEnabled', true) ? this.onError : undefined
})));
}
}]);
}(_react.Component);
var _default_1 = (0, _utils.withImageLoader)(MediaNode);
var _default = exports.default = _default_1;