Merge branch 'os1-rc' of https://github.com/urbit/urbit into m/link-meta

This commit is contained in:
Fang 2020-03-03 00:16:41 +01:00
commit a382f7d41b
No known key found for this signature in database
GPG Key ID: EB035760C1BBA972
54 changed files with 527 additions and 270 deletions

View File

@ -202,15 +202,21 @@
?- -.act ?- -.act
%create %create
?> ?=(^ app-path.act) ?> ?=(^ app-path.act)
?> |(=(group-path.act app-path.act) =(~(tap in members.act) ~))
?^ (chat-scry app-path.act) ?^ (chat-scry app-path.act)
~& %chat-already-exists ~& %chat-already-exists
~ ~
%- zing %- zing
:~ (create-chat app-path.act security.act allow-history.act) :~ (create-chat app-path.act security.act allow-history.act)
(create-managed-group group-path.act security.act members.act) %- create-group
:* group-path.act
app-path.act
security.act
members.act
title.act
description.act
==
(create-metadata title.act description.act group-path.act app-path.act) (create-metadata title.act description.act group-path.act app-path.act)
(create-security group-path.act security.act)
~[(permission-hook-poke [%add-owned group-path.act group-path.act])]
== ==
:: ::
%delete %delete
@ -223,9 +229,7 @@
== ==
:: ::
?: (is-managed group-path) ~ ?: (is-managed group-path) ~
:~ (permission-hook-poke [%remove group-path]) :~ (group-poke [%unbundle group-path])
(permission-poke [%delete group-path])
(group-poke [%unbundle group-path])
(metadata-hook-poke [%remove group-path]) (metadata-hook-poke [%remove group-path])
== ==
== ==
@ -247,18 +251,40 @@
(chat-hook-poke [%add-owned path security history]) (chat-hook-poke [%add-owned path security history])
== ==
:: ::
++ create-managed-group ++ create-group
|= [=path security=rw-security ships=(set ship)] |= [=path app-path=path sec=rw-security ships=(set ship) title=@t desc=@t]
^- (list card) ^- (list card)
?^ (group-scry path) ~ =/ group (group-scry path)
?^ group
%- zing
%+ turn ~(tap in u.group)
|= =ship
?: =(ship our.bol) ~
[(send-invite app-path ship)]~
:: do not create a managed group if this is a sig path or a blacklist :: do not create a managed group if this is a sig path or a blacklist
:: ::
?: =(security %channel) ?: =(sec %channel)
~[(group-poke [%bundle path])] :~ (group-poke [%bundle path])
(create-security path sec)
(permission-hook-poke [%add-owned path path])
==
?: (is-managed path) ?: (is-managed path)
~[(contact-view-poke [%create path ships])] ~[(contact-view-poke [%create path ships title desc])]
:~ (group-poke [%bundle path]) :~ (group-poke [%bundle path])
(group-poke [%add ships path]) (group-poke [%add ships path])
(create-security path sec)
(permission-hook-poke [%add-owned path path])
==
::
++ create-security
|= [pax=path sec=rw-security]
^- card
?+ sec !!
%channel
(perm-group-hook-poke [%associate pax [[pax %black] ~ ~]])
::
%village
(perm-group-hook-poke [%associate pax [[pax %white] ~ ~]])
== ==
:: ::
++ create-metadata ++ create-metadata
@ -278,19 +304,8 @@
(metadata-hook-poke [%add-owned group-path]) (metadata-hook-poke [%add-owned group-path])
== ==
:: ::
++ create-security
|= [pax=path sec=rw-security]
^- (list card)
?+ sec ~
%channel
~[(perm-group-hook-poke [%associate pax [[pax %black] ~ ~]])]
::
%village
~[(perm-group-hook-poke [%associate pax [[pax %white] ~ ~]])]
==
::
++ contact-view-poke ++ contact-view-poke
|= act=[%create =path ships=(set ship)] |= act=[%create =path ships=(set ship) title=@t description=@t]
^- card ^- card
[%pass / %agent [our.bol %contact-view] %poke %contact-view-action !>(act)] [%pass / %agent [our.bol %contact-view] %poke %contact-view-action !>(act)]
:: ::
@ -308,7 +323,7 @@
!>(act) !>(act)
== ==
:: ::
++ send-invite-poke ++ send-invite
|= [=path =ship] |= [=path =ship]
^- card ^- card
=/ =invite =/ =invite

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1,6 +1,11 @@
:: contact-hook: :: contact-hook:
:: ::
/- *group-store, *group-hook, *contact-hook, *invite-store /- *group-store,
*group-hook,
*contact-hook,
*invite-store,
*metadata-hook,
*metadata-store
/+ *contact-json, default-agent /+ *contact-json, default-agent
|% |%
+$ card card:agent:gall +$ card card:agent:gall
@ -242,7 +247,10 @@
%delete %delete
=. synced (~(del by synced) path.fact) =. synced (~(del by synced) path.fact)
:_ state :_ state
[(group-poke [%unbundle path.fact])]~ :~ (group-poke [%unbundle path.fact])
(metadata-hook-poke [%remove path.fact])
(metadata-poke [%remove path.fact [%contacts path.fact]])
==
== ==
:: ::
++ foreign ++ foreign
@ -352,7 +360,9 @@
(poke-hook-action [%add-synced ship.invite.fact path.invite.fact]) (poke-hook-action [%add-synced ship.invite.fact path.invite.fact])
:- :-
%+ welp %+ welp
[(group-hook-poke [%add ship.invite.fact path.invite.fact])]~ :~ (group-hook-poke [%add ship.invite.fact path.invite.fact])
(metadata-hook-poke [%add-synced ship.invite.fact path.invite.fact])
==
-.changes -.changes
+.changes +.changes
== ==
@ -377,6 +387,16 @@
^- card ^- card
[%pass / %agent [our.bol %group-store] %poke %group-action !>(act)] [%pass / %agent [our.bol %group-store] %poke %group-action !>(act)]
:: ::
++ metadata-poke
|= act=metadata-action
^- card
[%pass / %agent [our.bol %metadata-store] %poke %metadata-action !>(act)]
::
++ metadata-hook-poke
|= act=metadata-hook-action
^- card
[%pass / %agent [our.bol %metadata-hook] %poke %metadata-hook-action !>(act)]
::
++ contacts-scry ++ contacts-scry
|= pax=path |= pax=path
^- (unit contacts) ^- (unit contacts)

View File

@ -1,7 +1,14 @@
:: contact-view: sets up contact JS client and combines commands :: contact-view: sets up contact JS client and combines commands
:: into semantic actions for the UI :: into semantic actions for the UI
:: ::
/- *group-store, *group-hook, *invite-store, *contact-hook /- *group-store,
*group-hook,
*invite-store,
*contact-hook,
*metadata-store,
*metadata-hook,
*permission-group-hook,
*permission-hook
/+ *server, *contact-json, base64, default-agent /+ *server, *contact-json, base64, default-agent
/= index /= index
/^ octs /^ octs
@ -126,18 +133,24 @@
?- -.act ?- -.act
%create %create
?> ?=([@ *] path.act) ?> ?=([@ *] path.act)
%+ weld
:~ (group-poke [%bundle path.act]) :~ (group-poke [%bundle path.act])
(contact-poke [%create path.act]) (contact-poke [%create path.act])
(contact-hook-poke [%add-owned path.act]) (contact-hook-poke [%add-owned path.act])
(group-hook-poke [%add our.bol path.act]) (group-hook-poke [%add our.bol path.act])
(group-poke [%add (~(put in ships.act) our.bol) path.act]) (group-poke [%add (~(put in ships.act) our.bol) path.act])
(perm-group-hook-poke [%associate path.act [[path.act %white] ~ ~]])
(permission-hook-poke [%add-owned path.act path.act])
== ==
(create-metadata path.act title.act description.act)
:: ::
%delete %delete
%+ weld
:~ (group-poke [%unbundle path.act]) :~ (group-poke [%unbundle path.act])
(contact-poke [%delete path.act]) (contact-poke [%delete path.act])
(contact-hook-poke [%remove path.act]) (contact-hook-poke [%remove path.act])
== ==
(delete-metadata path.act)
:: ::
%remove %remove
:~ (group-poke [%remove [ship.act ~ ~] path.act]) :~ (group-poke [%remove [ship.act ~ ~] path.act])
@ -218,6 +231,51 @@
^- card ^- card
[%pass / %agent [our.bol %group-hook] %poke %group-hook-action !>(act)] [%pass / %agent [our.bol %group-hook] %poke %group-hook-action !>(act)]
:: ::
++ metadata-poke
|= act=metadata-action
^- card
[%pass / %agent [our.bol %metadata-store] %poke %metadata-action !>(act)]
::
++ metadata-hook-poke
|= act=metadata-hook-action
^- card
[%pass / %agent [our.bol %metadata-hook] %poke %metadata-hook-action !>(act)]
::
++ perm-group-hook-poke
|= act=permission-group-hook-action
^- card
:* %pass / %agent [our.bol %permission-group-hook]
%poke %permission-group-hook-action !>(act)
==
::
++ permission-hook-poke
|= act=permission-hook-action
^- card
:* %pass / %agent [our.bol %permission-hook]
%poke %permission-hook-action !>(act)
==
::
++ create-metadata
|= [=path title=@t description=@t]
^- (list card)
=/ =metadata
%* . *metadata
title title
description description
date-created now.bol
creator our.bol
==
:~ (metadata-poke [%add path [%contacts path] metadata])
(metadata-hook-poke [%add-owned path])
==
::
++ delete-metadata
|= =path
^- (list card)
:~ (metadata-poke [%remove path [%contacts path]])
(metadata-hook-poke [%remove path])
==
::
++ all-scry ++ all-scry
^- rolodex ^- rolodex
.^(rolodex %gx /=contact-store/(scot %da now.bol)/all/noun) .^(rolodex %gx /=contact-store/(scot %da now.bol)/all/noun)

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -196,7 +196,7 @@
=/ book=notebook-info =/ book=notebook-info
[title.old '' =(%open comments.old) / /] [title.old '' =(%open comments.old) / /]
=+ ^- [grp-car=(list card) write-pax=path read-pax=path] =+ ^- [grp-car=(list card) write-pax=path read-pax=path]
(make-groups:main book-name group-pax ~ %.n %.n) (make-groups:main book-name [group-pax ~ %.n %.n] '' '')
=. writers.book write-pax =. writers.book write-pax
=. subscribers.book read-pax =. subscribers.book read-pax
=/ inv-car (send-invites book-name (~(get ju old-subs) book-name)) =/ inv-car (send-invites book-name (~(get ju old-subs) book-name))
@ -563,7 +563,7 @@
=+ ^- [grp-car=(list card) write-pax=path read-pax=path] =+ ^- [grp-car=(list card) write-pax=path read-pax=path]
?: =(writers.new-book /) ?: =(writers.new-book /)
=/ group-path /~/publish/(scot %p our.bol)/[book-name] =/ group-path /~/publish/(scot %p our.bol)/[book-name]
(make-groups book-name group-path ~ %.n %.n) (make-groups book-name [group-path ~ %.n %.n] '' '')
[~ writers.info subscribers.info] [~ writers.info subscribers.info]
=. writers.new-book write-pax =. writers.new-book write-pax
=. subscribers.new-book read-pax =. subscribers.new-book read-pax
@ -876,7 +876,8 @@
[%pass / %agent [our.bol %group-hook] %poke %group-hook-action !>(act)] [%pass / %agent [our.bol %group-hook] %poke %group-hook-action !>(act)]
:: ::
++ contact-view-create ++ contact-view-create
|= act=[%create path (set ship)] |= [=path ships=(set ship) title=@t description=@t]
=/ act [%create path ships title description]
^- card ^- card
[%pass / %agent [our.bol %contact-view] %poke %contact-view-action !>(act)] [%pass / %agent [our.bol %contact-view] %poke %contact-view-action !>(act)]
:: ::
@ -923,18 +924,6 @@
(perm-group-hook-poke [%associate write [[write write-type] ~ ~]]) (perm-group-hook-poke [%associate write [[write write-type] ~ ~]])
== ==
:: ::
++ create-managed-group
|= [pax=path security=rw-security ships=(set ship)]
^- (list card)
=/ grp
.^((unit group) %gx ;:(weld /=group-store/(scot %da now.bol) pax /noun))
?^ grp
~
?> ?=(^ pax)
?: |(=('~' i.pax) !=(%village security))
[(group-poke [%bundle pax])]~
[(contact-view-create [%create pax ships])]~
::
++ generate-invites ++ generate-invites
|= [book=@tas invitees=(set ship)] |= [book=@tas invitees=(set ship)]
^- (list card) ^- (list card)
@ -949,7 +938,7 @@
[%pass / %agent [our.bol %invite-hook] %poke %invite-action !>(act)] [%pass / %agent [our.bol %invite-hook] %poke %invite-action !>(act)]
:: ::
++ make-groups ++ make-groups
|= [book=@tas group=group-info] |= [book=@tas group=group-info title=@t about=@t]
^- [(list card) write=path read=path] ^- [(list card) write=path read=path]
?> ?=(^ group-path.group) ?> ?=(^ group-path.group)
?: use-preexisting.group ?: use-preexisting.group
@ -976,7 +965,7 @@
=/ whole-grp (~(put in invitees.group) our.bol) =/ whole-grp (~(put in invitees.group) our.bol)
:_ [group-path.group group-path.group] :_ [group-path.group group-path.group]
%- zing %- zing
:~ [(contact-view-create [%create group-path.group whole-grp])]~ :~ [(contact-view-create [group-path.group whole-grp title about])]~
(create-security group-path.group group-path.group %village) (create-security group-path.group group-path.group %village)
[(perm-hook-poke [%add-owned group-path.group group-path.group])]~ [(perm-hook-poke [%add-owned group-path.group group-path.group])]~
(generate-invites book (~(del in invitees.group) our.bol)) (generate-invites book (~(del in invitees.group) our.bol))
@ -1013,7 +1002,7 @@
?: (~(has by books) book.act) ?: (~(has by books) book.act)
~|("notebook already exists: {<book.act>}" !!) ~|("notebook already exists: {<book.act>}" !!)
=+ ^- [cards=(list card) write-pax=path read-pax=path] =+ ^- [cards=(list card) write-pax=path read-pax=path]
(make-groups book.act group.act) (make-groups book.act group.act title.act about.act)
=/ new-book=notebook-info =/ new-book=notebook-info
:* title.act :* title.act
about.act about.act
@ -1086,7 +1075,7 @@
=+ ^- [cards=(list card) write-pax=path read-pax=path] =+ ^- [cards=(list card) write-pax=path read-pax=path]
?~ group.act ?~ group.act
[~ writers.u.book subscribers.u.book] [~ writers.u.book subscribers.u.book]
(make-groups book.act u.group.act) (make-groups book.act u.group.act title.act about.act)
=/ new-info=notebook-info =/ new-info=notebook-info
:* title.act :* title.act
about.act about.act

View File

@ -109,6 +109,8 @@
%- ot %- ot
:~ [%path pa] :~ [%path pa]
[%ships (as (su ;~(pfix sig fed:ag)))] [%ships (as (su ;~(pfix sig fed:ag)))]
[%title so]
[%description so]
== ==
:: ::
++ delete (ot [%path pa]~) ++ delete (ot [%path pa]~)

View File

@ -3,7 +3,7 @@
+$ contact-view-action +$ contact-view-action
$% :: %create: create in both groups and contacts $% :: %create: create in both groups and contacts
:: ::
[%create =path ships=(set ship)] [%create =path ships=(set ship) title=@t description=@t]
:: %remove: remove from both groups and contacts :: %remove: remove from both groups and contacts
:: ::
[%remove =path =ship] [%remove =path =ship]

View File

@ -276,7 +276,7 @@ h2 {
a { a {
color: #fff; color: #fff;
} }
.hover-black-d:hover { .hover-bg-gray1-d:hover {
color: #000; color: #4d4d4d;
} }
} }

View File

@ -78,11 +78,11 @@ class UrbitApi {
} }
groupsAction(data) { groupsAction(data) {
this.action("group-store", "group-action", data); return this.action("group-store", "group-action", data);
} }
groupAdd(members, path) { groupAdd(members, path) {
this.groupsAction({ return this.groupsAction({
add: { add: {
members, path members, path
} }

View File

@ -33,6 +33,7 @@ export class ChatScreen extends Component {
componentDidMount() { componentDidMount() {
this.updateReadNumber(); this.updateReadNumber();
this.scrollToBottom("auto");
} }
componentWillUnmount() { componentWillUnmount() {

View File

@ -1,40 +1,25 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import classnames from 'classnames'; import { InviteSearch } from './invite-search';
import { Sigil } from '/components/lib/icons/sigil';
import { deSig } from '/lib/util';
import urbitOb from 'urbit-ob';
export class InviteElement extends Component { export class InviteElement extends Component {
constructor(props) { constructor(props) {
super(props); super(props);
this.state = { this.state = {
members: '', members: [],
error: false, error: false,
success: false success: false
}; };
this.setInvite = this.setInvite.bind(this);
} }
modifyMembers() { modifyMembers() {
const { props, state } = this; const { props, state } = this;
let aud = []; let aud = state.members.map(mem => `~${mem}`);
let isValid = true;
if (state.members.length > 2) {
aud = state.members
.split(',')
.map((mem) => `~${deSig(mem.trim())}`);
aud.forEach((mem) => { if (state.members.length === 0) {
if (!urbitOb.isValidPatp(mem)) {
isValid = false;
}
});
}
if (!isValid || (state.members.length > 0 && state.members.length < 3)) {
this.setState({ this.setState({
error: true, error: true,
success: false success: false
@ -42,40 +27,27 @@ export class InviteElement extends Component {
return; return;
} }
if (this.textarea) { props.api.setSpinner(true);
this.textarea.value = '';
}
this.setState({ this.setState({
error: false, error: false,
success: true, success: true,
members: '' members: []
}, () => { }, () => {
props.api.groups.add(aud, props.path); props.api.groups.add(aud, props.path).then(() => {
props.api.setSpinner(false);
});
}); });
} }
modifyMembersChange(e) { setInvite(invite) {
this.setState({ this.setState({members: invite.ships});
members: e.target.value
});
} }
render() { render() {
const { props, state } = this; const { props, state } = this;
let errorElem = !!state.error ? (
<p className="pt2 red2 f8">Invalid ship name.</p>
) : (
<div></div>
);
let successElem = !!state.success ? ( let modifyButtonClasses = "mt4 db f9 ba pa2 white-d bg-gray0-d b--black b--gray2-d pointer";
<p className="pt2 green2 f8">Success!</p>
) : (
<div></div>
);
let modifyButtonClasses = "db f9 ba pa2 white-d bg-gray0-d b--black b--gray2-d pointer";
if (state.error) { if (state.error) {
modifyButtonClasses = modifyButtonClasses + ' gray3'; modifyButtonClasses = modifyButtonClasses + ' gray3';
} }
@ -89,23 +61,21 @@ export class InviteElement extends Component {
return ( return (
<div> <div>
<textarea <InviteSearch
ref={ e => { this.textarea = e; } } groups={{}}
className="f7 mono ba b--gray3 bg-gray0-d white-d pa3 mb4 db w-100" contacts={props.contacts}
style={{ groupResults={false}
resize: 'none', invites={{
height: 50 groups: [],
ships: this.state.members
}} }}
spellCheck="false" setInvite={this.setInvite}
placeholder="~zod, ~bus" />
onChange={this.modifyMembersChange.bind(this)}></textarea>
<button <button
onClick={this.modifyMembers.bind(this)} onClick={this.modifyMembers.bind(this)}
className={modifyButtonClasses}> className={modifyButtonClasses}>
{buttonText} {buttonText}
</button> </button>
{errorElem}
{successElem}
</div> </div>
); );
} }

View File

@ -222,8 +222,8 @@ export class InviteSearch extends Component {
<li <li
key={group} key={group}
className={ className={
"list mono white-d f8 pv2 ph3 pointer" + "list mono mix-blend-diff white f8 pv2 ph3 pointer" +
" hover-bg-gray4 hover-black-d" " hover-bg-gray4 hover-bg-gray1-d"
} }
onClick={e => this.addGroup(group)}> onClick={e => this.addGroup(group)}>
{group} {group}
@ -233,14 +233,16 @@ export class InviteSearch extends Component {
let shipResults = state.searchResults.ships.map(ship => { let shipResults = state.searchResults.ships.map(ship => {
let nicknames = (this.state.contacts.has(ship)) let nicknames = (this.state.contacts.has(ship))
? this.state.contacts.get(ship).join(", ") ? this.state.contacts.get(ship)
.filter(e => { return !(e === "") })
.join(", ")
: ""; : "";
return ( return (
<li <li
key={ship} key={ship}
className={ className={
"list mono white-d f8 pv1 ph3 pointer" + "list mono white-d f8 pv1 ph3 pointer" +
" hover-bg-gray4 hover-black-d relative" " hover-bg-gray4 hover-bg-gray1-d relative"
} }
onClick={e => this.addShip(ship)}> onClick={e => this.addShip(ship)}>
<Sigil <Sigil
@ -249,8 +251,8 @@ export class InviteSearch extends Component {
color="#000000" color="#000000"
classes="mix-blend-diff v-mid" classes="mix-blend-diff v-mid"
/> />
<span className="v-mid ml2 mw5 truncate dib">{"~" + ship}</span> <span className="v-mid ml2 mw5 truncate dib mix-blend-diff white">{"~" + ship}</span>
<span className="absolute right-1 di truncate mw4 inter f9 pt1">{nicknames}</span> <span className="absolute right-1 di truncate mw4 inter f9 pt1 mix-blend-diff white">{nicknames}</span>
</li> </li>
); );
}); });

View File

@ -42,6 +42,8 @@ export class SidebarItem extends Component {
return lett.url; return lett.url;
} else if ('code' in lett) { } else if ('code' in lett) {
return lett.code.expression; return lett.code.expression;
} else if ('me' in lett) {
return lett.me;
} else { } else {
return ''; return '';
} }
@ -51,35 +53,41 @@ export class SidebarItem extends Component {
const { props, state } = this; const { props, state } = this;
let unreadElem = !!props.unread let unreadElem = !!props.unread
? "fw7 green2" ? "green2"
: ""; : "";
let title = props.title.substr(1); let title = props.title;
let description = this.getLetter(props.description); let box = props.box.substr(1);
let latest = this.getLetter(props.latest);
let selectedCss = !!props.selected ? 'bg-gray5 bg-gray1-d gray3-d' : 'bg-white bg-gray0-d gray3-d pointer'; let selectedCss = !!props.selected ? 'bg-gray5 bg-gray1-d gray3-d' : 'bg-white bg-gray0-d gray3-d pointer';
let authorCss = (props.nickname === props.ship)
? "mono" : "";
let author = (props.nickname === props.ship)
? `~${props.ship}` : props.nickname;
return ( return (
<div <div
className={"z1 pa3 pt4 pb4 bb b--gray4 b--gray1-d " + selectedCss} className={"z1 pa3 pt2 pb2 bb b--gray4 b--gray1-d " + selectedCss}
onClick={this.onClick.bind(this)}> onClick={this.onClick.bind(this)}>
<div className="w-100 v-mid"> <div className="w-100 v-mid">
<p className={"dib mono f8 " + unreadElem }> <p className="dib f8">
<span className={(unreadElem === "") ? "gray3 gray2-d" : ""}> {title}
{title.substr(0, title.indexOf("/"))}/
</span>
{title.substr(title.indexOf("/") + 1)}
</p> </p>
<p className="f8 db mono gray3 gray2-d pt1">{box}</p>
</div> </div>
<div className="w-100 pt1"> <div className="w-100 pt3">
<p className="dib mono f9 mr3"> <p className={((unreadElem === "") ? "black white-d" : "") +
{props.ship === "" ? "" : "~"} unreadElem + " dib f9 mr3 mw4 truncate v-mid " + authorCss}>
{props.ship} {(author === "~") ? "" : author}
</p> </p>
<p className="dib mono f9 gray3">{state.timeSinceNewestMessage}</p> <p className="dib mono f9 gray3 v-mid">{state.timeSinceNewestMessage}</p>
</div> </div>
<p className="f8 clamp-3 pt2">{description}</p> <p className="f8 clamp-3 pt1">{latest}</p>
</div> </div>
); );
} }

View File

@ -29,9 +29,12 @@ export class MemberScreen extends Component {
modifyText = 'Invite someone to this chat.'; modifyText = 'Invite someone to this chat.';
} }
let contacts = (props.station in props.contacts)
? props.contacts[props.station] : {};
let members = perm.map((mem) => { let members = perm.map((mem) => {
let contact = (mem in props.contacts) let contact = (mem in contacts)
? props.contacts[mem] : false; ? contacts[mem] : false;
return ( return (
<MemberElement <MemberElement
@ -91,6 +94,7 @@ export class MemberScreen extends Component {
<InviteElement <InviteElement
path={props.station} path={props.station}
permissions={props.permission} permissions={props.permission}
contacts={props.contacts}
api={props.api} api={props.api}
/> />
) : null} ) : null}

View File

@ -46,13 +46,16 @@ export class Root extends Component {
let contacts = !!state.contacts ? state.contacts : {}; let contacts = !!state.contacts ? state.contacts : {};
const renderChannelSidebar = (props) => ( const renderChannelSidebar = (props, station) => (
<Sidebar <Sidebar
inbox={state.inbox} inbox={state.inbox}
messagePreviews={messagePreviews} messagePreviews={messagePreviews}
associations={state.associations || new Map}
contacts={contacts}
invites={invites} invites={invites}
unreads={unreads} unreads={unreads}
api={api} api={api}
station={station}
{...props} {...props}
/> />
); );
@ -161,7 +164,7 @@ export class Root extends Component {
spinner={state.spinner} spinner={state.spinner}
popout={popout} popout={popout}
sidebarShown={state.sidebarShown} sidebarShown={state.sidebarShown}
sidebar={renderChannelSidebar(props)} sidebar={renderChannelSidebar(props, station)}
> >
<ChatScreen <ChatScreen
station={station} station={station}
@ -200,8 +203,6 @@ export class Root extends Component {
}; };
let popout = props.match.url.includes("/popout/"); let popout = props.match.url.includes("/popout/");
let roomContacts = (station in contacts)
? contacts[station] : {};
return ( return (
<Skeleton <Skeleton
@ -216,7 +217,7 @@ export class Root extends Component {
api={api} api={api}
station={station} station={station}
permission={permission} permission={permission}
contacts={roomContacts} contacts={contacts}
permissions={state.permissions} permissions={state.permissions}
popout={popout} popout={popout}
sidebarShown={state.sidebarShown} sidebarShown={state.sidebarShown}

View File

@ -19,7 +19,6 @@ export class Sidebar extends Component {
render() { render() {
const { props, state } = this; const { props, state } = this;
let station = `/${props.match.params.ship}/${props.match.params.station}`;
let sidebarInvites = Object.keys(props.invites) let sidebarInvites = Object.keys(props.invites)
.map((uid) => { .map((uid) => {
@ -39,14 +38,28 @@ export class Sidebar extends Component {
: {text: 'No messages yet'}; : {text: 'No messages yet'};
let author = !!msg ? msg.author : ''; let author = !!msg ? msg.author : '';
let when = !!msg ? msg.when : 0; let when = !!msg ? msg.when : 0;
let title = box;
if ((props.associations.has(box)) && (props.associations.get(box).metadata)) {
title = (props.associations.get(box).metadata.title)
? props.associations.get(box).metadata.title : box;
}
let nickname = author;
if ((props.contacts[box]) && (author in props.contacts[box])) {
nickname = (props.contacts[box][author].nickname)
? props.contacts[box][author].nickname : author;
}
return { return {
msg, msg,
when, when,
author, author,
nickname,
letter, letter,
box, box,
title: box, title: title,
selected: station === box selected: props.station === box
}; };
}) })
.sort((a, b) => { .sort((a, b) => {
@ -58,10 +71,11 @@ export class Sidebar extends Component {
<SidebarItem <SidebarItem
key={obj.box + '/' + obj.when} key={obj.box + '/' + obj.when}
title={obj.title} title={obj.title}
description={obj.letter} latest={obj.letter}
box={obj.box} box={obj.box}
when={obj.when} when={obj.when}
ship={obj.author} ship={obj.author}
nickname={obj.nickname}
selected={obj.selected} selected={obj.selected}
unread={unread} unread={unread}
history={props.history} history={props.history}

View File

@ -0,0 +1,32 @@
import _ from 'lodash';
export class MetadataReducer {
reduce(json, state) {
let data = _.get(json, 'metadata-update', false);
if (data) {
this.associations(data, state);
this.add(data, state);
}
}
associations(json, state) {
let data = _.get(json, 'associations', false);
if (data) {
let metadata = new Map;
Object.keys(data).map((channel) => {
let channelObj = data[channel];
metadata.set(channelObj["app-path"], channelObj);
})
state.associations = metadata;
}
}
add(json, state) {
let data = _.get(json, 'add', false);
if (data) {
let metadata = state.associations;
metadata.set(data["app-path"], data);
state.associations = metadata;
}
}
}

View File

@ -4,6 +4,7 @@ import { ContactUpdateReducer } from '/reducers/contact-update';
import { ChatUpdateReducer } from '/reducers/chat-update'; import { ChatUpdateReducer } from '/reducers/chat-update';
import { InviteUpdateReducer } from '/reducers/invite-update'; import { InviteUpdateReducer } from '/reducers/invite-update';
import { PermissionUpdateReducer } from '/reducers/permission-update'; import { PermissionUpdateReducer } from '/reducers/permission-update';
import { MetadataReducer } from '/reducers/metadata-update.js';
import { LocalReducer } from '/reducers/local.js'; import { LocalReducer } from '/reducers/local.js';
@ -15,6 +16,7 @@ class Store {
contacts: {}, contacts: {},
permissions: {}, permissions: {},
invites: {}, invites: {},
associations: new Map,
spinner: false, spinner: false,
sidebarShown: true, sidebarShown: true,
pendingMessages: new Map([]), pendingMessages: new Map([]),
@ -27,6 +29,7 @@ class Store {
this.contactUpdateReducer = new ContactUpdateReducer(); this.contactUpdateReducer = new ContactUpdateReducer();
this.chatUpdateReducer = new ChatUpdateReducer(); this.chatUpdateReducer = new ChatUpdateReducer();
this.inviteUpdateReducer = new InviteUpdateReducer(); this.inviteUpdateReducer = new InviteUpdateReducer();
this.metadataReducer = new MetadataReducer();
this.localReducer = new LocalReducer(); this.localReducer = new LocalReducer();
this.setState = () => {}; this.setState = () => {};
} }
@ -45,6 +48,7 @@ class Store {
this.contactUpdateReducer.reduce(json, this.state); this.contactUpdateReducer.reduce(json, this.state);
this.chatUpdateReducer.reduce(json, this.state); this.chatUpdateReducer.reduce(json, this.state);
this.inviteUpdateReducer.reduce(json, this.state); this.inviteUpdateReducer.reduce(json, this.state);
this.metadataReducer.reduce(json, this.state);
this.localReducer.reduce(json, this.state); this.localReducer.reduce(json, this.state);
this.setState(this.state); this.setState(this.state);

View File

@ -113,9 +113,56 @@ a {
} }
} }
/* dark */ @media all and (prefers-color-scheme: dark) {
/* @media all and (prefers-color-scheme: dark) { body {
background-color: #333;
}
.bg-black-d {
background-color: black;
}
.white-d {
color: white;
}
.gray1-d {
color: #4d4d4d;
}
.gray2-d {
color: #7f7f7f;
}
.gray3-d {
color: #b1b2b3;
}
.gray4-d {
color: #e6e6e6;
}
.bg-gray0-d {
background-color: #333;
}
.bg-gray1-d {
background-color: #4d4d4d;
}
.b--gray0-d {
border-color: #333;
}
.b--gray2-d {
border-color: #7f7f7f;
}
.b--white-d {
border-color: #fff;
}
.invert-d { .invert-d {
filter: invert(1); filter: invert(1);
} }
} */ .o-60-d {
opacity: .6;
}
a {
color: #fff;
}
.focus-b--white-d:focus {
border-color: #fff;
}
.hover-bg-gray1-d:hover {
color: #4d4d4d;
}
}

View File

@ -69,7 +69,14 @@ class UrbitApi {
} }
contactCreate(path, ships = []) { contactCreate(path, ships = []) {
return this.contactViewAction({ create: { path, ships }}); return this.contactViewAction({
create: {
path,
ships,
title: '',
description: ''
}
});
} }
groupAdd(path, ships = []) { groupAdd(path, ships = []) {

View File

@ -313,7 +313,8 @@ export class ContactCard extends Component {
style={{ width: "fit-content" }}> style={{ width: "fit-content" }}>
<p className="f9 gray2 lh-copy">Sigil Color</p> <p className="f9 gray2 lh-copy">Sigil Color</p>
<textarea <textarea
className="b--gray4 black f7 ba db pl2 focus-b--black" className={"b--gray4 b--gray2-d black white-d bg-gray0-d f7 ba db pl2 " +
"focus-b--black focus-b--white-d"}
onChange={this.sigilColorSet} onChange={this.sigilColorSet}
defaultValue={defaultColor} defaultValue={defaultColor}
key={"default" + defaultColor} key={"default" + defaultColor}
@ -499,7 +500,8 @@ export class ContactCard extends Component {
//TODO "Share card" if it's /me -> sends to /~/default of recipient //TODO "Share card" if it's /me -> sends to /~/default of recipient
return ( return (
<div className="w-100 h-100 overflow-hidden"> <div className="w-100 h-100 overflow-hidden">
<div className="flex justify-between w-100 bg-white bb b--gray4 "> <div className={"flex justify-between w-100 bg-white bg-gray0-d " +
"bb b--gray4 b--gray2-d "}>
<div className="w-100 h2 dn-m dn-l dn-xl inter pb6 pl3 pt3 f8"> <div className="w-100 h2 dn-m dn-l dn-xl inter pb6 pl3 pt3 f8">
<Link to="/~contacts/">{"⟵"}</Link> <Link to="/~contacts/">{"⟵"}</Link>
</div> </div>
@ -512,16 +514,16 @@ export class ContactCard extends Component {
this.editToggle(); this.editToggle();
} }
}} }}
className={`ml3 mt2 mb2 f9 pa1 pointer` + ourOpt}> className={`white-d bg-gray0-d ml3 mt2 mb2 f9 pa1 pointer ` + ourOpt}>
{editInfoText} {editInfoText}
</button> </button>
<button className={`ml3 mt2 mb2 f9 pa1 pointer` + localOpt}> <button className={`white-d bg-gray0-d ml3 mt2 mb2 f9 pa1 pointer ` + localOpt}>
Share Share
</button> </button>
</div> </div>
<button <button
className={ className={
`pr4 mt4 mb4 f9 red2 pointer ` + adminOpt `bg-gray0-d pr4 mt4 mb4 f9 red2 pointer ` + adminOpt
} }
onClick={this.removeFromGroup}> onClick={this.removeFromGroup}>
{( {(
@ -530,7 +532,7 @@ export class ContactCard extends Component {
)} )}
</button> </button>
</div> </div>
<div className="h-100 w-100 overflow-x-hidden pb8">{card}</div> <div className="h-100 w-100 overflow-x-hidden pb8 white-d">{card}</div>
</div> </div>
); );
} }

View File

@ -8,7 +8,7 @@ export class ContactItem extends Component {
render() { render() {
const { props } = this; const { props } = this;
let selectedClass = (props.selected) ? "bg-gray4" : ""; let selectedClass = (props.selected) ? "bg-gray4 bg-gray1-d" : "";
let hexColor = uxToHex(props.color); let hexColor = uxToHex(props.color);
let name = (props.nickname) ? props.nickname : "~" + props.ship; let name = (props.nickname) ? props.nickname : "~" + props.ship;

View File

@ -51,7 +51,8 @@ export class ContactSidebar extends Component {
return ( return (
<div <div
key={member} key={member}
className="pl4 pt1 pb1 f9 flex justify-start content-center bg-white"> className={"pl4 pt1 pb1 f9 flex justify-start content-center " +
"bg-white bg-gray0-d"}>
<Sigil <Sigil
ship={member} ship={member}
color="#000000" color="#000000"
@ -81,7 +82,7 @@ export class ContactSidebar extends Component {
className={((props.path.includes(window.ship)) className={((props.path.includes(window.ship))
? "dib" ? "dib"
: "dn")}> : "dn")}>
<p className="f9 pl4 pt0 pt4-m pt4-l pt4-xl gray2 bn">Invite</p> <p className="f9 pl4 pt0 pt4-m pt4-l pt4-xl green2 bn">Add to Group</p>
</Link> </Link>
<Link to={detailHref} <Link to={detailHref}
className="dib dn-m dn-l dn-xl f9 pl4 pt0 pt4-m pt4-l pt4-xl gray2 bn">Channels</Link> className="dib dn-m dn-l dn-xl f9 pl4 pt0 pt4-m pt4-l pt4-xl gray2 bn">Channels</Link>

View File

@ -29,7 +29,8 @@ export class EditElement extends Component {
<div className="w-100 flex"> <div className="w-100 flex">
<textarea <textarea
ref={props.title} ref={props.title}
className="w-100 ba pl3 b--gray4 focus-b--black" className={"w-100 ba pl3 white-d bg-gray0-d b--gray4 b--gray2-d " +
"focus-b--black focus-b--white-d"}
style={ inputStyles } style={ inputStyles }
onChange={(e) => { onChange={(e) => {
let val = (' ' + e.target.value).slice(1); let val = (' ' + e.target.value).slice(1);
@ -43,7 +44,7 @@ export class EditElement extends Component {
{!!props.showButtons ? ( {!!props.showButtons ? (
<button <button
className={ className={
"f9 pointer ml3 ba pa2 pl3 pr3 b--red2 red2 " + "bg-gray0-d f9 pointer ml3 ba pa2 pl3 pr3 b--red2 red2 " +
(showDelete ? "dn" : "dib") (showDelete ? "dn" : "dib")
} }
onClick={() => { onClick={() => {
@ -57,8 +58,8 @@ export class EditElement extends Component {
{!!props.showButtons ? ( {!!props.showButtons ? (
<button <button
className={ className={
"pointer db mv2 f9 ba pa2 pl3 pr3 " + "bg-gray0-d white-d pointer db mv2 f9 ba pa2 pl3 pr3 " +
(allowSave ? "b--black" : "b--gray4 gray4") (allowSave ? "b--black b--white-d" : "b--gray4 gray4 b--gray2-d gray2-d")
} }
onClick={() => { onClick={() => {
if (!allowSave) { return; } if (!allowSave) { return; }

View File

@ -10,29 +10,27 @@ export class GroupDetail extends Component {
props.activeDrawer === "detail" ? "db" : "dn db-ns"; props.activeDrawer === "detail" ? "db" : "dn db-ns";
let groupPath = props.path || ""; let groupPath = props.path || "";
let channelsForGroup = (groupPath in props.associations) ?
props.associations[groupPath] : {};
let groupChannels = props.associations.get(groupPath) || {};
let isEmpty = Object.entries(groupChannels).length === 0 && let isEmpty = Object.keys(channelsForGroup).length === 0;
groupChannels.constructor === Object; let channelList = (<div />);
let channelList = <div/> channelList = Object.keys(channelsForGroup).map((key) => {
let channel = channelsForGroup[key];
channelList = Object.keys(groupChannels).map((channel) => { if (!('metadata' in channel)) {
let channelObj = groupChannels[channel]; return (<div />);
if (!channelObj) {
return false;
} }
let title = channelObj.metadata.title || channelObj["app-path"] || ""; let title = channel.metadata.title || channel["app-path"] || "";
let color = uxToHex(channelObj.metadata.color) || "000000"; let color = uxToHex(channel.metadata.color) || "000000";
let app = channelObj["app-name"] || "Unknown"; let app = channel["app-name"] || "Unknown";
let channelPath = channelObj["app-path"]; let channelPath = channel["app-path"];
let link = `/~${app}/join${channelPath}` let link = `/~${app}/join${channelPath}`
app = app.charAt(0).toUpperCase() + app.slice(1) app = app.charAt(0).toUpperCase() + app.slice(1)
return( return(
<li <li key={channel} className="f9 list flex pv2 w-100">
key={channel}
className="f9 list flex pv2 w-100">
<div className="dib" <div className="dib"
style={{backgroundColor: `#${color}`, height: 32, width: 32}} style={{backgroundColor: `#${color}`, height: 32, width: 32}}
></div> ></div>
@ -51,12 +49,16 @@ export class GroupDetail extends Component {
let backLink = props.location.pathname; let backLink = props.location.pathname;
backLink = backLink.slice(0, props.location.pathname.indexOf("/detail")); backLink = backLink.slice(0, props.location.pathname.indexOf("/detail"));
let emptyGroup = <div className={isEmpty ? "dt w-100 h-100" : "dn"}> let emptyGroup = (
<p className="gray2 f9 tc v-mid dtc">This group has no channels. To add a channel, invite this group using any application.</p> <div className={isEmpty ? "dt w-100 h-100" : "dn"}>
<p className="gray2 f9 tc v-mid dtc">
This group has no channels. To add a channel, invite this group using any application.
</p>
</div> </div>
);
return ( return (
<div className={"h-100 w-100 overflow-x-hidden bg-white pa4 " <div className={"h-100 w-100 overflow-x-hidden bg-white bg-gray0-d white-d pa4 "
+ responsiveClass}> + responsiveClass}>
<div className="pb5 f8 db dn-m dn-l dn-xl"> <div className="pb5 f8 db dn-m dn-l dn-xl">
<Link to={backLink}> Contacts</Link> <Link to={backLink}> Contacts</Link>

View File

@ -6,7 +6,7 @@ export class GroupItem extends Component {
render() { render() {
const { props } = this; const { props } = this;
let selectedClass = (props.selected) ? "bg-gray4" : ""; let selectedClass = (props.selected) ? "bg-gray4 bg-gray1-d" : "";
let memberCount = Math.max( let memberCount = Math.max(
props.group.size, props.group.size,
Object.keys(props.contacts).length Object.keys(props.contacts).length

View File

@ -12,7 +12,7 @@ export class GroupSidebar extends Component {
render() { render() {
const { props, state } = this; const { props, state } = this;
let selectedClass = (props.selected === "me") ? "bg-gray4" : "bg-white"; let selectedClass = (props.selected === "me") ? "bg-gray4 bg-gray1-d" : "bg-white bg-gray0-d";
let rootIdentity = <Link let rootIdentity = <Link
key={1} key={1}
@ -78,9 +78,9 @@ export class GroupSidebar extends Component {
let activeClasses = (this.props.activeDrawer === "groups") ? "" : "dn-s"; let activeClasses = (this.props.activeDrawer === "groups") ? "" : "dn-s";
return ( return (
<div className={`bn br-m br-l br-xl b--gray3 lh-copy h-100 flex-basis-100-s <div className={"bn br-m br-l br-xl b--gray2 lh-copy h-100 flex-basis-100-s " +
flex-basis-30-ns flex-shrink-0 mw5-m mw5-l mw5-xl pt3 pt0-m pt0-l pt0-xl "flex-basis-30-ns flex-shrink-0 mw5-m mw5-l mw5-xl pt3 pt0-m pt0-l pt0-xl " +
relative overflow-hidden ` + activeClasses}> "relative overflow-hidden " + activeClasses}>
{/*TODO Add invite items */} {/*TODO Add invite items */}
<a className="db dn-m dn-l dn-xl f8 pb6 pl3" href="/"> Landscape</a> <a className="db dn-m dn-l dn-xl f8 pb6 pl3" href="/"> Landscape</a>
<div className="overflow-auto pb8 h-100"> <div className="overflow-auto pb8 h-100">

View File

@ -222,8 +222,8 @@ export class InviteSearch extends Component {
<li <li
key={group} key={group}
className={ className={
"list mono white-d f8 pv2 ph3 pointer" + "list mono mix-blend-diff white f8 pv2 ph3 pointer" +
" hover-bg-gray4 hover-black-d" " hover-bg-gray4 hover-bg-gray1-d"
} }
onClick={e => this.addGroup(group)}> onClick={e => this.addGroup(group)}>
{group} {group}
@ -233,14 +233,16 @@ export class InviteSearch extends Component {
let shipResults = state.searchResults.ships.map(ship => { let shipResults = state.searchResults.ships.map(ship => {
let nicknames = (this.state.contacts.has(ship)) let nicknames = (this.state.contacts.has(ship))
? this.state.contacts.get(ship).join(", ") ? this.state.contacts.get(ship)
.filter(e => {return !(e === "")})
.join(", ")
: ""; : "";
return ( return (
<li <li
key={ship} key={ship}
className={ className={
"list mono white-d f8 pv1 ph3 pointer" + "list mono white-d f8 pv1 ph3 pointer" +
" hover-bg-gray4 hover-black-d relative" " hover-bg-gray4 hover-bg-gray1-d relative"
} }
onClick={e => this.addShip(ship)}> onClick={e => this.addShip(ship)}>
<Sigil <Sigil
@ -249,8 +251,8 @@ export class InviteSearch extends Component {
color="#000000" color="#000000"
classes="mix-blend-diff v-mid" classes="mix-blend-diff v-mid"
/> />
<span className="v-mid ml2 mw5 truncate dib">{"~" + ship}</span> <span className="v-mid ml2 mw5 truncate dib mix-blend-diff white">{"~" + ship}</span>
<span className="absolute right-1 di truncate mw4 inter f9 pt1">{nicknames}</span> <span className="absolute right-1 di truncate mw4 inter f9 pt1 mix-blend-diff white">{nicknames}</span>
</li> </li>
); );
}); });

View File

@ -13,7 +13,7 @@ export class ShareSheet extends Component {
return ( return (
<div> <div>
<p className="pt4 pb2 pl4 pr4 f8 gray2 f9">Share Your Profile</p> <p className="pt4 pb2 pl4 pr4 f8 gray2 f9">Group Identity</p>
<ContactItem <ContactItem
key={props.ship} key={props.ship}
ship={props.ship} ship={props.ship}
@ -22,11 +22,11 @@ export class ShareSheet extends Component {
path={props.path} path={props.path}
selected={props.selected} selected={props.selected}
share={true} /> share={true} />
<p className="pt2 pb3 pl4 pr4 f9"> <p className="pt2 pb3 pl4 pr4 f9 white-d">
Your personal information is hidden to others in this group Your personal information is hidden to others in this group
by default. by default.
</p> </p>
<p className="pl4 pr4 f9"> <p className="pl4 pr4 f9 white-d">
Share whenever you are ready, or edit its contents for this group. Share whenever you are ready, or edit its contents for this group.
</p> </p>
</div> </div>

View File

@ -91,7 +91,7 @@ export class NewScreen extends Component {
} }
return ( return (
<div className="h-100 w-100 flex flex-column overflow-y-scroll"> <div className="h-100 w-100 flex flex-column overflow-y-scroll white-d">
<div className="w-100 dn-m dn-l dn-xl inter pt1 pb6 pl3 pt3 f8"> <div className="w-100 dn-m dn-l dn-xl inter pt1 pb6 pl3 pt3 f8">
<Link to="/~contacts/">{"⟵ All Groups"}</Link> <Link to="/~contacts/">{"⟵ All Groups"}</Link>
</div> </div>
@ -102,7 +102,8 @@ export class NewScreen extends Component {
Alphanumeric characters and hyphens only Alphanumeric characters and hyphens only
</p> </p>
<textarea <textarea
className="f7 ba b--gray3 w-100 pa3 ml3 mt2 focus-b--black" className={"f7 bg-gray0-d white-d ba b--gray3 w-100 pa3 ml3 mt2 " +
"focus-b--black focus-b--white-d"}
rows={1} rows={1}
placeholder="example-group-name" placeholder="example-group-name"
style={{ style={{

View File

@ -34,7 +34,7 @@ export class Root extends Component {
let invites = let invites =
(!!state.invites && '/contacts' in state.invites) ? (!!state.invites && '/contacts' in state.invites) ?
state.invites['/contacts'] : {}; state.invites['/contacts'] : {};
let associations = !! state.associations ? state.associations : new Map; let associations = !! state.associations ? state.associations : {};
return ( return (
<BrowserRouter> <BrowserRouter>

View File

@ -12,17 +12,15 @@ export class MetadataReducer {
associations(json, state) { associations(json, state) {
let data = _.get(json, 'associations', false); let data = _.get(json, 'associations', false);
if (data) { if (data) {
let metadata = new Map; let metadata = {};
Object.keys(data).map((channel) => { Object.keys(data).forEach((key) => {
let channelObj = data[channel]; let val = data[key];
if (metadata.has(channelObj["group-path"])) { let groupPath = val['group-path'];
let groupMetadata = metadata.get(channelObj["group-path"]); if (!(groupPath in metadata)) {
groupMetadata[channel] = channelObj; metadata[groupPath] = {};
metadata.set(channelObj["group-path"], groupMetadata);
} else {
metadata.set(channelObj["group-path"], {[channel]: channelObj});
} }
}) metadata[groupPath][key] = val;
});
state.associations = metadata; state.associations = metadata;
} }
} }
@ -31,12 +29,12 @@ export class MetadataReducer {
let data = _.get(json, 'add', false); let data = _.get(json, 'add', false);
if (data) { if (data) {
let metadata = state.associations; let metadata = state.associations;
if (metadata.has(data["group-path"])) { if (!(data['group-path'] in metadata)) {
let groupMetadata = metadata.get(data["group-path"]); metadata[data['group-path']] = {};
groupMetadata[`${data["group-path"]}/${data["app-name"]}${data["app-path"]}`] = data;
} else {
metadata.set(data["group-path"], data);
} }
metadata[data['group-path']]
[`${data["group-path"]}/${data["app-name"]}${data["app-path"]}`] = data;
state.associations = metadata; state.associations = metadata;
} }
} }

View File

@ -12,7 +12,7 @@ class Store {
this.state = { this.state = {
contacts: {}, contacts: {},
groups: {}, groups: {},
associations: new Map, associations: {},
permissions: {}, permissions: {},
invites: {}, invites: {},
spinner: false spinner: false

View File

@ -143,7 +143,7 @@ a {
font-family: 'Source Code Pro'; font-family: 'Source Code Pro';
} }
.CodeMirror-selected { background:#BAE3FE !important; } .CodeMirror-selected { background:#BAE3FE !important; color: black; }
.cm-s-tlon span { font-family: "Source Code Pro"} .cm-s-tlon span { font-family: "Source Code Pro"}
.cm-s-tlon span.cm-meta { color: var(--gray); } .cm-s-tlon span.cm-meta { color: var(--gray); }
@ -192,6 +192,14 @@ a {
color: #7F7F7F; color: #7F7F7F;
} }
.options.open {
background-color: #e6e6e6;
}
.options.closed {
background-color: white;
}
.options::after { .options::after {
content: "⌃"; content: "⌃";
transform: rotate(180deg); transform: rotate(180deg);
@ -277,3 +285,67 @@ md img {
.mix-blend-diff { .mix-blend-diff {
mix-blend-mode: difference; mix-blend-mode: difference;
} }
@media all and (prefers-color-scheme: dark) {
body {
background-color: #333;
}
.bg-black-d {
background-color: black;
}
.white-d {
color: white;
}
.gray1-d {
color: #4d4d4d;
}
.gray2-d {
color: #7f7f7f;
}
.gray3-d {
color: #b1b2b3;
}
.gray4-d {
color: #e6e6e6;
}
.bg-gray0-d {
background-color: #333;
}
.bg-gray1-d {
background-color: #4d4d4d;
}
.b--gray0-d {
border-color: #333;
}
.b--gray2-d {
border-color: #7f7f7f;
}
.b--white-d {
border-color: #fff;
}
.invert-d {
filter: invert(1);
}
.o-60-d {
opacity: .6;
}
a {
color: #fff;
}
.focus-b--white-d:focus {
border-color: #fff;
}
.hover-bg-gray1-d:hover {
background-color: #4d4d4d;
}
.options.open {
background-color: #4d4d4d;
}
.options.closed {
background-color: #333;
}
.cm-s-tlon.CodeMirror {
background: #333;
color: #fff;
}
}

View File

@ -53,7 +53,7 @@ export class CommentItem extends Component {
return ( return (
<div> <div>
<div className="flex mv3 bg-white"> <div className="flex mv3 bg-white bg-gray0-d">
<Sigil <Sigil
ship={commentData.author} ship={commentData.author}
size={24} size={24}

View File

@ -53,8 +53,8 @@ export class Comments extends Component {
let disableComment = ((this.state.commentBody === '') || (this.state.disabled === true)); let disableComment = ((this.state.commentBody === '') || (this.state.disabled === true));
let commentClass = (disableComment) let commentClass = (disableComment)
? "f9 pa2 bg-white br1 ba b--gray2 gray2" ? "bg-transparent f9 pa2 br1 ba b--gray2 gray2"
: "f9 pa2 bg-white br1 ba b--gray2 black pointer"; : "bg-transparent f9 pa2 br1 ba b--gray2 black white-d pointer";
return ( return (
<div> <div>
@ -66,7 +66,7 @@ export class Comments extends Component {
name="comment" name="comment"
placeholder="Leave a comment here" placeholder="Leave a comment here"
className={"f9 db border-box w-100 ba b--gray3 pt3 ph3 pb8 br1 " + className={"f9 db border-box w-100 ba b--gray3 pt3 ph3 pb8 br1 " +
"mb2 focus-b--black"} "b--gray2-d mb2 focus-b--black focus-b--white-d white-d bg-gray0-d"}
aria-describedby="comment-desc" aria-describedby="comment-desc"
onChange={this.commentChange}> onChange={this.commentChange}>
</textarea> </textarea>

View File

@ -43,8 +43,8 @@ export class Dropdown extends Component {
let display = (this.state.open) let display = (this.state.open)
? "block" : "none"; ? "block" : "none";
let optionsColor = (this.state.open) let optionsClass = (this.state.open)
? '#e6e6e6' : 'white'; ? "open" : "closed";
let leftAlign = ""; let leftAlign = "";
let rightAlign = "0"; let rightAlign = "0";
@ -64,14 +64,13 @@ export class Dropdown extends Component {
}); });
return ( return (
<div className="options relative dib" <div className={"options relative dib " + optionsClass}
ref={(el) => {this.optsButton = el}}> ref={(el) => {this.optsButton = el}}>
<button className="pr3 mb1 pointer br2 pa2 pr4" <button className="bg-transparent white-d pr3 mb1 pointer br2 pa2 pr4"
style={{backgroundColor: optionsColor}}
onClick={this.toggleDropdown}> onClick={this.toggleDropdown}>
{this.props.buttonText} {this.props.buttonText}
</button> </button>
<div className="absolute flex flex-column pv2 ba b--gray4 br2 z-1 bg-white" <div className="absolute flex flex-column pv2 ba b--gray4 br2 z-1 bg-white bg-gray0-d"
ref={(el) => {this.optsList = el}} ref={(el) => {this.optsList = el}}
style={{left: leftAlign, right: rightAlign, width:this.props.width, display: display}}> style={{left: leftAlign, right: rightAlign, width:this.props.width, display: display}}>
{optionsList} {optionsList}

View File

@ -26,14 +26,14 @@ export class HeaderBar extends Component {
} }
return ( return (
<div className={"bg-white w-100 justify-between relative tc pt3 " <div className={"bg-white bg-gray0-d white-d w-100 justify-between relative tc pt3 "
+ popout} + popout}
style={{ height: 40 }}> style={{ height: 40 }}>
<a className="dib gray2 f9 inter absolute left-0" <a className="dib gray2 f9 inter absolute left-0"
href='/' href='/'
style={{top: 14}}> style={{top: 14}}>
<IconHome classes={spinnerClasses}/> <IconHome classes={spinnerClasses}/>
<span className="ml2 v-top lh-title" <span className="ml2 v-top lh-title white-d"
style={{paddingTop: 3}}> style={{paddingTop: 3}}>
Home Home
</span> </span>

View File

@ -6,7 +6,10 @@ export class IconHome extends Component {
let classes = !!this.props.classes ? this.props.classes : ""; let classes = !!this.props.classes ? this.props.classes : "";
return ( return (
<img className={classes} src="/~publish/Home.png" width={16} height={16} /> <img className={"invert-d " + classes}
src="/~publish/Home.png"
width={16}
height={16} />
); );
} }
} }

View File

@ -16,7 +16,7 @@ export class SidebarSwitcher extends Component {
api.sidebarToggle(); api.sidebarToggle();
}}> }}>
<img <img
className={`pr3 invert-d dn ` + popoutSwitcher} className={`pr3 dn ` + popoutSwitcher}
src={ src={
this.props.sidebarShown this.props.sidebarShown
? "/~link/img/SwitcherOpen.png" ? "/~link/img/SwitcherOpen.png"

View File

@ -222,8 +222,8 @@ export class InviteSearch extends Component {
<li <li
key={group} key={group}
className={ className={
"list mono white-d f8 pv2 ph3 pointer" + "list mono mix-blend-diff white f8 pv2 ph3 pointer" +
" hover-bg-gray4 hover-black-d" " hover-bg-gray4 hover-bg-gray1-d"
} }
onClick={e => this.addGroup(group)}> onClick={e => this.addGroup(group)}>
{group} {group}
@ -233,14 +233,16 @@ export class InviteSearch extends Component {
let shipResults = state.searchResults.ships.map(ship => { let shipResults = state.searchResults.ships.map(ship => {
let nicknames = (this.state.contacts.has(ship)) let nicknames = (this.state.contacts.has(ship))
? this.state.contacts.get(ship).join(", ") ? this.state.contacts.get(ship)
.filter(e => { return !(e === "") })
.join(", ")
: ""; : "";
return ( return (
<li <li
key={ship} key={ship}
className={ className={
"list mono white-d f8 pv1 ph3 pointer" + "list mono white-d f8 pv1 ph3 pointer" +
" hover-bg-gray4 hover-black-d relative" " hover-bg-gray4 hover-bg-gray1-d relative"
} }
onClick={e => this.addShip(ship)}> onClick={e => this.addShip(ship)}>
<Sigil <Sigil
@ -249,8 +251,8 @@ export class InviteSearch extends Component {
color="#000000" color="#000000"
classes="mix-blend-diff v-mid" classes="mix-blend-diff v-mid"
/> />
<span className="v-mid ml2 mw5 truncate dib">{"~" + ship}</span> <span className="v-mid ml2 mw5 truncate dib mix-blend-diff white">{"~" + ship}</span>
<span className="absolute right-1 di truncate mw4 inter f9 pt1">{nicknames}</span> <span className="absolute right-1 di truncate mw4 inter f9 pt1 mix-blend-diff white">{nicknames}</span>
</li> </li>
); );
}); });

View File

@ -137,7 +137,7 @@ export class JoinScreen extends Component {
<textarea <textarea
ref={ e => { this.textarea = e; } } ref={ e => { this.textarea = e; } }
className={"f7 mono ba bg-gray0-d white-d pa3 mb2 db " + className={"f7 mono ba bg-gray0-d white-d pa3 mb2 db " +
"focus-b--black b--gray3 b--gray2-d nowrap "} "focus-b--black focus-b--white-d b--gray3 b--gray2-d nowrap "}
placeholder="~zod/dream-journal" placeholder="~zod/dream-journal"
spellCheck="false" spellCheck="false"
rows={1} rows={1}

View File

@ -105,7 +105,7 @@ export class NewPost extends Component {
popout={props.popout} popout={props.popout}
/> />
<button <button
className={"v-mid w-100 mw6 tl h1 pl4"} className={"bg-transparent v-mid w-100 mw6 tl h1 pl4"}
disabled={!state.submit} disabled={!state.submit}
style={submitStyle} style={submitStyle}
onClick={this.postSubmit}> onClick={this.postSubmit}>
@ -126,7 +126,7 @@ export class NewPost extends Component {
<input <input
autoFocus autoFocus
type="text" type="text"
className="w-100 pb2" className="bg-transparent white-d w-100 pb2"
onChange={this.titleChange} onChange={this.titleChange}
placeholder="New Post" placeholder="New Post"
/> />

View File

@ -139,7 +139,7 @@ export class NewScreen extends Component {
return ( return (
<div <div
className={ className={
"h-100 w-100 mw6 pa3 pt4 overflow-x-hidden flex flex-column" "h-100 w-100 mw6 pa3 pt4 overflow-x-hidden flex flex-column white-d"
}> }>
<div className="w-100 dn-m dn-l dn-xl inter pt1 pb6 f8"> <div className="w-100 dn-m dn-l dn-xl inter pt1 pb6 f8">
<Link to="/~publish/">{"⟵ All Notebooks"}</Link> <Link to="/~publish/">{"⟵ All Notebooks"}</Link>
@ -152,7 +152,7 @@ export class NewScreen extends Component {
</p> </p>
<textarea <textarea
className={"f7 ba bg-gray0-d white-d pa3 db w-100 " + className={"f7 ba bg-gray0-d white-d pa3 db w-100 " +
"focus-b--black b--gray3 b--gray2-d"} "focus-b--black focus-b--white-d b--gray3 b--gray2-d"}
placeholder="eg. My Journal" placeholder="eg. My Journal"
rows={1} rows={1}
style={{ style={{
@ -169,7 +169,7 @@ export class NewScreen extends Component {
<p className="f9 gray2 db mb2 pt1">What's your notebook about?</p> <p className="f9 gray2 db mb2 pt1">What's your notebook about?</p>
<textarea <textarea
className={"f7 ba bg-gray0-d white-d pa3 db w-100 " + className={"f7 ba bg-gray0-d white-d pa3 db w-100 " +
"focus-b--black b--gray3 b--gray2-d"} "focus-b--black focus-b--white-d b--gray3 b--gray2-d"}
placeholder="Notebook description" placeholder="Notebook description"
rows={1} rows={1}
style={{ style={{

View File

@ -5,7 +5,7 @@ export class NotebookItem extends Component {
render() { render() {
let { props } = this; let { props } = this;
let selectedClass = (props.selected) ? "bg-gray5 b--gray4" : "b--gray4"; let selectedClass = (props.selected) ? "bg-gray5 bg-gray1-d b--gray4 b--gray2-d" : "b--gray4 b--gray2-d";
let postCount = (props.total === 1) let postCount = (props.total === 1)
? `${props.total} post` : `${props.total} posts`; ? `${props.total} post` : `${props.total} posts`;

View File

@ -81,12 +81,12 @@ export class Notebook extends Component {
let notebook = props.notebooks[props.ship][props.book]; let notebook = props.notebooks[props.ship][props.book];
let tabStyles = { let tabStyles = {
posts: "bb b--gray4 gray2 pv4 ph2", posts: "bb b--gray4 b--gray2-d gray2 pv4 ph2",
about: "bb b--gray4 gray2 pv4 ph2", about: "bb b--gray4 b--gray2-d gray2 pv4 ph2",
subscribers: "bb b--gray4 gray2 pv4 ph2", subscribers: "bb b--gray4 b--gray2-d gray2 pv4 ph2",
settings: "bb b--gray4 pr2 gray2 pv4 ph2", settings: "bb b--gray4 b--gray2-d pr2 gray2 pv4 ph2",
}; };
tabStyles[props.view] = "bb b--black black pv4 ph2"; tabStyles[props.view] = "bb b--black b--white-d black white-d pv4 ph2";
let inner = null; let inner = null;
switch (props.view) { switch (props.view) {
@ -154,7 +154,7 @@ export class Notebook extends Component {
let unsub = (window.ship === props.ship.slice(1)) let unsub = (window.ship === props.ship.slice(1))
? null ? null
: <button onClick={this.unsubscribe} : <button onClick={this.unsubscribe}
className="NotebookButton bg-white black ba b--black ml3"> className="NotebookButton bg-white bg-gray0-d black white-d ba b--black b--gray2-d ml3">
Unsubscribe Unsubscribe
</button> </button>
@ -224,7 +224,7 @@ export class Notebook extends Component {
</Link> </Link>
{subsComponent} {subsComponent}
{settingsComponent} {settingsComponent}
<div className="bb b--gray4 gray2 pv4 ph2" <div className="bb b--gray4 b--gray2-d gray2 pv4 ph2"
style={{ flexGrow: 1 }}></div> style={{ flexGrow: 1 }}></div>
</div> </div>

View File

@ -24,7 +24,7 @@ export class Settings extends Component {
<p className="f9 gray2 db mb4"> <p className="f9 gray2 db mb4">
Permanently delete this notebook. (All current members will no longer see this notebook) Permanently delete this notebook. (All current members will no longer see this notebook)
</p> </p>
<button className="b--red2 red2 pointer dib f9 ba pa2" <button className="bg-transparent b--red2 red2 pointer dib f9 ba pa2"
onClick={this.deleteNotebook}> onClick={this.deleteNotebook}>
Delete this notebook Delete this notebook
</button> </button>

View File

@ -155,28 +155,28 @@ export class Sidebar extends Component {
<Link to="/~publish/join" className="f9 gray2"> <Link to="/~publish/join" className="f9 gray2">
Join Notebook Join Notebook
</Link> </Link>
<div className="pl2 pv2 bb b--gray4"> <div className="pl2 pv2 bb b--gray4 b--gray2-d">
<Dropdown <Dropdown
width="16rem" width="16rem"
align="left" align="left"
options={[ options={[
{ {
cls: "w-100 tl pointer db ph2 pv3 hover-bg-gray4", cls: "white-d w-100 tl pointer db ph2 pv3 hover-bg-gray4 hover-bg-gray1-d bg-transparent",
txt: "Oldest", txt: "Oldest",
action: () => {this.setState({sort: "oldest"})} action: () => {this.setState({sort: "oldest"})}
}, },
{ {
cls: "w-100 tl pointer db ph2 pv3 hover-bg-gray4", cls: "white-d w-100 tl pointer db ph2 pv3 hover-bg-gray4 hover-bg-gray1-d bg-transparent",
txt: "Newest", txt: "Newest",
action: () => {this.setState({sort: "newest"})} action: () => {this.setState({sort: "newest"})}
}, },
{ {
cls: "w-100 tl pointer db ph2 pv3 hover-bg-gray4", cls: "white-d w-100 tl pointer db ph2 pv3 hover-bg-gray4 hover-bg-gray1-d bg-transparent",
txt: "A -> Z", txt: "A -> Z",
action: () => {this.setState({sort: "alphabetical"})} action: () => {this.setState({sort: "alphabetical"})}
}, },
{ {
cls: "w-100 tl pointer db ph2 pv3 hover-bg-gray4", cls: "white-d w-100 tl pointer db ph2 pv3 hover-bg-gray4 hover-bg-gray1-d bg-transparent",
txt: "Z -> A", txt: "Z -> A",
action: () => {this.setState({sort: "reverseAlphabetical"})} action: () => {this.setState({sort: "reverseAlphabetical"})}
} }

View File

@ -54,14 +54,14 @@ export class Subscribers extends Component {
width = 258; width = 258;
let url = `/~contacts${writePath}`; let url = `/~contacts${writePath}`;
options = [{ options = [{
cls: "tl pointer w-100 db hover-bg-gray4 ph2 pv3", cls: "bg-transparent white-d tl pointer w-100 db hover-bg-gray4 hover-bg-gray1-d ph2 pv3",
txt: "Manage this group in the contacts view", txt: "Manage this group in the contacts view",
action: () => {this.redirect(url)} action: () => {this.redirect(url)}
}]; }];
} else { } else {
width = 157; width = 157;
options = [{ options = [{
cls: "tl pointer w-100 db hover-bg-gray4 ph2 pv3", cls: "bg-transparent white-d tl pointer w-100 db hover-bg-gray4 hover-bg-gray1-d ph2 pv3",
txt: "Demote to subscriber", txt: "Demote to subscriber",
action: () => {this.removeUser(`~${who}`, writePath)} action: () => {this.removeUser(`~${who}`, writePath)}
}]; }];

View File

@ -31,7 +31,7 @@ export class Skeleton extends Component {
path={props.path} path={props.path}
invites={props.invites} invites={props.invites}
/> />
<div className={"h-100 w-100 relative " + rightPanelHide} style={{ <div className={"h-100 w-100 relative white-d " + rightPanelHide} style={{
flexGrow: 1, flexGrow: 1,
}}> }}>
{props.children} {props.children}