Skip to content

Instantly share code, notes, and snippets.

@steveluscher
Last active December 2, 2015 16:26
Show Gist options
  • Save steveluscher/3abe6d6f598d70423f76 to your computer and use it in GitHub Desktop.
Save steveluscher/3abe6d6f598d70423f76 to your computer and use it in GitHub Desktop.
[WIP] New Relay Tutorial App
class Comment extends React.Component {
render() {
var {comment} = this.props;
var {author} = comment;
var savePending = this.props.relay.hasOptimisticUpdate(this.props.comment);
return (
<div style={{opacity: savePending ? 0.4 : null}}>
<img src={author.avatar} width={16} /> <strong>{author.name}</strong> {comment.text}
{savePending &&
<img src="" style={{marginLeft: 4, verticalAlign: 'text-top'}} />
}
</div>
);
}
}
Comment = Relay.createContainer(Comment, {
fragments: {
comment: () => Relay.QL`
fragment on Comment {
author { avatar, name }
text
}
`,
},
});
class Story extends React.Component {
_augmentNumCommentsToShow = (delta) => {
this.props.relay.setVariables({
numCommentsToShow: this.props.relay.variables.numCommentsToShow + delta,
});
}
_handleKeyDown = (e) => {
if (e.keyCode === 13 && e.target.value != '') { // enter key
Relay.Store.update(
new AddCommentMutation({
story: this.props.story,
text: e.target.value,
viewer: this.props.viewer,
})
);
this._augmentNumCommentsToShow(1);
e.target.value = '';
}
}
_handleMoreCommentsClick = (e) => {
e.preventDefault();
this._augmentNumCommentsToShow(5);
}
render() {
var {story} = this.props;
var {author, comments} = story;
return (
<div>
<header>
<img src={author.avatar} width={32} /> <strong>{author.name}</strong>
</header>
<p>{story.text}</p>
{comments.pageInfo.hasPreviousPage &&
<a href="#" onClick={this._handleMoreCommentsClick}>
View previous comments
</a>
}
<ul>
{comments && comments.edges.map(commentEdge =>
<li key={commentEdge.node.id}>
<Comment comment={commentEdge.node} />
</li>
)}
<li>
<input
onKeyDown={this._handleKeyDown}
placeholder="Leave a comment&hellip;"
type="text"
/>
</li>
</ul>
</div>
);
}
}
Story = Relay.createContainer(Story, {
initialVariables: {
numCommentsToShow: 3,
},
fragments: {
story: () => Relay.QL`
fragment on Story {
author { avatar, name }
comments(last: $numCommentsToShow) {
edges {
node {
id
${Comment.getFragment('comment')}
}
}
pageInfo { hasPreviousPage }
}
text
${AddCommentMutation.getFragment('story')}
}
`,
viewer: () => Relay.QL`
fragment on Viewer {
avatar
id
name
${AddCommentMutation.getFragment('viewer')}
}
`,
},
});
class StoriesApp extends React.Component {
_handleLoadMoreClick = () => {
this.props.relay.setVariables({
numStoriesToLoad: this.props.relay.variables.numStoriesToLoad + 3,
});
}
render() {
var {storyFeed} = this.props.viewer;
return (
<div>
<ul>
{storyFeed.edges.map(edge =>
<li key={edge.node.id}>
<Story
story={edge.node}
viewer={this.props.viewer}
/>
</li>
)}
</ul>
{storyFeed.pageInfo.hasNextPage &&
<button onClick={this._handleLoadMoreClick}>
Load more stories
</button>
}
</div>
);
}
}
StoriesApp = Relay.createContainer(StoriesApp, {
initialVariables: {
numStoriesToLoad: 3,
},
fragments: {
viewer: () => Relay.QL`
fragment on Viewer {
storyFeed(first: $numStoriesToLoad) {
edges {
node {
id
${Story.getFragment('story')},
}
}
pageInfo { hasNextPage }
}
${Story.getFragment('viewer')}
}
`,
}
});
class AddCommentMutation extends Relay.Mutation {
static fragments = {
story: () => Relay.QL`
fragment on Story {
id
}
`,
viewer: () => Relay.QL`
fragment on Viewer {
id
}
`,
};
getCollisionKey() {
return `story-${this.props.story.id}`;
}
getMutation() {
return Relay.QL`mutation{addComment}`;
}
getFatQuery() {
return Relay.QL`
fragment on AddCommentPayload {
commentEdge
story { comments }
}
`;
}
getConfigs() {
return [{
type: 'RANGE_ADD',
parentName: 'story',
parentID: this.props.story.id,
connectionName: 'comments',
edgeName: 'commentEdge',
rangeBehaviors: {
'': 'append',
},
}];
}
getVariables() {
return {
storyId: this.props.story.id,
text: this.props.text,
};
}
getOptimisticResponse() {
return {
commentEdge: {
node: {
author: {
avatar: this.props.viewer.avatar,
id: this.props.viewer.id,
name: this.props.viewer.name,
},
story: {
id: this.props.story.id,
},
text: this.props.text,
},
},
};
}
}
class StoriesRoute extends Relay.Route {
static params = {};
static queries = {
viewer: (Component) => Relay.QL`
query ViewerQuery {
viewer {
${Component.getFragment('viewer')},
}
}
`,
};
static routeName = 'Stories';
}
ReactDOM.render(
<Relay.RootContainer
Component={StoriesApp}
route={new StoriesRoute()}
/>,
mountNode
);
import {
GraphQLBoolean,
GraphQLEnumType,
GraphQLFloat,
GraphQLID,
GraphQLInputObjectType,
GraphQLInt,
GraphQLInterfaceType,
GraphQLList,
GraphQLNonNull,
GraphQLObjectType,
GraphQLSchema,
GraphQLString,
GraphQLUnionType,
} from 'graphql';
import {
connectionArgs,
connectionDefinitions,
connectionFromArray,
cursorForObjectInConnection,
fromGlobalId,
globalIdField,
mutationWithClientMutationId,
nodeDefinitions,
toGlobalId,
} from 'graphql-relay';
/**
* Set up some test data
*/
var TUTORIAL_VERSION = 1;
class Comment {
constructor(data) {
this.authorId = data.authorId;
this.id = data.id;
this.storyId = data.storyId;
this.text = data.text;
}
}
class Person {
constructor(data) {
this.avatar = data.avatar;
this.id = data.id;
this.name = data.name;
}
}
class Story {
constructor(data) {
this.authorId = data.authorId;
this.id = data.id;
this.text = data.text;
}
}
class Viewer {
constructor(data) {
this.avatar = data.avatar;
this.id = data.id;
this.name = data.name;
}
}
var COMMENTS;
try {
COMMENTS = JSON.parse(localStorage.getItem(`relay-tutorial-${TUTORIAL_VERSION}-comments`));
} catch(e) {}
if (COMMENTS == null) {
COMMENTS = [
new Comment({id: '0', authorId: '2', storyId: '0', text: 'Yeah!'}),
new Comment({id: '1', authorId: '3', storyId: '1', text: 'OK!'}),
];
}
var PEOPLE;
try {
PEOPLE = JSON.parse(localStorage.getItem(`relay-tutorial-${TUTORIAL_VERSION}-people`));
} catch(e) {}
if (PEOPLE == null) {
PEOPLE = [
new Person({id: '0', name: 'You', avatar: ''}),
new Person({id: '1', name: 'Steve', avatar: ''}),
new Person({id: '2', name: 'Yuzhi', avatar: ''}),
new Person({id: '3', name: 'Joe', avatar: ''}),
new Person({id: '4', name: 'Tim', avatar: ''}),
new Person({id: '5', name: 'Jan', avatar: ''}),
];
}
var STORIES;
try {
STORIES = JSON.parse(localStorage.getItem(`relay-tutorial-${TUTORIAL_VERSION}-stories`));
} catch(e) {}
if (STORIES == null) {
STORIES = [
new Story({id: '0', text: 'Everybody ready to publish a new version?', authorId: '1'}),
new Story({id: '1', text: 'Anyone want to grab lunch?', authorId: '2'}),
new Story({id: '2', text: 'I have a new idea; anyone want to grab a whiteboard and sketch it out?', authorId: '3'}),
new Story({id: '3', text: '#728131, #711151, and #817129 are fixed and landed.', authorId: '4'}),
new Story({id: '4', text: 'I\'m working on something that should increase developer efficiency. Stay tuned!', authorId: '5'}),
];
}
var VIEWER = new Viewer(PEOPLE[0]);
/**
* Let Relay map between:
* - global IDs and the object they represent
* - objects and the GraphQL type associated with them
*/
var {nodeInterface, nodeField} = nodeDefinitions(
(globalId) => {
var {type, id} = fromGlobalId(globalId);
if (type === 'Comment') {
return COMMENTS.find(obj => obj.id === id);
} else if (type === 'Person') {
return PEOPLE.find(obj => obj.id === id);
} else if (type === 'Story') {
return STORIES.find(obj => obj.id === id);
} else if (type === 'Viewer') {
return VIEWER;
}
return null;
},
(obj) => {
if (obj instanceof Comment) {
return CommentType;
} else if (obj instanceof Person) {
return PersonType;
} else if (obj instanceof Story) {
return StoryType;
} else if (obj instanceof Viewer) {
return ViewerType;
}
return null;
}
);
/**
* Define an interface that all person-like objects will conform to
*/
var PersonableInterface = new GraphQLInterfaceType({
name: 'Personable',
fields: () => ({
avatar: {
type: GraphQLString,
description: 'The URL of a person\'s avatar image',
},
comments: {
type: CommentConnectionType,
description: 'Comments made on stories by this person',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
COMMENTS.filter(comment => comment.authorId === obj.id),
args
),
},
name: { type: GraphQLString },
stories: {
type: StoryConnectionType,
description: 'Stories written by this person',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
STORIES.filter(story => story.authorId === obj.id),
args
),
},
}),
resolveType(obj) {
return obj instanceof Person ? PersonType :
obj instanceof Viewer ? ViewerType :
null;
},
});
/**
* Configure each type of object: Comments, People, Stories, and the Viewer
*/
var CommentType = new GraphQLObjectType({
name: 'Comment',
description: 'A comment on a story',
fields: () => ({
author: {
type: PersonType,
description: 'The author of this comment',
resolve: (obj) => PEOPLE[obj.authorId],
},
id: globalIdField('Comment'),
story: {
type: StoryType,
description: 'The story this comment is attached to',
resolve: () => STORIES[obj.storyId],
},
text: {
type: GraphQLString,
},
}),
interfaces: [nodeInterface],
});
var PersonType = new GraphQLObjectType({
name: 'Person',
description: 'A person who writes stories and comments',
fields: () => ({
avatar: {
type: GraphQLString,
description: 'The URL of a person\'s avatar image',
},
comments: {
type: CommentConnectionType,
description: 'Comments made on stories by this person',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
COMMENTS.filter(comment => comment.authorId === obj.id),
args
),
},
id: globalIdField('Person'),
name: { type: GraphQLString },
stories: {
type: StoryConnectionType,
description: 'Stories written by this person',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
STORIES.filter(story => story.authorId === obj.id),
args
),
},
}),
interfaces: [nodeInterface, PersonableInterface],
});
var StoryType = new GraphQLObjectType({
name: 'Story',
description: 'A story written by a person',
fields: () => ({
author: {
type: PersonType,
description: 'The author of this story',
resolve: (obj) => PEOPLE[obj.authorId],
},
comments: {
type: CommentConnectionType,
description: 'Comments people have made on this story',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
COMMENTS.filter(comment => comment.storyId === obj.id),
args
),
},
id: globalIdField('Story'),
text: {
type: GraphQLString,
},
}),
interfaces: [nodeInterface],
});
var ViewerType = new GraphQLObjectType({
name: 'Viewer',
description: 'The acting person (eg. the logged in visitor)',
fields: () => ({
avatar: {
type: GraphQLString,
description: 'The URL of the viewer\'s avatar image',
},
comments: {
type: CommentConnectionType,
description: 'Comments on stories, made by the viewer',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
COMMENTS.filter(c => c.authorId === obj.id),
args
),
},
id: globalIdField('Viewer'),
name: { type: GraphQLString },
friends: {
type: PersonConnectionType,
description: 'Friends of the viewer',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
PEOPLE.filter(p => p.id !== obj.id),
args
),
},
stories: {
type: StoryConnectionType,
description: 'Stories written by the viewer',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(
STORIES.filter(story => story.authorId === obj.id),
args
),
},
storyFeed: {
type: StoryConnectionType,
description: 'Stories visible to the viewer',
args: connectionArgs,
resolve: (obj, args) => connectionFromArray(STORIES, args),
},
}),
interfaces: [nodeInterface, PersonableInterface],
});
/**
* Set up the connections between types
*/
var {
connectionType: CommentConnectionType,
edgeType: CommentEdgeType,
} = connectionDefinitions({
name: 'Comment',
nodeType: CommentType,
});
var {
connectionType: PersonConnectionType,
edgeType: PersonEdgeType,
} = connectionDefinitions({
name: 'Person',
nodeType: PersonType,
});
var {
connectionType: StoryConnectionType,
edgeType: StoryEdgeType,
} = connectionDefinitions({
name: 'Story',
nodeType: StoryType,
});
/**
* Configure a mutation to allow people to make comments
*/
var AddCommentMutation = mutationWithClientMutationId({
name: 'AddComment',
inputFields: {
storyId: { type: new GraphQLNonNull(GraphQLID) },
text: { type: new GraphQLNonNull(GraphQLString) },
},
outputFields: {
commentEdge: {
type: CommentEdgeType,
resolve: ({localCommentId}) => {
var comment = COMMENTS.find((c) => c.id === localCommentId);
return {
cursor: cursorForObjectInConnection(COMMENTS, comment),
node: comment,
};
},
},
story: {
type: StoryType,
resolve: ({storyId}) => STORIES.find(s => s.id === storyId),
},
viewer: {
type: ViewerType,
resolve: () => VIEWER,
},
},
mutateAndGetPayload: async ({storyId, text}) => {
var localCommentId = COMMENTS.length;
COMMENTS.push({
authorId: VIEWER.id,
id: localCommentId,
storyId: fromGlobalId(storyId, 'Story').id,
text,
});
localStorage.setItem(`relay-tutorial-${TUTORIAL_VERSION}-comments`, JSON.stringify(COMMENTS));
await new Promise(r => setTimeout(r, 500 + Math.random() * 1000));
return {localCommentId, storyId};
}
});
/**
* Finally, configure the root Query and Mutation type
*/
var QueryType = new GraphQLObjectType({
name: 'Query',
fields: () => ({
node: nodeField,
viewer: {
type: ViewerType,
resolve: () => VIEWER,
},
}),
});
var MutationType = new GraphQLObjectType({
name: 'Mutation',
fields: {
addComment: AddCommentMutation,
},
});
export default new GraphQLSchema({
query: QueryType,
mutation: MutationType,
});
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment