1
0
mirror of https://github.com/stolksdorf/homebrewery.git synced 2025-12-18 18:01:29 +00:00

Adding new navitems and finishing the edit and share page

This commit is contained in:
Scott Tolksdorf
2016-05-09 16:54:32 -04:00
parent d5b8c60317
commit ed7decb42b
16 changed files with 350 additions and 204 deletions

View File

@@ -9,7 +9,7 @@ var PAGE_HEIGHT = 1056 + 30;
var BrewRenderer = React.createClass({ var BrewRenderer = React.createClass({
getDefaultProps: function() { getDefaultProps: function() {
return { return {
brewText : '' text : ''
}; };
}, },
getInitialState: function() { getInitialState: function() {
@@ -68,7 +68,7 @@ var BrewRenderer = React.createClass({
}, },
renderPages : function(){ renderPages : function(){
var pages = this.props.brewText.split('\\page'); var pages = this.props.text.split('\\page');
this.totalPages = pages.length; this.totalPages = pages.length;
return _.map(pages, (page, index)=>{ return _.map(pages, (page, index)=>{

View File

@@ -1,31 +1,23 @@
@import (less) './client/homebrew/phbStyle/phb.style.less'; @import (less) './client/homebrew/phbStyle/phb.style.less';
.pane{ .pane{
position: relative; position : relative;
} }
.brewRenderer{ .brewRenderer{
overflow-y: scroll; overflow-y : scroll;
//height : calc(~"100%");
.pageInfo{ .pageInfo{
background-color: #333; position : absolute;
color : White;
position: absolute;
padding: 8px 10px;
font-weight: 800;
font-size: 10px;
z-index : 1000;
bottom: 0;
right : 17px; right : 17px;
bottom : 0;
z-index : 1000;
padding : 8px 10px;
background-color : #333;
font-size : 10px;
font-weight : 800;
color : white;
} }
.pages{ .pages{
margin: 30px 0px; margin : 30px 0px;
&>.phb{ &>.phb{
margin-right : auto; margin-right : auto;
margin-bottom : 30px; margin-bottom : 30px;
@@ -33,6 +25,4 @@
box-shadow : 1px 4px 14px #000; box-shadow : 1px 4px 14px #000;
} }
} }
} }

View File

@@ -1,74 +1,51 @@
.editor{ .editor{
position : relative; position : relative;
width : 100%; width : 100%;
//height : 500px;
.snippetBar{ .snippetBar{
display : flex;
padding : 5px; padding : 5px;
background-color: #ddd; background-color : #ddd;
align-items : center;
display: flex;
align-items: center;
.snippetGroup{ .snippetGroup{
.animate(background-color); .animate(background-color);
margin : 0px 10px;
padding : 5px;
font-size : 15px;
border-radius : 5px; border-radius : 5px;
&:hover, &.selected{ &:hover, &.selected{
background-color: #999; background-color : #999;
} }
padding : 5px;
font-size: 15px;
margin: 0px 10px;
.text{ .text{
line-height: 20px; line-height : 20px;
.groupName{ .groupName{
margin-left: 6px; margin-left : 6px;
font-size: 12px; font-size : 12px;
} }
} }
//cursor : pointer;
&:hover{ &:hover{
.dropdown{ .dropdown{
visibility: visible; visibility : visible;
} }
} }
.dropdown{ .dropdown{
position: absolute; position : absolute;
visibility : hidden;
z-index : 1000; z-index : 1000;
background-color: #ddd;
padding : 5px; padding : 5px;
visibility: hidden; background-color : #ddd;
.snippet{ .snippet{
font-size: 13px;
padding : 10px;
.animate(background-color); .animate(background-color);
padding : 10px;
cursor : pointer; cursor : pointer;
font-size : 13px;
&:hover{ &:hover{
background-color: #999; background-color : #999;
} }
} }
} }
} }
} }
.codeEditor{ .codeEditor{
height : 100%; height : 100%;
} }

View File

@@ -17,7 +17,8 @@ var Homebrew = React.createClass({
welcomeText : "", welcomeText : "",
changelog : "", changelog : "",
brew : { brew : {
text : "", title : '',
text : '',
shareId : null, shareId : null,
editId : null, editId : null,
createdAt : null, createdAt : null,
@@ -28,14 +29,14 @@ var Homebrew = React.createClass({
componentWillMount: function() { componentWillMount: function() {
Router = CreateRouter({ Router = CreateRouter({
'/homebrew/edit/:id' : (args) => { '/homebrew/edit/:id' : (args) => {
return <EditPage id={args.id} entry={this.props.brew} /> return <EditPage id={args.id} brew={this.props.brew} />
}, },
'/homebrew/share/:id' : (args) => { '/homebrew/share/:id' : (args) => {
return <SharePage id={args.id} entry={this.props.brew} /> return <SharePage id={args.id} brew={this.props.brew} />
}, },
'/homebrew/changelog' : (args) => { '/homebrew/changelog' : (args) => {
return <SharePage entry={{text : this.props.changelog}} /> return <SharePage brew={{title : 'Changelog', text : this.props.changelog}} />
}, },
'/homebrew/new' : (args) => { '/homebrew/new' : (args) => {
return <NewPage /> return <NewPage />

View File

@@ -10,7 +10,6 @@ const MAX_URL_SIZE = 2083;
const MAIN_URL = "https://www.reddit.com/r/UnearthedArcana/submit?selftext=true" const MAIN_URL = "https://www.reddit.com/r/UnearthedArcana/submit?selftext=true"
var RedditShare = React.createClass({ var RedditShare = React.createClass({
getDefaultProps: function() { getDefaultProps: function() {
return { return {

View File

@@ -39,4 +39,11 @@
} }
} }
} }
.brewTitle.navItem{
font-size : 12px;
font-weight : 800;
color : white;
text-align : center;
text-transform: initial;
}
} }

View File

@@ -0,0 +1,51 @@
var React = require('react');
var _ = require('lodash');
var cx = require('classnames');
//var striptags = require('striptags');
var Nav = require('naturalcrit/nav/nav.jsx');
const MAX_URL_SIZE = 2083;
const MAIN_URL = "https://www.reddit.com/r/UnearthedArcana/submit?selftext=true"
var RedditShare = React.createClass({
getDefaultProps: function() {
return {
brew : {
title : '',
sharedId : '',
text : ''
}
};
},
getText : function(){
},
handleClick : function(){
var url = [
MAIN_URL,
'title=' + encodeURIComponent(this.props.brew.title ? this.props.brew.title : 'Check out my brew!'),
'text=' + encodeURIComponent(this.props.brew.text)
].join('&');
window.open(url, '_blank');
},
render : function(){
return <Nav.item icon='fa-reddit-alien' color='red' onClick={this.handleClick}>
share on reddit
</Nav.item>
},
});
module.exports = RedditShare;

View File

@@ -0,0 +1,51 @@
var React = require('react');
var _ = require('lodash');
var cx = require('classnames');
//var striptags = require('striptags');
var Nav = require('naturalcrit/nav/nav.jsx');
const MAX_URL_SIZE = 2083;
const MAIN_URL = "https://www.reddit.com/r/UnearthedArcana/submit?selftext=true"
var RedditShare = React.createClass({
getDefaultProps: function() {
return {
brew : {
title : '',
sharedId : '',
text : ''
}
};
},
getText : function(){
},
handleClick : function(){
var url = [
MAIN_URL,
'title=' + encodeURIComponent(this.props.brew.title ? this.props.brew.title : 'Check out my brew!'),
'text=' + encodeURIComponent(this.props.brew.text)
].join('&');
window.open(url, '_blank');
},
render : function(){
return <Nav.item icon='fa-reddit-alien' color='red' onClick={this.handleClick}>
share on reddit
</Nav.item>
},
});
module.exports = RedditShare;

View File

@@ -0,0 +1,51 @@
var React = require('react');
var _ = require('lodash');
var cx = require('classnames');
//var striptags = require('striptags');
var Nav = require('naturalcrit/nav/nav.jsx');
const MAX_URL_SIZE = 2083;
const MAIN_URL = "https://www.reddit.com/r/UnearthedArcana/submit?selftext=true"
var RedditShare = React.createClass({
getDefaultProps: function() {
return {
brew : {
title : '',
sharedId : '',
text : ''
}
};
},
getText : function(){
},
handleClick : function(){
var url = [
MAIN_URL,
'title=' + encodeURIComponent(this.props.brew.title ? this.props.brew.title : 'Check out my brew!'),
'text=' + encodeURIComponent(this.props.brew.text)
].join('&');
window.open(url, '_blank');
},
render : function(){
return <Nav.item icon='fa-reddit-alien' color='red' onClick={this.handleClick}>
share on reddit
</Nav.item>
},
});
module.exports = RedditShare;

View File

@@ -1,15 +1,9 @@
@import (less) './client/homebrew/phbStyle/phb.style.less';
@import (less) './client/homebrew/phbStyle/phb.style.less';
.pageContainer{ .pageContainer{
background-color : @steel; background-color : @steel;
// overflow-y: scroll;
//height :100%;
.pages{ .pages{
padding : 30px 0px; padding : 30px 0px;
&>.phb{ &>.phb{
margin-right : auto; margin-right : auto;
margin-bottom : 30px; margin-bottom : 30px;
@@ -17,5 +11,4 @@
box-shadow : 1px 4px 14px #000; box-shadow : 1px 4px 14px #000;
} }
} }
} }

View File

@@ -1,16 +1,32 @@
var React = require('react'); var React = require('react');
var _ = require('lodash'); var _ = require('lodash');
var cx = require('classnames'); var cx = require('classnames');
var request = require("superagent");
var Nav = require('naturalcrit/nav/nav.jsx');
var Navbar = require('../../navbar/navbar.jsx');
var EditTitle = require('../../navbar/editTitle.navitem.jsx');
var SplitPane = require('naturalcrit/splitPane/splitPane.jsx');
var Editor = require('../../editor/editor.jsx');
var BrewRenderer = require('../../brewRenderer/brewRenderer.jsx');
var Statusbar = require('../../statusbar/statusbar.jsx'); var Statusbar = require('../../statusbar/statusbar.jsx');
var PageContainer = require('../../pageContainer/pageContainer.jsx'); var PageContainer = require('../../pageContainer/pageContainer.jsx');
var Editor = require('../../editor/editor.jsx'); var Editor = require('../../editor/editor.jsx');
var FullClassGen = require('../../editor/snippets/fullclass.gen.js');
var request = require("superagent");
var SAVE_TIMEOUT = 3000;
const SAVE_TIMEOUT = 3000;
@@ -18,8 +34,9 @@ var EditPage = React.createClass({
getDefaultProps: function() { getDefaultProps: function() {
return { return {
id : null, id : null,
entry : { brew : {
text : "", title : '',
text : '',
shareId : null, shareId : null,
editId : null, editId : null,
createdAt : null, createdAt : null,
@@ -30,9 +47,13 @@ var EditPage = React.createClass({
getInitialState: function() { getInitialState: function() {
return { return {
text: this.props.entry.text, title : this.props.brew.title,
text: this.props.brew.text,
isSaving : false,
errors : null,
pending : false, pending : false,
lastUpdated : this.props.entry.updatedAt
lastUpdated : this.props.brew.updatedAt
}; };
}, },
@@ -43,6 +64,21 @@ var EditPage = React.createClass({
return "You have unsaved changes!"; return "You have unsaved changes!";
} }
}, },
componentWillUnmount: function() {
window.onbeforeunload = function(){};
},
handleSplitMove : function(){
this.refs.editor.update();
},
handleTitleChange : function(title){
this.setState({
title : title,
pending : true
});
this.save();
},
handleTextChange : function(text){ handleTextChange : function(text){
this.setState({ this.setState({
@@ -52,9 +88,20 @@ var EditPage = React.createClass({
this.save(); this.save();
}, },
handleDelete : function(){
if(!confirm("are you sure you want to delete this brew?")) return;
if(!confirm("are you REALLY sure? You will not be able to recover it")) return;
request.get('/homebrew/api/remove/' + this.props.brew.editId)
.send()
.end(function(err, res){
window.location.href = '/homebrew';
});
},
save : _.debounce(function(){ save : _.debounce(function(){
request request
.put('/homebrew/update/' + this.props.id) .put('/homebrew/api/update/' + this.props.id)
.send({text : this.state.text}) .send({text : this.state.text})
.end((err, res) => { .end((err, res) => {
this.setState({ this.setState({
@@ -64,25 +111,44 @@ var EditPage = React.createClass({
}) })
}, SAVE_TIMEOUT), }, SAVE_TIMEOUT),
renderNavbar : function(){
return <Navbar>
<Nav.section>
<EditTitle title={this.state.title} onChange={this.handleTitleChange} />
</Nav.section>
<Nav.section>
<Nav.item newTab={true} href='https://github.com/stolksdorf/naturalcrit/issues' color='red' icon='fa-bug'>
report issue
</Nav.item>
<Nav.item newTab={true} href={'/homebrew/share/' + this.props.brew.shareId} color='teal' icon='fa-share'>
Share
</Nav.item>
<Nav.item newTab={true} href={'/homebrew/print/' + this.props.brew.sharedId} color='orange' icon='fa-print'>
print
</Nav.item>
<Nav.item color='lightred' icon='fa-trash' onClick={this.handleDelete}>
Delete
</Nav.item>
</Nav.section>
</Navbar>
},
render : function(){ render : function(){
return <div className='editPage'> return <div className='editPage page'>
{this.renderNavbar()}
<div className='content'>
<SplitPane onDragFinish={this.handleSplitMove} ref='pane'>
<Statusbar <Editor value={this.state.text} onChange={this.handleTextChange} ref='editor'/>
editId={this.props.entry.editId} <BrewRenderer text={this.state.text} />
shareId={this.props.entry.shareId} </SplitPane>
printId={this.props.entry.shareId}
lastUpdated={this.state.lastUpdated}
isPending={this.state.pending} />
<div className='paneSplit'>
<div className='leftPane'>
<Editor text={this.state.text} onChange={this.handleTextChange} />
</div>
<div className='rightPane'>
<PageContainer text={this.state.text} />
</div>
</div> </div>
</div> </div>
} }

View File

@@ -4,7 +4,6 @@ var cx = require('classnames');
var Nav = require('naturalcrit/nav/nav.jsx'); var Nav = require('naturalcrit/nav/nav.jsx');
var Navbar = require('../../navbar/navbar.jsx'); var Navbar = require('../../navbar/navbar.jsx');
var RedditShare = require('../../navbar/redditShare.navitem.jsx');
@@ -14,64 +13,35 @@ var BrewRenderer = require('../../brewRenderer/brewRenderer.jsx');
var HomePage = React.createClass({ var HomePage = React.createClass({
getDefaultProps: function() { getDefaultProps: function() {
return { return {
welcomeText : "" welcomeText : ""
}; };
}, },
getInitialState: function() { getInitialState: function() {
return { return {
text: this.props.welcomeText text: this.props.welcomeText
}; };
}, },
handleSplitMove : function(){ handleSplitMove : function(){
this.refs.editor.update(); this.refs.editor.update();
}, },
handleTextChange : function(text){ handleTextChange : function(text){
this.setState({ this.setState({
text : text text : text
}); });
//localStorage.setItem(KEY, text);
}, },
renderNavbar : function(){ renderNavbar : function(){
return <Navbar> return <Navbar>
<Nav.section> <Nav.section>
<Nav.item>Bad Ass Brew</Nav.item> <Nav.item newTab={true} href='https://github.com/stolksdorf/naturalcrit/issues' color='red' icon='fa-bug'>
</Nav.section>
<Nav.section>
<RedditShare brew={{text : this.state.text}}/>
<Nav.item
newTab={true}
href='https://github.com/stolksdorf/naturalcrit/issues'
color='red'
icon='fa-bug'>
report issue report issue
</Nav.item> </Nav.item>
<Nav.item <Nav.item newTab={true} href='/homebrew/changelog' color='purple' icon='fa-file-text-o'>
newTab={true}
href='/homebrew/changelog'
color='purple'
icon='fa-file-text-o'>
Changelog Changelog
</Nav.item> </Nav.item>
<Nav.item <Nav.item href='/homebrew/new' color='green' icon='fa-external-link'>
href='/homebrew/new'
color='green'
icon='fa-external-link'>
New Brew New Brew
</Nav.item> </Nav.item>
</Nav.section> </Nav.section>
@@ -85,45 +55,15 @@ var HomePage = React.createClass({
<div className='content'> <div className='content'>
<SplitPane onDragFinish={this.handleSplitMove} ref='pane'> <SplitPane onDragFinish={this.handleSplitMove} ref='pane'>
<Editor value={this.state.text} onChange={this.handleTextChange} ref='editor'/> <Editor value={this.state.text} onChange={this.handleTextChange} ref='editor'/>
<BrewRenderer brewText={this.state.text} /> <BrewRenderer text={this.state.text} />
</SplitPane> </SplitPane>
</div> </div>
{/*
<a href='/homebrew/new' className='floatingNewButton'> <a href='/homebrew/new' className='floatingNewButton'>
Create your own <i className='fa fa-magic' /> Create your own <i className='fa fa-magic' />
</a> </a>
*/}
</div> </div>
} }
}); });
module.exports = HomePage; module.exports = HomePage;
/*
<SplitPane>
<PageContainer text={this.state.text} />
<Editor text={this.state.text} onChange={this.handleTextChange} />
</SplitPane>
*/
/* Test code
<div className='content'>
<SplitPane>
<div className='woo'>
one
</div>
<div className='temp'>
yo
<div className='tooBig' />
</div>
</SplitPane>
</div>
*/

View File

@@ -32,8 +32,9 @@ var NewPage = React.createClass({
text : storage text : storage
}) })
} }
window.onbeforeunload = function(e){ window.onbeforeunload = (e)=>{
//return "Your homebrew isn't saved. Are you sure you want to leave?"; if(this.state.text == '') return;
return "Your homebrew isn't saved. Are you sure you want to leave?";
}; };
}, },
@@ -104,15 +105,9 @@ var NewPage = React.createClass({
<Nav.section> <Nav.section>
{this.renderSaveButton()} {this.renderSaveButton()}
<Nav.item newTab={true} href='https://github.com/stolksdorf/naturalcrit/issues' color='red' icon='fa-bug'>
<Nav.item
newTab={true}
href='https://github.com/stolksdorf/naturalcrit/issues'
color='red'
icon='fa-bug'>
report issue report issue
</Nav.item> </Nav.item>
</Nav.section> </Nav.section>
</Navbar> </Navbar>
}, },
@@ -125,7 +120,7 @@ var NewPage = React.createClass({
<div className='content'> <div className='content'>
<SplitPane onDragFinish={this.handleSplitMove} ref='pane'> <SplitPane onDragFinish={this.handleSplitMove} ref='pane'>
<Editor value={this.state.text} onChange={this.handleTextChange} ref='editor'/> <Editor value={this.state.text} onChange={this.handleTextChange} ref='editor'/>
<BrewRenderer brewText={this.state.text} /> <BrewRenderer text={this.state.text} />
</SplitPane> </SplitPane>
</div> </div>
</div> </div>

View File

@@ -2,18 +2,22 @@ var React = require('react');
var _ = require('lodash'); var _ = require('lodash');
var cx = require('classnames'); var cx = require('classnames');
var Statusbar = require('../../statusbar/statusbar.jsx'); var Nav = require('naturalcrit/nav/nav.jsx');
var Navbar = require('../../navbar/navbar.jsx');
var PageContainer = require('../../pageContainer/pageContainer.jsx'); var BrewRenderer = require('../../brewRenderer/brewRenderer.jsx');
var replaceAll = function(str, find, replace) {
return str.replace(new RegExp(find, 'g'), replace);
}
var SharePage = React.createClass({ var SharePage = React.createClass({
getDefaultProps: function() { getDefaultProps: function() {
return { return {
id : null, brew : {
entry : { title : '',
text : "", text : '',
shareId : null, shareId : null,
editId : null,
createdAt : null, createdAt : null,
updatedAt : null, updatedAt : null,
views : 0 views : 0
@@ -21,20 +25,38 @@ var SharePage = React.createClass({
}; };
}, },
render : function(){ openSourceWindow : function(){
return( var sourceWindow = window.open();
<div className='sharePage'> var content = replaceAll(this.props.brew.text, '<', '&lt;');
<Statusbar content = replaceAll(content, '>', '&gt;');
sourceText={this.props.entry.text} sourceWindow.document.write('<code><pre>' + content + '</pre></code>');
lastUpdated={this.props.entry.updatedAt} },
views={this.props.entry.views}
printId={this.props.entry.shareId}
shareId={this.props.entry.shareId}
/>
<PageContainer text={this.props.entry.text} />
render : function(){
console.log('brew', this.props.brew);
return <div className='sharePage page'>
<Navbar>
<Nav.section>
<Nav.item className='brewTitle'>{this.props.brew.title}</Nav.item>
</Nav.section>
<Nav.section>
<Nav.item newTab={true} href={'/homebrew/print/' + this.props.brew.sharedId} color='orange' icon='fa-print'>
print
</Nav.item>
<Nav.item onClick={this.openSourceWindow} color='teal' icon='fa-code'>
source
</Nav.item>
</Nav.section>
</Navbar>
<div className='content'>
<BrewRenderer text={this.props.brew.text} />
</div>
</div> </div>
);
} }
}); });

View File

@@ -5,6 +5,7 @@ var _ = require('lodash');
var HomebrewSchema = mongoose.Schema({ var HomebrewSchema = mongoose.Schema({
shareId : {type : String, default: shortid.generate}, shareId : {type : String, default: shortid.generate},
editId : {type : String, default: shortid.generate}, editId : {type : String, default: shortid.generate},
title : {type : String, default : ""},
text : {type : String, default : ""}, text : {type : String, default : ""},
createdAt : { type: Date, default: Date.now }, createdAt : { type: Date, default: Date.now },

View File

@@ -8,7 +8,7 @@ X Add `infoBox` to BrewRenderer to show views, and current pages
- remove old status bar - remove old status bar
X remove jsoneditor (if we don't need it) X remove jsoneditor (if we don't need it)
X Add in markdown editor X Add in markdown editor
- Add the '/new' page and force save to reduce database size X Add the '/new' page and force save to reduce database size
X Add pagniation and query to the homebrew api X Add pagniation and query to the homebrew api
X Update the admin page with pagnition and a query box X Update the admin page with pagnition and a query box
X Test the old/small brew filtering for deleteion X Test the old/small brew filtering for deleteion
@@ -17,8 +17,10 @@ X Partial rendering kills style tags on unrendered pages. Detect if pages have s
- Add in brew title, use for metadata? - Add in brew title, use for metadata?
- Add in specific entry point rendering in server.js - Add in specific entry point rendering in server.js
- Add in a tutorial page? - Add in a tutorial page?
- Add in a localstorage fallback on the `/new` page, clear it when they save X Add in a localstorage fallback on the `/new` page, clear it when they save
X Rename `/client/naturalCrit` -> `/client/main` X Rename `/client/naturalCrit` -> `/client/main`
- Move snippets into their new groups
- Replace pseudo-elements with encoded images
## v1.6 ## v1.6
- Add error handling to the saving wdiget in the status bar - Add error handling to the saving wdiget in the status bar