forked from treehouse/mastodon
Rework search
parent
553e6dd07c
commit
b4046c5957
|
@ -1,9 +1,12 @@
|
||||||
import api from '../api'
|
import api from '../api'
|
||||||
|
|
||||||
export const SEARCH_CHANGE = 'SEARCH_CHANGE';
|
export const SEARCH_CHANGE = 'SEARCH_CHANGE';
|
||||||
export const SEARCH_SUGGESTIONS_CLEAR = 'SEARCH_SUGGESTIONS_CLEAR';
|
export const SEARCH_CLEAR = 'SEARCH_CLEAR';
|
||||||
export const SEARCH_SUGGESTIONS_READY = 'SEARCH_SUGGESTIONS_READY';
|
export const SEARCH_SHOW = 'SEARCH_SHOW';
|
||||||
export const SEARCH_RESET = 'SEARCH_RESET';
|
|
||||||
|
export const SEARCH_FETCH_REQUEST = 'SEARCH_FETCH_REQUEST';
|
||||||
|
export const SEARCH_FETCH_SUCCESS = 'SEARCH_FETCH_SUCCESS';
|
||||||
|
export const SEARCH_FETCH_FAIL = 'SEARCH_FETCH_FAIL';
|
||||||
|
|
||||||
export function changeSearch(value) {
|
export function changeSearch(value) {
|
||||||
return {
|
return {
|
||||||
|
@ -12,42 +15,55 @@ export function changeSearch(value) {
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export function clearSearchSuggestions() {
|
export function clearSearch() {
|
||||||
return {
|
return {
|
||||||
type: SEARCH_SUGGESTIONS_CLEAR
|
type: SEARCH_CLEAR
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export function readySearchSuggestions(value, { accounts, hashtags, statuses }) {
|
export function submitSearch() {
|
||||||
return {
|
|
||||||
type: SEARCH_SUGGESTIONS_READY,
|
|
||||||
value,
|
|
||||||
accounts,
|
|
||||||
hashtags,
|
|
||||||
statuses
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export function fetchSearchSuggestions(value) {
|
|
||||||
return (dispatch, getState) => {
|
return (dispatch, getState) => {
|
||||||
if (getState().getIn(['search', 'loaded_value']) === value) {
|
const value = getState().getIn(['search', 'value']);
|
||||||
return;
|
|
||||||
}
|
dispatch(fetchSearchRequest());
|
||||||
|
|
||||||
api(getState).get('/api/v1/search', {
|
api(getState).get('/api/v1/search', {
|
||||||
params: {
|
params: {
|
||||||
q: value,
|
q: value,
|
||||||
resolve: true,
|
resolve: true
|
||||||
limit: 4
|
|
||||||
}
|
}
|
||||||
}).then(response => {
|
}).then(response => {
|
||||||
dispatch(readySearchSuggestions(value, response.data));
|
dispatch(fetchSearchSuccess(response.data));
|
||||||
|
}).catch(error => {
|
||||||
|
dispatch(fetchSearchFail(error));
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export function resetSearch() {
|
export function fetchSearchRequest() {
|
||||||
return {
|
return {
|
||||||
type: SEARCH_RESET
|
type: SEARCH_FETCH_REQUEST
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function fetchSearchSuccess(results) {
|
||||||
|
return {
|
||||||
|
type: SEARCH_FETCH_SUCCESS,
|
||||||
|
results,
|
||||||
|
accounts: results.accounts,
|
||||||
|
statuses: results.statuses
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function fetchSearchFail(error) {
|
||||||
|
return {
|
||||||
|
type: SEARCH_FETCH_FAIL,
|
||||||
|
error
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function showSearch() {
|
||||||
|
return {
|
||||||
|
type: SEARCH_SHOW
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,44 +0,0 @@
|
||||||
import { Link } from 'react-router';
|
|
||||||
import { injectIntl, defineMessages } from 'react-intl';
|
|
||||||
|
|
||||||
const messages = defineMessages({
|
|
||||||
start: { id: 'getting_started.heading', defaultMessage: 'Getting started' },
|
|
||||||
public: { id: 'navigation_bar.public_timeline', defaultMessage: 'Whole Known Network' },
|
|
||||||
community: { id: 'navigation_bar.community_timeline', defaultMessage: 'Local timeline' },
|
|
||||||
preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
|
|
||||||
logout: { id: 'navigation_bar.logout', defaultMessage: 'Logout' }
|
|
||||||
});
|
|
||||||
|
|
||||||
const Drawer = ({ children, withHeader, intl }) => {
|
|
||||||
let header = '';
|
|
||||||
|
|
||||||
if (withHeader) {
|
|
||||||
header = (
|
|
||||||
<div className='drawer__header'>
|
|
||||||
<Link title={intl.formatMessage(messages.start)} className='drawer__tab' to='/getting-started'><i className='fa fa-fw fa-asterisk' /></Link>
|
|
||||||
<Link title={intl.formatMessage(messages.community)} className='drawer__tab' to='/timelines/public/local'><i className='fa fa-fw fa-users' /></Link>
|
|
||||||
<Link title={intl.formatMessage(messages.public)} className='drawer__tab' to='/timelines/public'><i className='fa fa-fw fa-globe' /></Link>
|
|
||||||
<a title={intl.formatMessage(messages.preferences)} className='drawer__tab' href='/settings/preferences'><i className='fa fa-fw fa-cog' /></a>
|
|
||||||
<a title={intl.formatMessage(messages.logout)} className='drawer__tab' href='/auth/sign_out' data-method='delete'><i className='fa fa-fw fa-sign-out' /></a>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className='drawer'>
|
|
||||||
{header}
|
|
||||||
|
|
||||||
<div className='drawer__inner'>
|
|
||||||
{children}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
Drawer.propTypes = {
|
|
||||||
withHeader: React.PropTypes.bool,
|
|
||||||
children: React.PropTypes.node,
|
|
||||||
intl: React.PropTypes.object
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(Drawer);
|
|
|
@ -1,123 +1,67 @@
|
||||||
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
||||||
import ImmutablePropTypes from 'react-immutable-proptypes';
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
||||||
import Autosuggest from 'react-autosuggest';
|
|
||||||
import AutosuggestAccountContainer from '../containers/autosuggest_account_container';
|
|
||||||
import AutosuggestStatusContainer from '../containers/autosuggest_status_container';
|
|
||||||
import { debounce } from 'react-decoration';
|
|
||||||
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
placeholder: { id: 'search.placeholder', defaultMessage: 'Search' }
|
placeholder: { id: 'search.placeholder', defaultMessage: 'Search' }
|
||||||
});
|
});
|
||||||
|
|
||||||
const getSuggestionValue = suggestion => suggestion.value;
|
|
||||||
|
|
||||||
const renderSuggestion = suggestion => {
|
|
||||||
if (suggestion.type === 'account') {
|
|
||||||
return <AutosuggestAccountContainer id={suggestion.id} />;
|
|
||||||
} else if (suggestion.type === 'hashtag') {
|
|
||||||
return <span>#{suggestion.id}</span>;
|
|
||||||
} else {
|
|
||||||
return <AutosuggestStatusContainer id={suggestion.id} />;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const renderSectionTitle = section => (
|
|
||||||
<strong><FormattedMessage id={`search.${section.title}`} defaultMessage={section.title} /></strong>
|
|
||||||
);
|
|
||||||
|
|
||||||
const getSectionSuggestions = section => section.items;
|
|
||||||
|
|
||||||
const outerStyle = {
|
|
||||||
padding: '10px',
|
|
||||||
lineHeight: '20px',
|
|
||||||
position: 'relative'
|
|
||||||
};
|
|
||||||
|
|
||||||
const iconStyle = {
|
|
||||||
position: 'absolute',
|
|
||||||
top: '18px',
|
|
||||||
right: '20px',
|
|
||||||
fontSize: '18px',
|
|
||||||
pointerEvents: 'none'
|
|
||||||
};
|
|
||||||
|
|
||||||
const Search = React.createClass({
|
const Search = React.createClass({
|
||||||
|
|
||||||
contextTypes: {
|
|
||||||
router: React.PropTypes.object
|
|
||||||
},
|
|
||||||
|
|
||||||
propTypes: {
|
propTypes: {
|
||||||
suggestions: React.PropTypes.array.isRequired,
|
|
||||||
value: React.PropTypes.string.isRequired,
|
value: React.PropTypes.string.isRequired,
|
||||||
onChange: React.PropTypes.func.isRequired,
|
onChange: React.PropTypes.func.isRequired,
|
||||||
|
onSubmit: React.PropTypes.func.isRequired,
|
||||||
onClear: React.PropTypes.func.isRequired,
|
onClear: React.PropTypes.func.isRequired,
|
||||||
onFetch: React.PropTypes.func.isRequired,
|
onShow: React.PropTypes.func.isRequired,
|
||||||
onReset: React.PropTypes.func.isRequired,
|
|
||||||
intl: React.PropTypes.object.isRequired
|
intl: React.PropTypes.object.isRequired
|
||||||
},
|
},
|
||||||
|
|
||||||
mixins: [PureRenderMixin],
|
mixins: [PureRenderMixin],
|
||||||
|
|
||||||
onChange (_, { newValue }) {
|
handleChange (e) {
|
||||||
if (typeof newValue !== 'string') {
|
this.props.onChange(e.target.value);
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.props.onChange(newValue);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
onSuggestionsClearRequested () {
|
handleClear (e) {
|
||||||
|
e.preventDefault();
|
||||||
this.props.onClear();
|
this.props.onClear();
|
||||||
},
|
},
|
||||||
|
|
||||||
@debounce(500)
|
handleKeyDown (e) {
|
||||||
onSuggestionsFetchRequested ({ value }) {
|
if (e.key === 'Enter') {
|
||||||
value = value.replace('#', '');
|
e.preventDefault();
|
||||||
this.props.onFetch(value.trim());
|
this.props.onSubmit();
|
||||||
},
|
|
||||||
|
|
||||||
onSuggestionSelected (_, { suggestion }) {
|
|
||||||
if (suggestion.type === 'account') {
|
|
||||||
this.context.router.push(`/accounts/${suggestion.id}`);
|
|
||||||
} else if(suggestion.type === 'hashtag') {
|
|
||||||
this.context.router.push(`/timelines/tag/${suggestion.id}`);
|
|
||||||
} else {
|
|
||||||
this.context.router.push(`/statuses/${suggestion.id}`);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
handleFocus () {
|
||||||
|
this.props.onShow();
|
||||||
|
},
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
const inputProps = {
|
const { intl, value } = this.props;
|
||||||
placeholder: this.props.intl.formatMessage(messages.placeholder),
|
const hasValue = value.length > 0;
|
||||||
value: this.props.value,
|
|
||||||
onChange: this.onChange,
|
|
||||||
className: 'search__input'
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='search' style={outerStyle}>
|
<div className='search'>
|
||||||
<Autosuggest
|
<input
|
||||||
multiSection={true}
|
className='search__input'
|
||||||
suggestions={this.props.suggestions}
|
type='text'
|
||||||
focusFirstSuggestion={true}
|
placeholder={intl.formatMessage(messages.placeholder)}
|
||||||
focusInputOnSuggestionClick={false}
|
value={value}
|
||||||
alwaysRenderSuggestions={false}
|
onChange={this.handleChange}
|
||||||
onSuggestionsFetchRequested={this.onSuggestionsFetchRequested}
|
onKeyUp={this.handleKeyDown}
|
||||||
onSuggestionsClearRequested={this.onSuggestionsClearRequested}
|
onFocus={this.handleFocus}
|
||||||
onSuggestionSelected={this.onSuggestionSelected}
|
|
||||||
getSuggestionValue={getSuggestionValue}
|
|
||||||
renderSuggestion={renderSuggestion}
|
|
||||||
renderSectionTitle={renderSectionTitle}
|
|
||||||
getSectionSuggestions={getSectionSuggestions}
|
|
||||||
inputProps={inputProps}
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div style={iconStyle}><i className='fa fa-search' /></div>
|
<div className='search__icon'>
|
||||||
|
<i className={`fa fa-search ${hasValue ? '' : 'active'}`} />
|
||||||
|
<i className={`fa fa-times-circle ${hasValue ? 'active' : ''}`} onClick={this.handleClear} />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,68 @@
|
||||||
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
||||||
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
||||||
|
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
||||||
|
import AccountContainer from '../../../containers/account_container';
|
||||||
|
import StatusContainer from '../../../containers/status_container';
|
||||||
|
import { Link } from 'react-router';
|
||||||
|
|
||||||
|
const SearchResults = React.createClass({
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
results: ImmutablePropTypes.map.isRequired
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [PureRenderMixin],
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const { results } = this.props;
|
||||||
|
|
||||||
|
let accounts, statuses, hashtags;
|
||||||
|
let count = 0;
|
||||||
|
|
||||||
|
if (results.get('accounts') && results.get('accounts').size > 0) {
|
||||||
|
count += results.get('accounts').size;
|
||||||
|
accounts = (
|
||||||
|
<div className='search-results__section'>
|
||||||
|
{results.get('accounts').map(accountId => <AccountContainer key={accountId} id={accountId} />)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (results.get('statuses') && results.get('statuses').size > 0) {
|
||||||
|
count += results.get('statuses').size;
|
||||||
|
statuses = (
|
||||||
|
<div className='search-results__section'>
|
||||||
|
{results.get('statuses').map(statusId => <StatusContainer key={statusId} id={statusId} />)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (results.get('hashtags') && results.get('hashtags').size > 0) {
|
||||||
|
count += results.get('hashtags').size;
|
||||||
|
hashtags = (
|
||||||
|
<div className='search-results__section'>
|
||||||
|
{results.get('hashtags').map(hashtag =>
|
||||||
|
<Link className='search-results__hashtag' to={`/timelines/tag/${hashtag}`}>
|
||||||
|
#{hashtag}
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className='search-results'>
|
||||||
|
<div className='search-results__header'>
|
||||||
|
<FormattedMessage id='search_results.total' defaultMessage='{count} {count, plural, one {result} other {results}}' values={{ count }} />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{accounts}
|
||||||
|
{statuses}
|
||||||
|
{hashtags}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default SearchResults;
|
|
@ -1,31 +0,0 @@
|
||||||
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
|
||||||
import { FormattedMessage } from 'react-intl';
|
|
||||||
import Toggle from 'react-toggle';
|
|
||||||
import Collapsable from '../../../components/collapsable';
|
|
||||||
|
|
||||||
const SensitiveToggle = React.createClass({
|
|
||||||
|
|
||||||
propTypes: {
|
|
||||||
hasMedia: React.PropTypes.bool,
|
|
||||||
isSensitive: React.PropTypes.bool,
|
|
||||||
onChange: React.PropTypes.func.isRequired
|
|
||||||
},
|
|
||||||
|
|
||||||
mixins: [PureRenderMixin],
|
|
||||||
|
|
||||||
render () {
|
|
||||||
const { hasMedia, isSensitive, onChange } = this.props;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Collapsable isVisible={hasMedia} fullHeight={39.5}>
|
|
||||||
<label className='compose-form__label'>
|
|
||||||
<Toggle checked={isSensitive} onChange={onChange} />
|
|
||||||
<span className='compose-form__label__text'><FormattedMessage id='compose_form.sensitive' defaultMessage='Mark media as sensitive' /></span>
|
|
||||||
</label>
|
|
||||||
</Collapsable>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
export default SensitiveToggle;
|
|
|
@ -1,27 +0,0 @@
|
||||||
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
|
||||||
import { FormattedMessage } from 'react-intl';
|
|
||||||
import Toggle from 'react-toggle';
|
|
||||||
|
|
||||||
const SpoilerToggle = React.createClass({
|
|
||||||
|
|
||||||
propTypes: {
|
|
||||||
isSpoiler: React.PropTypes.bool,
|
|
||||||
onChange: React.PropTypes.func.isRequired
|
|
||||||
},
|
|
||||||
|
|
||||||
mixins: [PureRenderMixin],
|
|
||||||
|
|
||||||
render () {
|
|
||||||
const { isSpoiler, onChange } = this.props;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<label className='compose-form__label with-border' style={{ marginTop: '10px' }}>
|
|
||||||
<Toggle checked={isSpoiler} onChange={onChange} />
|
|
||||||
<span className='compose-form__label__text'><FormattedMessage id='compose_form.spoiler' defaultMessage='Hide text behind warning' /></span>
|
|
||||||
</label>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
export default SpoilerToggle;
|
|
|
@ -1,14 +1,13 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import {
|
import {
|
||||||
changeSearch,
|
changeSearch,
|
||||||
clearSearchSuggestions,
|
clearSearch,
|
||||||
fetchSearchSuggestions,
|
submitSearch,
|
||||||
resetSearch
|
showSearch
|
||||||
} from '../../../actions/search';
|
} from '../../../actions/search';
|
||||||
import Search from '../components/search';
|
import Search from '../components/search';
|
||||||
|
|
||||||
const mapStateToProps = state => ({
|
const mapStateToProps = state => ({
|
||||||
suggestions: state.getIn(['search', 'suggestions']),
|
|
||||||
value: state.getIn(['search', 'value'])
|
value: state.getIn(['search', 'value'])
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -19,15 +18,15 @@ const mapDispatchToProps = dispatch => ({
|
||||||
},
|
},
|
||||||
|
|
||||||
onClear () {
|
onClear () {
|
||||||
dispatch(clearSearchSuggestions());
|
dispatch(clearSearch());
|
||||||
},
|
},
|
||||||
|
|
||||||
onFetch (value) {
|
onSubmit () {
|
||||||
dispatch(fetchSearchSuggestions(value));
|
dispatch(submitSearch());
|
||||||
},
|
},
|
||||||
|
|
||||||
onReset () {
|
onShow () {
|
||||||
dispatch(resetSearch());
|
dispatch(showSearch());
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
import SearchResults from '../components/search_results';
|
||||||
|
|
||||||
|
const mapStateToProps = state => ({
|
||||||
|
results: state.getIn(['search', 'results'])
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(mapStateToProps)(SearchResults);
|
|
@ -1,17 +1,34 @@
|
||||||
import Drawer from './components/drawer';
|
|
||||||
import ComposeFormContainer from './containers/compose_form_container';
|
import ComposeFormContainer from './containers/compose_form_container';
|
||||||
import UploadFormContainer from './containers/upload_form_container';
|
import UploadFormContainer from './containers/upload_form_container';
|
||||||
import NavigationContainer from './containers/navigation_container';
|
import NavigationContainer from './containers/navigation_container';
|
||||||
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
||||||
import SearchContainer from './containers/search_container';
|
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { mountCompose, unmountCompose } from '../../actions/compose';
|
import { mountCompose, unmountCompose } from '../../actions/compose';
|
||||||
|
import { Link } from 'react-router';
|
||||||
|
import { injectIntl, defineMessages } from 'react-intl';
|
||||||
|
import SearchContainer from './containers/search_container';
|
||||||
|
import { Motion, spring } from 'react-motion';
|
||||||
|
import SearchResultsContainer from './containers/search_results_container';
|
||||||
|
|
||||||
|
const messages = defineMessages({
|
||||||
|
start: { id: 'getting_started.heading', defaultMessage: 'Getting started' },
|
||||||
|
public: { id: 'navigation_bar.public_timeline', defaultMessage: 'Whole Known Network' },
|
||||||
|
community: { id: 'navigation_bar.community_timeline', defaultMessage: 'Local timeline' },
|
||||||
|
preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
|
||||||
|
logout: { id: 'navigation_bar.logout', defaultMessage: 'Logout' }
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapStateToProps = state => ({
|
||||||
|
showSearch: state.getIn(['search', 'submitted']) && !state.getIn(['search', 'hidden'])
|
||||||
|
});
|
||||||
|
|
||||||
const Compose = React.createClass({
|
const Compose = React.createClass({
|
||||||
|
|
||||||
propTypes: {
|
propTypes: {
|
||||||
dispatch: React.PropTypes.func.isRequired,
|
dispatch: React.PropTypes.func.isRequired,
|
||||||
withHeader: React.PropTypes.bool
|
withHeader: React.PropTypes.bool,
|
||||||
|
showSearch: React.PropTypes.bool,
|
||||||
|
intl: React.PropTypes.object.isRequired
|
||||||
},
|
},
|
||||||
|
|
||||||
mixins: [PureRenderMixin],
|
mixins: [PureRenderMixin],
|
||||||
|
@ -25,15 +42,46 @@ const Compose = React.createClass({
|
||||||
},
|
},
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
|
const { withHeader, showSearch, intl } = this.props;
|
||||||
|
|
||||||
|
let header = '';
|
||||||
|
|
||||||
|
if (withHeader) {
|
||||||
|
header = (
|
||||||
|
<div className='drawer__header'>
|
||||||
|
<Link title={intl.formatMessage(messages.start)} className='drawer__tab' to='/getting-started'><i className='fa fa-fw fa-asterisk' /></Link>
|
||||||
|
<Link title={intl.formatMessage(messages.community)} className='drawer__tab' to='/timelines/public/local'><i className='fa fa-fw fa-users' /></Link>
|
||||||
|
<Link title={intl.formatMessage(messages.public)} className='drawer__tab' to='/timelines/public'><i className='fa fa-fw fa-globe' /></Link>
|
||||||
|
<a title={intl.formatMessage(messages.preferences)} className='drawer__tab' href='/settings/preferences'><i className='fa fa-fw fa-cog' /></a>
|
||||||
|
<a title={intl.formatMessage(messages.logout)} className='drawer__tab' href='/auth/sign_out' data-method='delete'><i className='fa fa-fw fa-sign-out' /></a>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Drawer withHeader={this.props.withHeader}>
|
<div className='drawer'>
|
||||||
|
{header}
|
||||||
|
|
||||||
<SearchContainer />
|
<SearchContainer />
|
||||||
|
|
||||||
|
<div className='drawer__pager'>
|
||||||
|
<div className='drawer__inner'>
|
||||||
<NavigationContainer />
|
<NavigationContainer />
|
||||||
<ComposeFormContainer />
|
<ComposeFormContainer />
|
||||||
</Drawer>
|
</div>
|
||||||
|
|
||||||
|
<Motion defaultStyle={{ x: -300 }} style={{ x: spring(showSearch ? 0 : -300, { stiffness: 210, damping: 20 }) }}>
|
||||||
|
{({ x }) =>
|
||||||
|
<div className='drawer__inner darker' style={{ transform: `translateX(${x}px)` }}>
|
||||||
|
<SearchResultsContainer />
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</Motion>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export default connect()(Compose);
|
export default connect(mapStateToProps)(injectIntl(Compose));
|
||||||
|
|
|
@ -33,7 +33,7 @@ import {
|
||||||
STATUS_FETCH_SUCCESS,
|
STATUS_FETCH_SUCCESS,
|
||||||
CONTEXT_FETCH_SUCCESS
|
CONTEXT_FETCH_SUCCESS
|
||||||
} from '../actions/statuses';
|
} from '../actions/statuses';
|
||||||
import { SEARCH_SUGGESTIONS_READY } from '../actions/search';
|
import { SEARCH_FETCH_SUCCESS } from '../actions/search';
|
||||||
import {
|
import {
|
||||||
NOTIFICATIONS_UPDATE,
|
NOTIFICATIONS_UPDATE,
|
||||||
NOTIFICATIONS_REFRESH_SUCCESS,
|
NOTIFICATIONS_REFRESH_SUCCESS,
|
||||||
|
@ -97,7 +97,7 @@ export default function accounts(state = initialState, action) {
|
||||||
return normalizeAccounts(state, action.accounts);
|
return normalizeAccounts(state, action.accounts);
|
||||||
case NOTIFICATIONS_REFRESH_SUCCESS:
|
case NOTIFICATIONS_REFRESH_SUCCESS:
|
||||||
case NOTIFICATIONS_EXPAND_SUCCESS:
|
case NOTIFICATIONS_EXPAND_SUCCESS:
|
||||||
case SEARCH_SUGGESTIONS_READY:
|
case SEARCH_FETCH_SUCCESS:
|
||||||
return normalizeAccountsFromStatuses(normalizeAccounts(state, action.accounts), action.statuses);
|
return normalizeAccountsFromStatuses(normalizeAccounts(state, action.accounts), action.statuses);
|
||||||
case TIMELINE_REFRESH_SUCCESS:
|
case TIMELINE_REFRESH_SUCCESS:
|
||||||
case TIMELINE_EXPAND_SUCCESS:
|
case TIMELINE_EXPAND_SUCCESS:
|
||||||
|
|
|
@ -1,14 +1,17 @@
|
||||||
import {
|
import {
|
||||||
SEARCH_CHANGE,
|
SEARCH_CHANGE,
|
||||||
SEARCH_SUGGESTIONS_READY,
|
SEARCH_CLEAR,
|
||||||
SEARCH_RESET
|
SEARCH_FETCH_SUCCESS,
|
||||||
|
SEARCH_SHOW
|
||||||
} from '../actions/search';
|
} from '../actions/search';
|
||||||
|
import { COMPOSE_MENTION, COMPOSE_REPLY } from '../actions/compose';
|
||||||
import Immutable from 'immutable';
|
import Immutable from 'immutable';
|
||||||
|
|
||||||
const initialState = Immutable.Map({
|
const initialState = Immutable.Map({
|
||||||
value: '',
|
value: '',
|
||||||
loaded_value: '',
|
submitted: false,
|
||||||
suggestions: []
|
hidden: false,
|
||||||
|
results: Immutable.Map()
|
||||||
});
|
});
|
||||||
|
|
||||||
const normalizeSuggestions = (state, value, accounts, hashtags, statuses) => {
|
const normalizeSuggestions = (state, value, accounts, hashtags, statuses) => {
|
||||||
|
@ -69,14 +72,24 @@ export default function search(state = initialState, action) {
|
||||||
switch(action.type) {
|
switch(action.type) {
|
||||||
case SEARCH_CHANGE:
|
case SEARCH_CHANGE:
|
||||||
return state.set('value', action.value);
|
return state.set('value', action.value);
|
||||||
case SEARCH_SUGGESTIONS_READY:
|
case SEARCH_CLEAR:
|
||||||
return normalizeSuggestions(state, action.value, action.accounts, action.hashtags, action.statuses);
|
|
||||||
case SEARCH_RESET:
|
|
||||||
return state.withMutations(map => {
|
return state.withMutations(map => {
|
||||||
map.set('suggestions', []);
|
|
||||||
map.set('value', '');
|
map.set('value', '');
|
||||||
map.set('loaded_value', '');
|
map.set('results', Immutable.Map());
|
||||||
|
map.set('submitted', false);
|
||||||
|
map.set('hidden', false);
|
||||||
});
|
});
|
||||||
|
case SEARCH_SHOW:
|
||||||
|
return state.set('hidden', false);
|
||||||
|
case COMPOSE_REPLY:
|
||||||
|
case COMPOSE_MENTION:
|
||||||
|
return state.set('hidden', true);
|
||||||
|
case SEARCH_FETCH_SUCCESS:
|
||||||
|
return state.set('results', Immutable.Map({
|
||||||
|
accounts: Immutable.List(action.results.accounts.map(item => item.id)),
|
||||||
|
statuses: Immutable.List(action.results.statuses.map(item => item.id)),
|
||||||
|
hashtags: Immutable.List(action.results.hashtags)
|
||||||
|
})).set('submitted', true);
|
||||||
default:
|
default:
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,7 +32,7 @@ import {
|
||||||
FAVOURITED_STATUSES_FETCH_SUCCESS,
|
FAVOURITED_STATUSES_FETCH_SUCCESS,
|
||||||
FAVOURITED_STATUSES_EXPAND_SUCCESS
|
FAVOURITED_STATUSES_EXPAND_SUCCESS
|
||||||
} from '../actions/favourites';
|
} from '../actions/favourites';
|
||||||
import { SEARCH_SUGGESTIONS_READY } from '../actions/search';
|
import { SEARCH_FETCH_SUCCESS } from '../actions/search';
|
||||||
import Immutable from 'immutable';
|
import Immutable from 'immutable';
|
||||||
|
|
||||||
const normalizeStatus = (state, status) => {
|
const normalizeStatus = (state, status) => {
|
||||||
|
@ -109,7 +109,7 @@ export default function statuses(state = initialState, action) {
|
||||||
case NOTIFICATIONS_EXPAND_SUCCESS:
|
case NOTIFICATIONS_EXPAND_SUCCESS:
|
||||||
case FAVOURITED_STATUSES_FETCH_SUCCESS:
|
case FAVOURITED_STATUSES_FETCH_SUCCESS:
|
||||||
case FAVOURITED_STATUSES_EXPAND_SUCCESS:
|
case FAVOURITED_STATUSES_EXPAND_SUCCESS:
|
||||||
case SEARCH_SUGGESTIONS_READY:
|
case SEARCH_FETCH_SUCCESS:
|
||||||
return normalizeStatuses(state, action.statuses);
|
return normalizeStatuses(state, action.statuses);
|
||||||
case TIMELINE_DELETE:
|
case TIMELINE_DELETE:
|
||||||
return deleteStatus(state, action.id, action.references);
|
return deleteStatus(state, action.id, action.references);
|
||||||
|
|
|
@ -764,8 +764,19 @@ a.status__content__spoiler-link {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.drawer__pager {
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 0;
|
||||||
|
flex-grow: 1;
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
.drawer__inner {
|
.drawer__inner {
|
||||||
//background: linear-gradient(rgba(lighten($color1, 13%), 1), rgba(lighten($color1, 13%), 0.65));
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
background: lighten($color1, 13%);
|
background: lighten($color1, 13%);
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
@ -773,7 +784,12 @@ a.status__content__spoiler-link {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
flex-grow: 1;
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
|
||||||
|
&.darker {
|
||||||
|
background: $color1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.drawer__header {
|
.drawer__header {
|
||||||
|
@ -1224,26 +1240,6 @@ button.active i.fa-retweet {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.search {
|
|
||||||
.fa {
|
|
||||||
color: $color3;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.search__input {
|
|
||||||
box-sizing: border-box;
|
|
||||||
display: block;
|
|
||||||
width: 100%;
|
|
||||||
border: none;
|
|
||||||
padding: 10px;
|
|
||||||
padding-right: 30px;
|
|
||||||
font-family: inherit;
|
|
||||||
background: $color1;
|
|
||||||
color: $color3;
|
|
||||||
font-size: 14px;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.loading-indicator {
|
.loading-indicator {
|
||||||
color: $color2;
|
color: $color2;
|
||||||
}
|
}
|
||||||
|
@ -1723,3 +1719,100 @@ button.active i.fa-retweet {
|
||||||
box-shadow: 2px 4px 6px rgba($color8, 0.1);
|
box-shadow: 2px 4px 6px rgba($color8, 0.1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.search {
|
||||||
|
position: relative;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search__input {
|
||||||
|
padding-right: 30px;
|
||||||
|
color: $color2;
|
||||||
|
outline: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
border: none;
|
||||||
|
padding: 10px;
|
||||||
|
padding-right: 30px;
|
||||||
|
font-family: inherit;
|
||||||
|
background: $color1;
|
||||||
|
color: $color3;
|
||||||
|
font-size: 14px;
|
||||||
|
margin: 0;
|
||||||
|
|
||||||
|
&::-moz-focus-inner {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
&::-moz-focus-inner, &:focus, &:active {
|
||||||
|
outline: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:focus {
|
||||||
|
background: lighten($color1, 4%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.search__icon {
|
||||||
|
.fa {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: 10px;
|
||||||
|
z-index: 2;
|
||||||
|
display: inline-block;
|
||||||
|
opacity: 0;
|
||||||
|
transition: all 100ms linear;
|
||||||
|
font-size: 18px;
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
color: $color2;
|
||||||
|
cursor: default;
|
||||||
|
pointer-events: none;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
pointer-events: auto;
|
||||||
|
opacity: 0.3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.fa-search {
|
||||||
|
transform: translateZ(0) rotate(90deg);
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
pointer-events: none;
|
||||||
|
transform: translateZ(0) rotate(0deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.fa-times-circle {
|
||||||
|
top: 11px;
|
||||||
|
transform: translateZ(0) rotate(0deg);
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
transform: translateZ(0) rotate(90deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-results__header {
|
||||||
|
color: lighten($color1, 26%);
|
||||||
|
background: lighten($color1, 2%);
|
||||||
|
border-bottom: 1px solid darken($color1, 4%);
|
||||||
|
padding: 15px 10px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-results__hashtag {
|
||||||
|
display: block;
|
||||||
|
padding: 10px;
|
||||||
|
color: $color2;
|
||||||
|
text-decoration: none;
|
||||||
|
|
||||||
|
&:hover, &:active, &:focus {
|
||||||
|
color: lighten($color2, 4%);
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue