Skip to content
This repository has been archived by the owner on May 30, 2019. It is now read-only.

Commit

Permalink
profile pages
Browse files Browse the repository at this point in the history
  • Loading branch information
alaq committed Mar 21, 2018
1 parent ba6b55a commit 555bd14
Show file tree
Hide file tree
Showing 3 changed files with 156 additions and 30 deletions.
19 changes: 19 additions & 0 deletions website/db.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ export interface Database {
updateDoc(nbId: string, doc: NotebookDoc): Promise<void>;
clone(existingDoc: NotebookDoc): Promise<string>;
create(): Promise<string>;
queryProfile(uid: string): Promise<NbInfo[]>;
queryLatest(): Promise<NbInfo[]>;
signIn(): void;
signOut(): void;
Expand Down Expand Up @@ -167,6 +168,19 @@ class DatabaseFB implements Database {
return out.reverse();
}

async queryProfile(uid): Promise<NbInfo[]> {
lazyInit();
const query = nbCollection.where("owner.uid", "==", uid);
const snapshots = await query.get();
const out = [];
snapshots.forEach(snap => {
const nbId = snap.id;
const doc = snap.data();
out.unshift({ nbId, doc });
});
return out.reverse();
}

signIn() {
lazyInit();
const provider = new firebase.auth.GithubAuthProvider();
Expand Down Expand Up @@ -218,6 +232,11 @@ export class DatabaseMock implements Database {
return "createdNbId";
}

async queryProfile(uid: string): Promise<NbInfo[]> {
this.inc("queryProfile");
return [];
}

async queryLatest(): Promise<NbInfo[]> {
this.inc("queryLatest");
return [];
Expand Down
45 changes: 35 additions & 10 deletions website/main.scss
Original file line number Diff line number Diff line change
Expand Up @@ -478,6 +478,26 @@ header {
}
}

.blurb {
background-color: #fff;
box-shadow: inset 0 1px 0 #d8d8d8;
font-size: 1.2rem;
letter-spacing: 0.5px;
display: flex;
margin-top: 8px;

img {
margin-right: $size0;
}

a {
font-size: $size1;
line-height: $size2;
color: $textColor;
margin: $size1;
}
}

.propel-logo {
display: flex;
flex-direction: row;
Expand Down Expand Up @@ -610,20 +630,20 @@ header {
@include buttonHover;
}

.blurb {
background-color: #fff;
box-shadow: inset 0 1px 0 #d8d8d8;
font-size: 1.2rem;
letter-spacing: 0.5px;
display: flex;
margin-top: 8px;
.notebook-container {
max-width: 960px;
margin: 0 auto;
padding: 0 8px;

img {
margin-right: $size0;
.blurb {
box-shadow: inset 0 -1px 0 $borderColor;
background-color: #fff;
margin-bottom: 16px;
font-size: 18px;
}
}

.notebook-container {
.profile-blurb {
max-width: 960px;
margin: 0 auto;
padding: 0 8px;
Expand All @@ -633,6 +653,11 @@ header {
background-color: #fff;
margin-bottom: 16px;
}

p {
font-size: 18px;
font-weight: 100;
}
}

.most-recent {
Expand Down
122 changes: 102 additions & 20 deletions website/nb.ts
Original file line number Diff line number Diff line change
Expand Up @@ -403,10 +403,12 @@ export class FixedCell extends Component<FixedProps, CellState> {
export interface NotebookRootProps {
userInfo?: db.UserInfo;
nbId?: string;
uid?: string;
}

export interface NotebookRootState {
nbId?: string;
uid?: string;
}

export class NotebookRoot extends Component<NotebookRootProps,
Expand All @@ -422,7 +424,15 @@ export class NotebookRoot extends Component<NotebookRootProps,
nbId = matches ? matches[1] : null;
}

this.state = { nbId };
let uid;
if (this.props.uid) {
uid = this.props.uid;
} else {
const matches = window.location.search.match(/profile=(\w+)/);
uid = matches ? matches[1] : null;
}

this.state = { nbId, uid };
}

render() {
Expand All @@ -432,6 +442,11 @@ export class NotebookRoot extends Component<NotebookRootProps,
nbId: this.state.nbId,
userInfo: this.props.userInfo,
});
} else if (this.state.uid) {
body = h(Profile, {
uid: this.state.uid,
userInfo: this.props.userInfo
});
} else {
body = h(MostRecent, null);
}
Expand All @@ -448,10 +463,11 @@ export class NotebookRoot extends Component<NotebookRootProps,

export interface MostRecentState {
latest: db.NbInfo[];
own: db.NbInfo[];
}

function nbUrl(nbId: string): string {
// Careful, S3 is finicy about what URLs it serves. So
// Careful, S3 is finicky about what URLs it serves. So
// /notebook?nbId=blah will get redirect to /notebook/
// because it is a directory with an index.html in it.
const u = window.location.origin + "/notebook/?nbId=" + nbId;
Expand All @@ -464,7 +480,8 @@ export class MostRecent extends Component<any, MostRecentState> {
// This is to avoiding calling into firebase during static HTML generation.
if (IS_WEB) {
const latest = await db.active.queryLatest();
this.setState({latest});
const own = await db.active.queryProfile(this.props.uid);
this.setState({latest, own});
}
}

Expand All @@ -476,31 +493,78 @@ export class MostRecent extends Component<any, MostRecentState> {
}

render() {
if (!this.state.latest) {
if (!this.state.latest || !this.state.own) {
return h(Loading, null);
}
const notebookList = this.state.latest.map(info => {
const snippit = db.getInputCodes(info.doc).join("\n").slice(0, 100);
const href = nbUrl(info.nbId);
return h("a", { href },
h("li", null,
h("div", { "class": "code-snippit" }, snippit),
notebookBlurb(info.doc, false),
),
);
});
const empty = !this.state.own ?
h("p", {"class": "most-recent-header"}, "Nothing to show yet.") : "";
return h("div", { "class": "most-recent" },
h("div", {"class": "most-recent-header"},
h("div", {"class": "most-recent-header-title"},
h("h2", null, "Recently Updated"),
h("h2", null, "Your Most Recent Notebooks"),
),
h("div", {"class": "most-recent-header-cta"},
h("button", { "class": "create-notebook",
"onClick": () => this.onCreate(),
}, "+ New Notebook"),
),
),
h("ol", null, ...notebookList),
empty,
h("ol", null, ...notebookList(this.state.own.slice(0, 3))),
h("div", {"class": "most-recent-header"},
h("div", {"class": "most-recent-header-title"},
h("h2", null, "Recently Updated"),
),
),
h("ol", null, ...notebookList(this.state.latest)),
);
}
}

export interface ProfileState {
latest: db.NbInfo[];
profile: string;
}

export class Profile extends Component<any, ProfileState> {
async componentWillMount() {
// Only query firebase when in the browser.
// This is to avoiding calling into firebase during static HTML generation.
if (IS_WEB) {
const latest = await db.active.queryProfile(this.props.uid);
this.setState({ latest });
}
}

render() {
if (!this.state.latest) {
return h(Loading, null);
}
const doc = this.state.latest[0].doc;
const profileBlurb = h("div", { "class": "blurb" }, null, [
h("div", { "class": "blurb-avatar" },
h(Avatar, { userInfo: doc.owner }),
),
h("div", { "class": "blurb-name" },
h("p", { "class": "displayName" }, doc.owner.displayName),
),
h("div", { "class": "date-created" },
h("p", { "class": "created" },
`Most Recent Update ${fmtDate(doc.updated)}.`),
),
]);

return h("div", null,
h("div", {"class": "profile-blurb"}, profileBlurb),
h("div", { "class": "most-recent" },
h("div", {"class": "most-recent-header"},
h("div", {"class": "most-recent-header-title"},
h("h2", null,
doc.owner.displayName + "'s Recently Updated Notebooks")
),
),
h("ol", null, ...notebookList(this.state.latest)),
)
);
}
}
Expand Down Expand Up @@ -567,11 +631,11 @@ export class Notebook extends Component<NotebookProps, NotebookState> {
doc.title = this.state.typedTitle;
this.setState({ ...doc, editingTitle: false });
this.update(doc);
}
}

async onTypedTitle(event) {
this.setState({ typedTitle: event.target.value });
}
this.setState({ typedTitle: event.target.value });
}

async onDelete(i) {
const doc = this.state.doc;
Expand Down Expand Up @@ -687,6 +751,19 @@ export class Notebook extends Component<NotebookProps, NotebookState> {
}
}

function notebookList(notebooks: db.NbInfo[]): JSX.Element[] {
return notebooks.map(info => {
const snippit = db.getInputCodes(info.doc).join("\n").slice(0, 100);
const href = nbUrl(info.nbId);
return h("a", { href },
h("li", null,
h("div", { class: "code-snippit" }, snippit),
notebookBlurb(info.doc, false)
)
);
});
}

function notebookBlurb(doc: db.NotebookDoc, showDates = true): JSX.Element {
const dates = !showDates ? [] : [
h("div", { "class": "date-created" },
Expand All @@ -696,12 +773,17 @@ function notebookBlurb(doc: db.NotebookDoc, showDates = true): JSX.Element {
h("p", { "class": "updated" }, `Updated ${fmtDate(doc.updated)}.`),
),
];
const profileUrl = window.location.origin + "/notebook/?profile=" +
doc.owner.uid;
return h("div", { "class": "blurb" }, null, [
h("div", { "class": "blurb-avatar" },
h(Avatar, { userInfo: doc.owner }),
),
h("div", { "class": "blurb-name" },
h("p", { "class": "displayName" }, doc.owner.displayName),
h("a", {
"class": "displayName",
"href": profileUrl
}, doc.owner.displayName),
),
...dates
]);
Expand Down

0 comments on commit 555bd14

Please sign in to comment.