[Glitch] Add redraft function
Port bd0791d800902cdbdf6666e65d201df3ec22789c to glitch-soc
This commit is contained in:
		
							parent
							
								
									c08a249558
								
							
						
					
					
						commit
						a01faecfcb
					
				@ -23,6 +23,8 @@ export const STATUS_UNMUTE_REQUEST = 'STATUS_UNMUTE_REQUEST';
 | 
				
			|||||||
export const STATUS_UNMUTE_SUCCESS = 'STATUS_UNMUTE_SUCCESS';
 | 
					export const STATUS_UNMUTE_SUCCESS = 'STATUS_UNMUTE_SUCCESS';
 | 
				
			||||||
export const STATUS_UNMUTE_FAIL    = 'STATUS_UNMUTE_FAIL';
 | 
					export const STATUS_UNMUTE_FAIL    = 'STATUS_UNMUTE_FAIL';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export const REDRAFT = 'REDRAFT';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export function fetchStatusRequest(id, skipLoading) {
 | 
					export function fetchStatusRequest(id, skipLoading) {
 | 
				
			||||||
  return {
 | 
					  return {
 | 
				
			||||||
    type: STATUS_FETCH_REQUEST,
 | 
					    type: STATUS_FETCH_REQUEST,
 | 
				
			||||||
@ -70,13 +72,26 @@ export function fetchStatusFail(id, error, skipLoading) {
 | 
				
			|||||||
  };
 | 
					  };
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export function deleteStatus(id) {
 | 
					export function redraft(status) {
 | 
				
			||||||
 | 
					  return {
 | 
				
			||||||
 | 
					    type: REDRAFT,
 | 
				
			||||||
 | 
					    status,
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export function deleteStatus(id, withRedraft = false) {
 | 
				
			||||||
  return (dispatch, getState) => {
 | 
					  return (dispatch, getState) => {
 | 
				
			||||||
 | 
					    const status = getState().getIn(['statuses', id]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    dispatch(deleteStatusRequest(id));
 | 
					    dispatch(deleteStatusRequest(id));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    api(getState).delete(`/api/v1/statuses/${id}`).then(() => {
 | 
					    api(getState).delete(`/api/v1/statuses/${id}`).then(() => {
 | 
				
			||||||
      dispatch(deleteStatusSuccess(id));
 | 
					      dispatch(deleteStatusSuccess(id));
 | 
				
			||||||
      dispatch(deleteFromTimelines(id));
 | 
					      dispatch(deleteFromTimelines(id));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if (withRedraft) {
 | 
				
			||||||
 | 
					        dispatch(redraft(status));
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
    }).catch(error => {
 | 
					    }).catch(error => {
 | 
				
			||||||
      dispatch(deleteStatusFail(id, error));
 | 
					      dispatch(deleteStatusFail(id, error));
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
				
			|||||||
@ -10,6 +10,7 @@ import RelativeTimestamp from './relative_timestamp';
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const messages = defineMessages({
 | 
					const messages = defineMessages({
 | 
				
			||||||
  delete: { id: 'status.delete', defaultMessage: 'Delete' },
 | 
					  delete: { id: 'status.delete', defaultMessage: 'Delete' },
 | 
				
			||||||
 | 
					  redraft: { id: 'status.redraft', defaultMessage: 'Delete & re-draft' },
 | 
				
			||||||
  direct: { id: 'status.direct', defaultMessage: 'Direct message @{name}' },
 | 
					  direct: { id: 'status.direct', defaultMessage: 'Direct message @{name}' },
 | 
				
			||||||
  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
 | 
					  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
 | 
				
			||||||
  mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' },
 | 
					  mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' },
 | 
				
			||||||
@ -92,6 +93,10 @@ export default class StatusActionBar extends ImmutablePureComponent {
 | 
				
			|||||||
    this.props.onDelete(this.props.status);
 | 
					    this.props.onDelete(this.props.status);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  handleRedraftClick = () => {
 | 
				
			||||||
 | 
					    this.props.onDelete(this.props.status, true);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  handlePinClick = () => {
 | 
					  handlePinClick = () => {
 | 
				
			||||||
    this.props.onPin(this.props.status);
 | 
					    this.props.onPin(this.props.status);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@ -161,6 +166,7 @@ export default class StatusActionBar extends ImmutablePureComponent {
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick });
 | 
					      menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick });
 | 
				
			||||||
 | 
					      menu.push({ text: intl.formatMessage(messages.redraft), action: this.handleRedraftClick });
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
 | 
					      menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
 | 
				
			||||||
      menu.push({ text: intl.formatMessage(messages.direct, { name: status.getIn(['account', 'username']) }), action: this.handleDirectClick });
 | 
					      menu.push({ text: intl.formatMessage(messages.direct, { name: status.getIn(['account', 'username']) }), action: this.handleDirectClick });
 | 
				
			||||||
 | 
				
			|||||||
@ -28,6 +28,8 @@ import { boostModal, favouriteModal, deleteModal } from 'flavours/glitch/util/in
 | 
				
			|||||||
const messages = defineMessages({
 | 
					const messages = defineMessages({
 | 
				
			||||||
  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
 | 
					  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
 | 
				
			||||||
  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
 | 
					  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
 | 
				
			||||||
 | 
					  redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' },
 | 
				
			||||||
 | 
					  redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this status and re-draft it? You will lose all replies, boosts and favourites to it.' },
 | 
				
			||||||
  blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' },
 | 
					  blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' },
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -120,14 +122,14 @@ const mapDispatchToProps = (dispatch, { intl }) => ({
 | 
				
			|||||||
    dispatch(openModal('EMBED', { url: status.get('url') }));
 | 
					    dispatch(openModal('EMBED', { url: status.get('url') }));
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  onDelete (status) {
 | 
					  onDelete (status, withRedraft = false) {
 | 
				
			||||||
    if (!deleteModal) {
 | 
					    if (!deleteModal) {
 | 
				
			||||||
      dispatch(deleteStatus(status.get('id')));
 | 
					      dispatch(deleteStatus(status.get('id'), withRedraft));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      dispatch(openModal('CONFIRM', {
 | 
					      dispatch(openModal('CONFIRM', {
 | 
				
			||||||
        message: intl.formatMessage(messages.deleteMessage),
 | 
					        message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage),
 | 
				
			||||||
        confirm: intl.formatMessage(messages.deleteConfirm),
 | 
					        confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm),
 | 
				
			||||||
        onConfirm: () => dispatch(deleteStatus(status.get('id'))),
 | 
					        onConfirm: () => dispatch(deleteStatus(status.get('id'), withRedraft)),
 | 
				
			||||||
      }));
 | 
					      }));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
 | 
				
			|||||||
@ -8,6 +8,7 @@ import { me } from 'flavours/glitch/util/initial_state';
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const messages = defineMessages({
 | 
					const messages = defineMessages({
 | 
				
			||||||
  delete: { id: 'status.delete', defaultMessage: 'Delete' },
 | 
					  delete: { id: 'status.delete', defaultMessage: 'Delete' },
 | 
				
			||||||
 | 
					  redraft: { id: 'status.redraft', defaultMessage: 'Delete & re-draft' },
 | 
				
			||||||
  direct: { id: 'status.direct', defaultMessage: 'Direct message @{name}' },
 | 
					  direct: { id: 'status.direct', defaultMessage: 'Direct message @{name}' },
 | 
				
			||||||
  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
 | 
					  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
 | 
				
			||||||
  reply: { id: 'status.reply', defaultMessage: 'Reply' },
 | 
					  reply: { id: 'status.reply', defaultMessage: 'Reply' },
 | 
				
			||||||
@ -72,6 +73,10 @@ export default class ActionBar extends React.PureComponent {
 | 
				
			|||||||
    this.props.onDelete(this.props.status);
 | 
					    this.props.onDelete(this.props.status);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  handleRedraftClick = () => {
 | 
				
			||||||
 | 
					    this.props.onDelete(this.props.status, true);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  handleDirectClick = () => {
 | 
					  handleDirectClick = () => {
 | 
				
			||||||
    this.props.onDirect(this.props.status.get('account'), this.context.router.history);
 | 
					    this.props.onDirect(this.props.status.get('account'), this.context.router.history);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@ -133,6 +138,7 @@ export default class ActionBar extends React.PureComponent {
 | 
				
			|||||||
      menu.push({ text: intl.formatMessage(mutingConversation ? messages.unmuteConversation : messages.muteConversation), action: this.handleConversationMuteClick });
 | 
					      menu.push({ text: intl.formatMessage(mutingConversation ? messages.unmuteConversation : messages.muteConversation), action: this.handleConversationMuteClick });
 | 
				
			||||||
      menu.push(null);
 | 
					      menu.push(null);
 | 
				
			||||||
      menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick });
 | 
					      menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick });
 | 
				
			||||||
 | 
					      menu.push({ text: intl.formatMessage(messages.redraft), action: this.handleRedraftClick });
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
 | 
					      menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
 | 
				
			||||||
      menu.push({ text: intl.formatMessage(messages.direct, { name: status.getIn(['account', 'username']) }), action: this.handleDirectClick });
 | 
					      menu.push({ text: intl.formatMessage(messages.direct, { name: status.getIn(['account', 'username']) }), action: this.handleDirectClick });
 | 
				
			||||||
 | 
				
			|||||||
@ -42,6 +42,8 @@ import { attachFullscreenListener, detachFullscreenListener, isFullscreen } from
 | 
				
			|||||||
const messages = defineMessages({
 | 
					const messages = defineMessages({
 | 
				
			||||||
  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
 | 
					  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
 | 
				
			||||||
  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
 | 
					  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
 | 
				
			||||||
 | 
					  redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' },
 | 
				
			||||||
 | 
					  redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this status and re-draft it? You will lose all replies, boosts and favourites to it.' },
 | 
				
			||||||
  blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' },
 | 
					  blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' },
 | 
				
			||||||
  revealAll: { id: 'status.show_more_all', defaultMessage: 'Show more for all' },
 | 
					  revealAll: { id: 'status.show_more_all', defaultMessage: 'Show more for all' },
 | 
				
			||||||
  hideAll: { id: 'status.show_less_all', defaultMessage: 'Show less for all' },
 | 
					  hideAll: { id: 'status.show_less_all', defaultMessage: 'Show less for all' },
 | 
				
			||||||
@ -157,16 +159,16 @@ export default class Status extends ImmutablePureComponent {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  handleDeleteClick = (status) => {
 | 
					  handleDeleteClick = (status, withRedraft = false) => {
 | 
				
			||||||
    const { dispatch, intl } = this.props;
 | 
					    const { dispatch, intl } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!deleteModal) {
 | 
					    if (!deleteModal) {
 | 
				
			||||||
      dispatch(deleteStatus(status.get('id')));
 | 
					      dispatch(deleteStatus(status.get('id'), withRedraft));
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
      dispatch(openModal('CONFIRM', {
 | 
					      dispatch(openModal('CONFIRM', {
 | 
				
			||||||
        message: intl.formatMessage(messages.deleteMessage),
 | 
					        message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage),
 | 
				
			||||||
        confirm: intl.formatMessage(messages.deleteConfirm),
 | 
					        confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm),
 | 
				
			||||||
        onConfirm: () => dispatch(deleteStatus(status.get('id'))),
 | 
					        onConfirm: () => dispatch(deleteStatus(status.get('id'), withRedraft)),
 | 
				
			||||||
      }));
 | 
					      }));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
				
			|||||||
@ -32,6 +32,7 @@ import {
 | 
				
			|||||||
} from 'flavours/glitch/actions/compose';
 | 
					} from 'flavours/glitch/actions/compose';
 | 
				
			||||||
import { TIMELINE_DELETE } from 'flavours/glitch/actions/timelines';
 | 
					import { TIMELINE_DELETE } from 'flavours/glitch/actions/timelines';
 | 
				
			||||||
import { STORE_HYDRATE } from 'flavours/glitch/actions/store';
 | 
					import { STORE_HYDRATE } from 'flavours/glitch/actions/store';
 | 
				
			||||||
 | 
					import { REDRAFT } from 'flavours/glitch/actions/statuses';
 | 
				
			||||||
import { Map as ImmutableMap, List as ImmutableList, OrderedSet as ImmutableOrderedSet, fromJS } from 'immutable';
 | 
					import { Map as ImmutableMap, List as ImmutableList, OrderedSet as ImmutableOrderedSet, fromJS } from 'immutable';
 | 
				
			||||||
import uuid from 'flavours/glitch/util/uuid';
 | 
					import uuid from 'flavours/glitch/util/uuid';
 | 
				
			||||||
import { me } from 'flavours/glitch/util/initial_state';
 | 
					import { me } from 'flavours/glitch/util/initial_state';
 | 
				
			||||||
@ -226,6 +227,18 @@ const hydrate = (state, hydratedState) => {
 | 
				
			|||||||
  return state;
 | 
					  return state;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const domParser = new DOMParser();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const htmlToText = status => {
 | 
				
			||||||
 | 
					  const fragment = domParser.parseFromString(status.get('content'), 'text/html').documentElement;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  status.get('mentions').forEach(mention => {
 | 
				
			||||||
 | 
					    fragment.querySelector(`a[href="${mention.get('url')}"]`).textContent = `@${mention.get('acct')}`;
 | 
				
			||||||
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return fragment.textContent;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default function compose(state = initialState, action) {
 | 
					export default function compose(state = initialState, action) {
 | 
				
			||||||
  switch(action.type) {
 | 
					  switch(action.type) {
 | 
				
			||||||
  case STORE_HYDRATE:
 | 
					  case STORE_HYDRATE:
 | 
				
			||||||
@ -366,6 +379,24 @@ export default function compose(state = initialState, action) {
 | 
				
			|||||||
      }));
 | 
					      }));
 | 
				
			||||||
  case COMPOSE_DOODLE_SET:
 | 
					  case COMPOSE_DOODLE_SET:
 | 
				
			||||||
    return state.mergeIn(['doodle'], action.options);
 | 
					    return state.mergeIn(['doodle'], action.options);
 | 
				
			||||||
 | 
					  case REDRAFT:
 | 
				
			||||||
 | 
					    return state.withMutations(map => {
 | 
				
			||||||
 | 
					      map.set('text', htmlToText(action.status));
 | 
				
			||||||
 | 
					      map.set('in_reply_to', action.status.get('in_reply_to_id'));
 | 
				
			||||||
 | 
					      map.set('privacy', action.status.get('visibility'));
 | 
				
			||||||
 | 
					      map.set('media_attachments', action.status.get('media_attachments'));
 | 
				
			||||||
 | 
					      map.set('focusDate', new Date());
 | 
				
			||||||
 | 
					      map.set('caretPosition', null);
 | 
				
			||||||
 | 
					      map.set('idempotencyKey', uuid());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if (action.status.get('spoiler_text').length > 0) {
 | 
				
			||||||
 | 
					        map.set('spoiler', true);
 | 
				
			||||||
 | 
					        map.set('spoiler_text', action.status.get('spoiler_text'));
 | 
				
			||||||
 | 
					      } else {
 | 
				
			||||||
 | 
					        map.set('spoiler', false);
 | 
				
			||||||
 | 
					        map.set('spoiler_text', '');
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
  default:
 | 
					  default:
 | 
				
			||||||
    return state;
 | 
					    return state;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user