Merge branch 'release/next-js' into release/next-userspace

This commit is contained in:
Matilde Park 2021-01-20 12:47:31 -05:00
commit 31def6f57a
93 changed files with 1441 additions and 1342 deletions

View File

@ -1,4 +1,4 @@
FROM jaredtobin/janeway:v0.13.1 FROM jaredtobin/janeway:v0.13.3
COPY entrypoint.sh /entrypoint.sh COPY entrypoint.sh /entrypoint.sh
EXPOSE 22/tcp EXPOSE 22/tcp
ENTRYPOINT ["/entrypoint.sh"] ENTRYPOINT ["/entrypoint.sh"]

View File

@ -10,13 +10,7 @@ chmod 600 service-account
chmod 600 id_ssh chmod 600 id_ssh
chmod 600 id_ssh.pub chmod 600 id_ssh.pub
LANDSCAPE_STREAM="development" janeway release glob --dev --no-pill \
export LANDSCAPE_STREAM
LANDSCAPE_SHORTHASH="${GITHUB_SHA:0:7}"
export LANDSCAPE_SHORTHASH
janeway release glob --no-pill \
--credentials service-account \ --credentials service-account \
--ssh-key id_ssh \ --ssh-key id_ssh \
--do-it-live \ --do-it-live \

View File

@ -47,10 +47,22 @@ jobs:
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
# We only want the extra nix config on linux, where it is necessary
# for the docker build. We don't want in on Mac, where it isn't but
# it breaks the nix install. The two `if` clauses should be mutually
# exclusive
- uses: cachix/install-nix-action@v12 - uses: cachix/install-nix-action@v12
with:
extra_nix_config: |
system-features = nixos-test benchmark big-parallel kvm
if: ${{ matrix.os == 'ubuntu-latest' }}
- uses: cachix/install-nix-action@v12
if: ${{ matrix.os != 'ubuntu-latest' }}
- uses: cachix/cachix-action@v8 - uses: cachix/cachix-action@v8
with: with:
name: ares name: ${{ secrets.CACHIX_NAME }}
authToken: ${{ secrets.CACHIX_AUTH_TOKEN }} authToken: ${{ secrets.CACHIX_AUTH_TOKEN }}
- run: nix-build -A urbit --arg enableStatic true - run: nix-build -A urbit --arg enableStatic true
@ -58,6 +70,9 @@ jobs:
- if: ${{ matrix.os == 'ubuntu-latest' }} - if: ${{ matrix.os == 'ubuntu-latest' }}
run: nix-build -A urbit-tests run: nix-build -A urbit-tests
- if: ${{ matrix.os == 'ubuntu-latest' }}
run: nix-build -A docker-image
haskell: haskell:
strategy: strategy:
fail-fast: false fail-fast: false
@ -73,7 +88,7 @@ jobs:
- uses: cachix/install-nix-action@v12 - uses: cachix/install-nix-action@v12
- uses: cachix/cachix-action@v8 - uses: cachix/cachix-action@v8
with: with:
name: ares name: ${{ secrets.CACHIX_NAME }}
authToken: ${{ secrets.CACHIX_AUTH_TOKEN }} authToken: ${{ secrets.CACHIX_AUTH_TOKEN }}
- run: nix-build -A hs.urbit-king.components.exes.urbit-king --arg enableStatic true - run: nix-build -A hs.urbit-king.components.exes.urbit-king --arg enableStatic true

View File

@ -3,9 +3,6 @@ on:
push: push:
branches: branches:
- 'release/next-js' - 'release/next-js'
pull_request:
branches:
- 'release/next-js'
jobs: jobs:
glob: glob:
runs-on: ubuntu-latest runs-on: ubuntu-latest

17
.github/workflows/merge.yml vendored Normal file
View File

@ -0,0 +1,17 @@
name: merge
on:
push:
branches:
- 'master'
jobs:
merge-to-next-js:
runs-on: ubuntu-latest
name: "Merge master to release/next-js"
steps:
- uses: actions/checkout@v2
- uses: devmasx/merge-branch@v1.3.1
with:
type: now
target_branch: release/next-js
github_token: ${{ secrets.JANEWAY_BOT_TOKEN }}

51
.github/workflows/release-docker.yml vendored Normal file
View File

@ -0,0 +1,51 @@
name: release-docker
on:
release: null
push:
tags: ['urbit-v*']
jobs:
upload:
strategy:
matrix:
include:
- { os: ubuntu-latest, system: x86_64-linux }
runs-on: ${{ matrix.os }}
steps:
- uses: actions/checkout@v2
- uses: cachix/install-nix-action@v12
with:
extra_nix_config: |
system-features = nixos-test benchmark big-parallel kvm
- uses: cachix/cachix-action@v8
with:
name: ${{ secrets.CACHIX_NAME }}
authToken: ${{ secrets.CACHIX_AUTH_TOKEN }}
- uses: docker/docker-login-action@v1.8.0
with:
username: ${{ secrets.DOCKERHUB_USERNAME }}
password: ${{ secrets.DOCKERHUB_TOKEN }}
- uses: christian-korneck/update-container-description-action@v1
env:
DOCKER_USER: ${{ secrets.DOCKERHUB_USERNAME }}
DOCKER_PASS: ${{ secrets.DOCKERHUB_TOKEN }}
with:
destination_container_repo: ${{ secrets.DOCKERHUB_USERNAME }}/urbit
provider: dockerhub
short_description: 'Urbit: a clean-slate OS and network for the 21st century'
readme_file: 'pkg/docker-image/README.md'
- run: |
version="$(cat ./pkg/urbit/version)"
image="$(nix-build -A docker-image)"
imageName="$(nix-instantiate --eval -A docker-image.imageName | cut -d'"' -f2)"
imageTag="$(nix-instantiate --eval -A docker-image.imageTag | cut -d'"' -f2)"
# Load the image from the nix-built tarball
docker load -i $image
docker tag "$imageName:$imageTag" ${{secrets.DOCKERHUB_USERNAME }}/urbit:v$version
docker tag "$imageName:$imageTag" ${{secrets.DOCKERHUB_USERNAME }}/urbit:latest
docker push ${{secrets.DOCKERHUB_USERNAME }}/urbit:v$version
docker push ${{secrets.DOCKERHUB_USERNAME }}/urbit:latest

View File

@ -20,7 +20,7 @@ jobs:
- uses: cachix/install-nix-action@v12 - uses: cachix/install-nix-action@v12
- uses: cachix/cachix-action@v8 - uses: cachix/cachix-action@v8
with: with:
name: ares name: ${{ secrets.CACHIX_NAME }}
authToken: ${{ secrets.CACHIX_AUTH_TOKEN }} authToken: ${{ secrets.CACHIX_AUTH_TOKEN }}
- uses: google-github-actions/setup-gcloud@v0.2.0 - uses: google-github-actions/setup-gcloud@v0.2.0

View File

@ -180,9 +180,9 @@ new fakezod with `urbit -F zod -B bin/solid.pill -A pkg/arvo`). Run
`:glob|make`, and this will output a file in `fakezod/.urb/put/glob-0vXXX.glob`. `:glob|make`, and this will output a file in `fakezod/.urb/put/glob-0vXXX.glob`.
Upload this file to bootstrap.urbit.org, and modify `+hash` at the top of Upload this file to bootstrap.urbit.org, and modify `+hash` at the top of
`pkg/arvo/app/glob.hoon` to match the hash in the filename of the `.glob` file. `pkg/arvo/app/glob.hoon` to match the hash in the filename of the `.glob` file.
Amend `pkg/arvo/app/landscape/index.html` to import the hashed JS bundle, instead Amend `pkg/arvo/app/landscape/index.html` to import the hashed JS bundle, instead
of the unversioned index.js. Do not commit the produced `index.js` and of the unversioned index.js. Do not commit the produced `index.js` and
make sure it doesn't end up in your pills (they should be less than 10MB each). make sure it doesn't end up in your pills (they should be less than 10MB each).
### Tag the resulting commit ### Tag the resulting commit
@ -306,6 +306,13 @@ $ herb zod -p hood -d "+hood/merge %kids our %home"
For Vere updates, this means simply shutting down each desired ship, installing For Vere updates, this means simply shutting down each desired ship, installing
the new binary, and restarting the pier with it. the new binary, and restarting the pier with it.
#### Continuous deployment
A subset of release branches are deployed continuously to the network. Thus far
this only includes `release/next-js`, which deploys livenet-compatible
JavaScript changes to select QA ships. Any push to master will automatically
merge master into `release/next-js` to keep the streams at parity.
### Announce the update ### Announce the update
Post an announcement to urbit-dev. The tag annotation, basically, is fine here Post an announcement to urbit-dev. The tag annotation, basically, is fine here

View File

@ -1,3 +1,3 @@
version https://git-lfs.github.com/spec/v1 version https://git-lfs.github.com/spec/v1
oid sha256:17eb2f5a123f5ad29b0cc9ff9069540c349dd97c6133a9ea33cbf81e0bfa4d6b oid sha256:288b3ab68e2e3946dbf0e0de05c2907c351ec97fcc08134e558569eab4121c94
size 8483784 size 8809816

View File

@ -115,6 +115,8 @@ let
urbit = callPackage ./nix/pkgs/urbit { inherit enableStatic; }; urbit = callPackage ./nix/pkgs/urbit { inherit enableStatic; };
docker-image = callPackage ./nix/pkgs/docker-image { };
hs = callPackage ./nix/pkgs/hs { hs = callPackage ./nix/pkgs/hs {
inherit enableStatic; inherit enableStatic;
inherit (pkgsCross) haskell-nix; inherit (pkgsCross) haskell-nix;
@ -158,6 +160,8 @@ let
}; };
}; };
inherit (pkgsNative) skopeo;
# A convenience function for constructing a shell.nix for any of the # A convenience function for constructing a shell.nix for any of the
# pkgsLocal derivations by automatically propagating any dependencies # pkgsLocal derivations by automatically propagating any dependencies
# to the nix-shell. # to the nix-shell.

View File

@ -0,0 +1,68 @@
{ urbit, libcap, coreutils, bashInteractive, dockerTools, writeScriptBin, amesPort ? 34343 }:
let
startUrbit = writeScriptBin "start-urbit" ''
#!${bashInteractive}/bin/bash
set -eu
# If the container is not started with the `-i` flag
# then STDIN will be closed and we need to start
# Urbit/vere with the `-t` flag.
ttyflag=""
if [ ! -t 0 ]; then
echo "Running with no STDIN"
ttyflag="-t"
fi
# Check if there is a keyfile, if so boot a ship with its name, and then remove the key
if [ -e *.key ]; then
# Get the name of the key
keynames="*.key"
keys=( $keynames )
keyname=''${keys[0]}
mv $keyname /tmp
# Boot urbit with the key, exit when done booting
urbit $ttyflag -w $(basename $keyname .key) -k /tmp/$keyname -c $(basename $keyname .key) -p ${toString amesPort} -x
# Remove the keyfile for security
rm /tmp/$keyname
rm *.key || true
elif [ -e *.comet ]; then
cometnames="*.comet"
comets=( $cometnames )
cometname=''${comets[0]}
rm *.comet
urbit $ttyflag -c $(basename $cometname .comet) -p ${toString amesPort} -x
fi
# Find the first directory and start urbit with the ship therein
dirnames="*/"
dirs=( $dirnames )
dirname=''${dirnames[0]}
urbit $ttyflag -p ${toString amesPort} $dirname
'';
in dockerTools.buildImage {
name = "urbit";
tag = "v${urbit.version}";
contents = [ bashInteractive urbit startUrbit coreutils ];
runAsRoot = ''
#!${bashInteractive}
mkdir -p /urbit
mkdir -p /tmp
${libcap}/bin/setcap 'cap_net_bind_service=+ep' /bin/urbit
'';
config = {
Cmd = [ "/bin/start-urbit" ];
Env = [ "PATH=/bin" ];
WorkingDir = "/urbit";
Volumes = {
"/urbit" = {};
};
Expose = [ "80/tcp" "${toString amesPort}/udp" ];
};
}

View File

@ -5,7 +5,7 @@
/- glob /- glob
/+ default-agent, verb, dbug /+ default-agent, verb, dbug
|% |%
++ hash 0v5.hvt1e.ie7it.b7i7l.1r7jj.dn9ib ++ hash 0v1.4u9gp.rs1fi.ki7ok.ib4cp.mgdvs
+$ state-0 [%0 hash=@uv glob=(unit (each glob:glob tid=@ta))] +$ state-0 [%0 hash=@uv glob=(unit (each glob:glob tid=@ta))]
+$ all-states +$ all-states
$% state-0 $% state-0

View File

@ -92,27 +92,6 @@
%run-updates (is-allowed resource.q.update bowl %.y) %run-updates (is-allowed resource.q.update bowl %.y)
== ==
:: ::
++ resource-for-update
|= =vase
^- (unit resource:res)
=/ =update:store !<(update:store vase)
?- -.q.update
%add-graph `resource.q.update
%remove-graph `resource.q.update
%add-nodes `resource.q.update
%remove-nodes `resource.q.update
%add-signatures `resource.uid.q.update
%remove-signatures `resource.uid.q.update
%archive-graph `resource.q.update
%unarchive-graph ~
%add-tag ~
%remove-tag ~
%keys ~
%tags ~
%tag-queries ~
%run-updates `resource.q.update
==
::
++ initial-watch ++ initial-watch
|= [=path =resource:res] |= [=path =resource:res]
^- vase ^- vase

View File

@ -230,7 +230,7 @@
?> ?=(%0 -.update) ?> ?=(%0 -.update)
=? p.update =(p.update *time) now.bowl =? p.update =(p.update *time) now.bowl
?- -.q.update ?- -.q.update
%add-graph (add-graph +.q.update) %add-graph (add-graph p.update +.q.update)
%remove-graph (remove-graph +.q.update) %remove-graph (remove-graph +.q.update)
%add-nodes (add-nodes p.update +.q.update) %add-nodes (add-nodes p.update +.q.update)
%remove-nodes (remove-nodes p.update +.q.update) %remove-nodes (remove-nodes p.update +.q.update)
@ -247,7 +247,8 @@
== ==
:: ::
++ add-graph ++ add-graph
|= $: =resource:store |= $: =time
=resource:store
=graph:store =graph:store
mark=(unit mark:store) mark=(unit mark:store)
overwrite=? overwrite=?
@ -258,9 +259,13 @@
!(~(has by graphs) resource) !(~(has by graphs) resource)
== == == ==
?> (validate-graph graph mark) ?> (validate-graph graph mark)
=/ =logged-update:store
[%0 time %add-graph resource graph mark overwrite]
=/ =update-log:store
(gas:orm-log ~ [time logged-update] ~)
:_ %_ state :_ %_ state
graphs (~(put by graphs) resource [graph mark]) graphs (~(put by graphs) resource [graph mark])
update-logs (~(put by update-logs) resource (gas:orm-log ~ ~)) update-logs (~(put by update-logs) resource update-log)
archive (~(del by archive) resource) archive (~(del by archive) resource)
:: ::
validators validators
@ -418,43 +423,81 @@
=/ =update-log:store (~(got by update-logs) resource) =/ =update-log:store (~(got by update-logs) resource)
=. update-log =. update-log
(put:orm-log update-log time [%0 time [%remove-nodes resource indices]]) (put:orm-log update-log time [%0 time [%remove-nodes resource indices]])
=/ [affected-indices=(set index:store) new-graph=graph:store]
(remove-indices resource graph (sort ~(tap in indices) by-lent))
:: ::
:- (give [/updates]~ [%remove-nodes resource indices]) :- (give [/updates]~ [%remove-nodes resource (~(uni in indices) affected-indices)])
%_ state %_ state
update-logs (~(put by update-logs) resource update-log) update-logs (~(put by update-logs) resource update-log)
graphs graphs
%+ ~(put by graphs) %+ ~(put by graphs)
resource resource
[(remove-indices resource graph ~(tap in indices)) mark] [new-graph mark]
== ==
:: ::
:: we always want to remove the deepest node first,
:: so we don't remove parents before children
++ by-lent
|* [a=(list) b=(list)]
^- ?
(gth (lent a) (lent b))
::
++ remove-indices ++ remove-indices
=| affected=(set index:store)
|= [=resource:store =graph:store indices=(list index:store)] |= [=resource:store =graph:store indices=(list index:store)]
^- graph:store ^- [(set index:store) graph:store]
?~ indices graph ?~ indices [affected graph]
=^ new-affected graph
(remove-index graph i.indices)
%_ $ %_ $
indices t.indices indices t.indices
graph (remove-index graph i.indices) affected (~(uni in affected) new-affected)
==
::
++ get-descendants
|= =graph:store
=| indices=(list index:store)
=/ nodes (tap:orm:store graph)
%- ~(gas in *(set index:store))
|- =* tap-nodes $
^+ indices
%- zing
%+ turn nodes
|= [atom =node:store]
^- (list index:store)
%+ welp
index.post.node^~
?. ?=(%graph -.children.node)
~
%_ tap-nodes
nodes (tap:orm p.children.node)
== ==
:: ::
++ remove-index ++ remove-index
=| indices=(set index:store)
|= [=graph:store =index:store] |= [=graph:store =index:store]
^- graph:store ^- [(set index:store) graph:store]
?~ index graph ?~ index [indices graph]
=* atom i.index =* atom i.index
:: last index in list :: last index in list
:: ::
?~ t.index ?~ t.index
+:`[* graph:store]`(del:orm graph atom) =^ rm-node graph (del:orm graph atom)
?~ rm-node `graph
?. ?=(%graph -.children.u.rm-node)
`graph
=/ new-indices
(get-descendants p.children.u.rm-node)
[(~(uni in indices) new-indices) graph]
=/ =node:store =/ =node:store
~| "parent index does not exist to remove a node from!" ~| "parent index does not exist to remove a node from!"
(need (get:orm graph atom)) (need (get:orm graph atom))
~| "child index does not exist to remove a node from!" ~| "child index does not exist to remove a node from!"
?> ?=(%graph -.children.node) ?> ?=(%graph -.children.node)
%^ put:orm =^ new-indices p.children.node
graph $(graph p.children.node, index t.index)
atom :- (~(uni in indices) new-indices)
node(p.children $(graph p.children.node, index t.index)) (put:orm graph atom node)
-- --
:: ::
++ add-signatures ++ add-signatures
@ -605,6 +648,7 @@
%- graph-update %- graph-update
^- update:store ^- update:store
?- -.q.update ?- -.q.update
%add-graph update(resource.q resource)
%add-nodes update(resource.q resource) %add-nodes update(resource.q resource)
%remove-nodes update(resource.q resource) %remove-nodes update(resource.q resource)
%add-signatures update(resource.uid.q resource) %add-signatures update(resource.uid.q resource)
@ -868,6 +912,15 @@
|= [=atom =node:store] |= [=atom =node:store]
^- [index:store node:store] ^- [index:store node:store]
[~[atom] node] [~[atom] node]
::
[%x %node-exists @ @ @ *]
=/ =ship (slav %p i.t.t.path)
=/ =term i.t.t.t.path
=/ =index:store
(turn t.t.t.t.path (cury slav %ud))
=/ node=(unit node:store)
(get-node ship term index)
``noun+!>(?=(^ node))
:: ::
[%x %node @ @ @ *] [%x %node @ @ @ *]
=/ =ship (slav %p i.t.t.path) =/ =ship (slav %p i.t.t.path)

View File

@ -142,15 +142,6 @@
== ==
-- --
:: ::
++ resource-for-update
|= =vase
^- (unit resource)
=/ =update:store
!<(update:store vase)
?: ?=(%initial -.update)
~
`resource.update
::
++ take-update ++ take-update
|= =vase |= =vase
^- [(list card) agent] ^- [(list card) agent]

View File

@ -9,11 +9,17 @@
+$ card card:agent:gall +$ card card:agent:gall
+$ versioned-state +$ versioned-state
$% state-0 $% state-0
state-1
== ==
:: ::
+$ state-0 +$ state-0
$: %0 [%0 base-state-0]
watching=(set [resource index:post]) ::
+$ state-1
[%1 base-state-0]
::
+$ base-state-0
$: watching=(set [resource index:post])
mentions=_& mentions=_&
watch-on-self=_& watch-on-self=_&
== ==
@ -36,7 +42,7 @@
:: ::
-- --
:: ::
=| state-0 =| state-1
=* state - =* state -
:: ::
=< =<
@ -57,13 +63,25 @@
:: ::
++ on-save !>(state) ++ on-save !>(state)
++ on-load ++ on-load
|= old=vase |= =vase
^- (quip card _this) ^- (quip card _this)
:_ this(state !<(state-0 old)) =+ !<(old=versioned-state vase)
=| cards=(list card)
|-
?: ?=(%0 -.old)
%_ $
-.old %1
::
cards
:_ cards
[%pass / %agent [our dap]:bowl %poke noun+!>(%rewatch-dms)]
==
:_ this(state old)
=. cards (flop cards)
%+ welp %+ welp
?: (~(has by wex.bowl) [/graph our.bowl %graph-store]) ?: (~(has by wex.bowl) [/graph our.bowl %graph-store])
~ cards
~[watch-graph:ha] [watch-graph:ha cards]
%+ turn %+ turn
^- (list mark) ^- (list mark)
:~ %graph-validator-chat :~ %graph-validator-chat
@ -103,9 +121,23 @@
?+ mark (on-poke:def mark vase) ?+ mark (on-poke:def mark vase)
%hark-graph-hook-action %hark-graph-hook-action
(hark-graph-hook-action !<(action:hook vase)) (hark-graph-hook-action !<(action:hook vase))
%noun
(poke-noun !<(* vase))
== ==
[cards this] [cards this]
:: ::
++ poke-noun
|= non=*
?> ?=(%rewatch-dms non)
=/ graphs=(list resource)
~(tap in get-keys:gra)
:- ~
%_ state
watching
%- ~(gas in watching)
(murn graphs |=(rid=resource ?:((should-watch:ha rid) `[rid ~] ~)))
==
::
++ hark-graph-hook-action ++ hark-graph-hook-action
|= =action:hook |= =action:hook
^- (quip card _state) ^- (quip card _state)
@ -167,16 +199,48 @@
:: ::
?(%remove-graph %archive-graph) ?(%remove-graph %archive-graph)
(remove-graph resource.q.update) (remove-graph resource.q.update)
::
%remove-nodes
(remove-nodes resource.q.update indices.q.update)
:: ::
%add-nodes %add-nodes
=* rid resource.q.update =* rid resource.q.update
(check-nodes ~(val by nodes.q.update) rid) (check-nodes ~(val by nodes.q.update) rid)
== ==
:: this is awful, but notification kind should always switch
:: on the index, so hopefully doesn't matter
:: TODO: rethink this
++ remove-nodes
|= [rid=resource indices=(set index:graph-store)]
=/ to-remove
%- ~(gas by *(set [resource index:graph-store]))
(turn ~(tap in indices) (lead rid))
:_ state(watching (~(dif in watching) to-remove))
=/ =tube:clay
(get-conversion:ha rid)
%+ roll
~(tap in indices)
|= [=index:graph-store out=(list card)]
=| =indexed-post:graph-store
=. index.p.indexed-post index
=+ !<(u-notif-kind=(unit notif-kind) (tube !>(indexed-post)))
?~ u-notif-kind out
=* notif-kind u.u-notif-kind
=/ =stats-index:store
[%graph rid (scag parent-lent.notif-kind index)]
?. ?=(%each mode.notif-kind) out
:_ out
(poke-hark %read-each stats-index index)
::
++ poke-hark
|= =action:store
^- card
[%pass / %agent [our.bowl %hark-store] %poke hark-action+!>(action)]
:: ::
++ remove-graph ++ remove-graph
|= rid=resource |= rid=resource
=/ unwatched =/ unwatched
%- ~(gas in *_watching) %- ~(gas in *(set [resource index:graph-store]))
%+ skim ~(tap in watching) %+ skim ~(tap in watching)
|= [r=resource idx=index:graph-store] |= [r=resource idx=index:graph-store]
=(r rid) =(r rid)
@ -191,23 +255,14 @@
++ add-graph ++ add-graph
|= rid=resource |= rid=resource
^- (quip card _state) ^- (quip card _state)
=/ group-rid=(unit resource)
(group-from-app-resource:met %graph rid)
?~ group-rid
~& no-group+rid
`state
=/ is-hidden=?
!(is-managed:grp u.group-rid)
=/ should-watch
|(is-hidden &(watch-on-self =(our.bowl entity.rid)))
?. should-watch
`state
=/ graph=graph:graph-store :: graph in subscription is bunted =/ graph=graph:graph-store :: graph in subscription is bunted
(get-graph-mop:gra rid) (get-graph-mop:gra rid)
=/ node=(unit node:graph-store) =/ node=(unit node:graph-store)
(bind (peek:orm:graph-store graph) |=([@ =node:graph-store] node)) (bind (peek:orm:graph-store graph) |=([@ =node:graph-store] node))
=^ cards state =^ cards state
(check-nodes (drop node) rid) (check-nodes (drop node) rid)
?. (should-watch:ha rid)
[cards state]
:_ state(watching (~(put in watching) [rid ~])) :_ state(watching (~(put in watching) [rid ~]))
(weld cards (give:ha ~[/updates] %listen [rid ~])) (weld cards (give:ha ~[/updates] %listen [rid ~]))
:: ::
@ -245,7 +300,19 @@
-- --
:: ::
|_ =bowl:gall |_ =bowl:gall
+* met ~(. metadata bowl)
grp ~(. grouplib bowl)
gra ~(. graph bowl)
:: ::
++ get-conversion
|= rid=resource
^- tube:clay
=+ %^ scry [our now]:bowl
,mark=(unit mark)
/gx/graph-store/graph-mark/(scot %p entity.rid)/[name.rid]/noun
?~ mark
|=(v=vase !>(~))
(scry-conversion [our now]:bowl q.byk.bowl u.mark)
:: ::
++ give ++ give
|= [paths=(list path) =update:hook] |= [paths=(list path) =update:hook]
@ -273,6 +340,16 @@
%.y %.y
$(contents t.contents) $(contents t.contents)
:: ::
++ should-watch
|= rid=resource
^- ?
=/ group-rid=(unit resource)
(group-from-app-resource:met %graph rid)
?~ group-rid %.n
?| !(is-managed:grp u.group-rid)
&(watch-on-self =(our.bowl entity.rid))
==
::
++ handle-update ++ handle-update
|_ $: rid=resource :: input |_ $: rid=resource :: input
updates=(list node:graph-store) updates=(list node:graph-store)
@ -288,13 +365,7 @@
update-core(rid r, updates upds, group grp, module mod) update-core(rid r, updates upds, group grp, module mod)
:: ::
++ get-conversion ++ get-conversion
^- tube:clay (^get-conversion rid)
=+ %^ scry [our now]:bowl
,mark=(unit mark)
/gx/graph-store/graph-mark/(scot %p entity.rid)/[name.rid]/noun
?~ mark
|=(v=vase !>(~))
(scry-conversion [our now]:bowl q.byk.bowl u.mark)
:: ::
++ abet ++ abet
^- (quip card _state) ^- (quip card _state)

View File

@ -148,7 +148,9 @@
|= [=index:store =notification:store] |= [=index:store =notification:store]
^- card ^- card
=- [%pass / %agent [our.bowl %hark-store] %poke -] =- [%pass / %agent [our.bowl %hark-store] %poke -]
hark-action+!>([%add index notification]) :- %hark-action
!> ^- action:store
[%add-note index notification]
-- --
:: ::
++ on-peek on-peek:def ++ on-peek on-peek:def

View File

@ -21,13 +21,13 @@
$% state:state-zero:store $% state:state-zero:store
state:state-one:store state:state-one:store
state-2 state-2
state-3
== ==
+$ unread-stats +$ unread-stats
[indices=(set index:graph-store) last=@da] [indices=(set index:graph-store) last=@da]
:: ::
+$ state-2 +$ base-state
$: %2 $: unreads-each=(jug stats-index:store index:graph-store)
unreads-each=(jug stats-index:store index:graph-store)
unreads-count=(map stats-index:store @ud) unreads-count=(map stats-index:store @ud)
last-seen=(map stats-index:store @da) last-seen=(map stats-index:store @da)
=notifications:store =notifications:store
@ -36,14 +36,20 @@
dnd=_| dnd=_|
== ==
:: ::
+$ state-2
[%2 base-state]
::
+$ state-3
[%3 base-state]
::
+$ inflated-state +$ inflated-state
$: state-2 $: state-3
cache cache
== ==
:: $cache: useful to have precalculated, but can be derived from state :: $cache: useful to have precalculated, but can be derived from state
:: albeit expensively :: albeit expensively
+$ cache +$ cache
$: by-index=(jug stats-index:store @da) $: by-index=(jug stats-index:store [time=@da =index:store])
~ ~
== ==
:: ::
@ -78,9 +84,19 @@
=| cards=(list card) =| cards=(list card)
|^ |^
?- -.old ?- -.old
%2 %3
:- cards :- (flop cards)
this(-.state old, +.state (inflate-cache:ha old)) this(-.state old, +.state (inflate-cache:ha old))
::
%2
%_ $
-.old %3
::
cards
:_ cards
[%pass / %agent [our dap]:bowl %poke noun+!>(%fix-dangling)]
==
:: ::
%1 %1
%_ $ %_ $
@ -212,6 +228,7 @@
[%count count] [%count count]
(~(gut by last-seen) stats-index *time) (~(gut by last-seen) stats-index *time)
== ==
::
++ give-each-unreads ++ give-each-unreads
^- (list [stats-index:store stats:store]) ^- (list [stats-index:store stats:store])
%+ turn %+ turn
@ -264,10 +281,41 @@
=^ cards state =^ cards state
?+ mark (on-poke:def mark vase) ?+ mark (on-poke:def mark vase)
%hark-action (hark-action !<(action:store vase)) %hark-action (hark-action !<(action:store vase))
%noun ~& +.state [~ state] %noun (poke-noun !<(* vase))
== ==
[cards this] [cards this]
:: ::
++ poke-noun
|= val=*
?+ val ~|(%bad-noun-poke !!)
%fix-dangling fix-dangling
%print ~&(+.state [~ state])
==
::
++ fix-dangling
=/ graphs get-keys:gra
:_ state
%+ roll
~(tap by unreads-each)
|= $: [=stats-index:store indices=(set index:graph-store)]
out=(list card)
==
?. ?=(%graph -.stats-index) out
?. (~(has in graphs) graph.stats-index)
:_(out (poke-us %remove-graph graph.stats-index))
%+ welp out
%+ turn
%+ skip
~(tap in indices)
|= =index:graph-store
(check-node-existence:gra graph.stats-index index)
|=(=index:graph-store (poke-us %read-each stats-index index))
::
++ poke-us
|= =action:store
^- card
[%pass / %agent [our dap]:bowl %poke hark-action+!>(action)]
::
++ hark-action ++ hark-action
|= =action:store |= =action:store
^- (quip card _state) ^- (quip card _state)
@ -338,6 +386,9 @@
|= [read=? time=@da =index:store] |= [read=? time=@da =index:store]
poke-core(+.state (^upd-cache read time index)) poke-core(+.state (^upd-cache read time index))
:: ::
++ rebuild-cache
poke-core(+.state (inflate-cache -.state))
::
++ put-notifs ++ put-notifs
|= [time=@da =timebox:store] |= [time=@da =timebox:store]
poke-core(notifications (put:orm notifications time timebox)) poke-core(notifications (put:orm notifications time timebox))
@ -380,17 +431,28 @@
(~(put by archive-box) index notification(read %.y)) (~(put by archive-box) index notification(read %.y))
(give %archive time index) (give %archive time index)
:: ::
:: if we detect cache inconsistencies, wipe and rebuild
++ change-read-status ++ change-read-status
|= [time=@da =index:store read=?] |= [time=@da =index:store read=?]
^+ poke-core
=. poke-core (upd-cache read time index) =. poke-core (upd-cache read time index)
%_ poke-core =/ tib=(unit timebox:store)
notifications (get:orm notifications time)
%^ jub-orm notifications time ?~ tib poke-core
|= =timebox:store =/ not=(unit notification:store)
%+ ~(jab by timebox) index (~(get by u.tib) index)
|= n=notification:store ?~ not poke-core
?>(!=(read read.n) n(read read)) =? poke-core
== :: cache is inconsistent iff we didn't directly
:: call this through %read-note or %unread-note
&(=(read read.u.not) !?=(?(%read-note %unread-note) -.in))
~& >> "Inconsistent hark cache, rebuilding"
rebuild-cache
=. u.tib
(~(put by u.tib) index u.not(read read))
=. notifications
(put:orm notifications time u.tib)
poke-core
:: ::
++ read-note ++ read-note
|= [time=@da =index:store] |= [time=@da =index:store]
@ -416,19 +478,16 @@
:: ::
++ read-index-each ++ read-index-each
|= [=stats-index:store ref=index:graph-store] |= [=stats-index:store ref=index:graph-store]
%+ read-index stats-index %- read-indices
%+ skim %+ skim
~(tap ^in (~(get ju by-index) stats-index)) ~(tap ^in (~(get ju by-index) stats-index))
|= time=@da |= [time=@da =index:store]
=/ =timebox:store =/ =timebox:store
(gut-orm notifications time) (gut-orm notifications time)
%+ roll =/ not=notification:store
~(tap ^in timebox) (~(got by timebox) index)
|= [[=index:store not=notification:store] out=?] ?. ?=(%graph -.index) %.n
?: out out ?. ?=(%graph -.contents.not) %.n
?. (stats-index-is-index:store stats-index index) out
?. ?=(%graph -.index) out
?. ?=(%graph -.contents.not) out
(lien list.contents.not |=(p=post:post =(index.p ref))) (lien list.contents.not |=(p=post:post =(index.p ref)))
:: ::
++ read-each ++ read-each
@ -456,31 +515,18 @@
++ read-count ++ read-count
|= =stats-index:store |= =stats-index:store
=. unreads-count (~(put by unreads-count) stats-index 0) =. unreads-count (~(put by unreads-count) stats-index 0)
=/ times=(list @da) =/ times=(list [@da index:store])
~(tap ^in (~(get ju by-index) stats-index)) ~(tap ^in (~(get ju by-index) stats-index))
(give:(read-index stats-index times) %read-count stats-index) (give:(read-indices times) %read-count stats-index)
:: ::
++ read-index ++ read-indices
|= [=stats-index:store times=(list @da)] |= times=(list [time=@da =index:store])
|- |-
?~ times poke-core ?~ times poke-core
=/ core =/ core
(read-stats-index i.times stats-index) (read-note i.times)
$(poke-core core, times t.times) $(poke-core core, times t.times)
:: ::
++ read-stats-index
|= [time=@da =stats-index:store]
=/ keys
~(tap ^in ~(key by (gut-orm notifications time)))
|- ^+ poke-core
?~ keys
poke-core
?. (stats-index-is-index:store stats-index i.keys)
$(keys t.keys)
=/ core
(read-note time i.keys)
$(poke-core core, keys t.keys)
::
++ seen-index ++ seen-index
|= [time=@da =stats-index:store] |= [time=@da =stats-index:store]
=/ new-time=@da =/ new-time=@da
@ -505,7 +551,7 @@
=. last-seen =. last-seen
((dif-map-by-key ,@da) last-seen indices) ((dif-map-by-key ,@da) last-seen indices)
=. by-index =. by-index
((dif-map-by-key ,(set @da)) by-index indices) ((dif-map-by-key ,(set [@da =index:store])) by-index indices)
poke-core poke-core
:: ::
++ get-stats-indices ++ get-stats-indices
@ -538,10 +584,10 @@
~(tap ^in set) ~(tap ^in set)
|- |-
?~ indices poke-core ?~ indices poke-core
=/ times=(list @da) =/ times=(list [time=@da =index:store])
~(tap ^in (~(get ju by-index) i.indices)) ~(tap ^in (~(get ju by-index) i.indices))
=. poke-core =. poke-core
(read-index i.indices times) (read-indices times)
$(indices t.indices) $(indices t.indices)
-- --
:: ::
@ -629,14 +675,14 @@
%_ +.state %_ +.state
:: ::
by-index by-index
%. [(to-stats-index:store index) time] %. [(to-stats-index:store index) time index]
?: read ?: read
~(del ju by-index) ~(del ju by-index)
~(put ju by-index) ~(put ju by-index)
== ==
:: ::
++ inflate-cache ++ inflate-cache
|= state-2 |= state-3
^+ +.state ^+ +.state
=/ nots=(list [p=@da =timebox:store]) =/ nots=(list [p=@da =timebox:store])
(tap:orm notifications) (tap:orm notifications)

View File

@ -1,7 +1,7 @@
<!doctype html> <!doctype html>
<html> <html>
<head> <head>
<title>OS1</title> <title>Landscape</title>
<meta charset="utf-8" /> <meta charset="utf-8" />
<meta name="viewport" <meta name="viewport"
content="width=device-width, initial-scale=1, shrink-to-fit=no,maximum-scale=1"/> content="width=device-width, initial-scale=1, shrink-to-fit=no,maximum-scale=1"/>
@ -12,8 +12,8 @@
<link rel="icon" type="image/png" href="/~landscape/img/Favicon.png"> <link rel="icon" type="image/png" href="/~landscape/img/Favicon.png">
<link rel="manifest" <link rel="manifest"
href='data:application/manifest+json,{ href='data:application/manifest+json,{
"name": "OS1", "name": "Landscape",
"short_name": "OS1", "short_name": "Landscape",
"description": "An%20interface%20to%20your%20Urbit.", "description": "An%20interface%20to%20your%20Urbit.",
"display": "standalone", "display": "standalone",
"background_color": "%23FFFFFF", "background_color": "%23FFFFFF",
@ -24,6 +24,6 @@
<div id="portal-root"></div> <div id="portal-root"></div>
<script src="/~landscape/js/channel.js"></script> <script src="/~landscape/js/channel.js"></script>
<script src="/~landscape/js/session.js"></script> <script src="/~landscape/js/session.js"></script>
<script src="/~landscape/js/bundle/index.2ddb586104e8758c6863.js"></script> <script src="/~landscape/js/bundle/index.86c6e416c338a305e1e9.js"></script>
</body> </body>
</html> </html>

View File

@ -63,7 +63,7 @@ class Channel {
} }
resetDebounceTimer() { resetDebounceTimer() {
if(this.debounceTimer) { if (this.debounceTimer) {
clearTimeout(this.debounceTimer); clearTimeout(this.debounceTimer);
this.debounceTimer = null; this.debounceTimer = null;
} }
@ -182,19 +182,19 @@ class Channel {
// sends a JSON command command to the server. // sends a JSON command command to the server.
// //
sendJSONToChannel(j) { sendJSONToChannel(j) {
if(!j && this.outstandingJSON.length === 0) {
return;
}
let req = new XMLHttpRequest(); let req = new XMLHttpRequest();
req.open("PUT", this.channelURL()); req.open("PUT", this.channelURL());
req.setRequestHeader("Content-Type", "application/json"); req.setRequestHeader("Content-Type", "application/json");
if (this.lastEventId == this.lastAcknowledgedEventId) { if (this.lastEventId == this.lastAcknowledgedEventId) {
if(j) { if (j) {
this.outstandingJSON.push(j); this.outstandingJSON.push(j);
} }
let x = JSON.stringify(this.outstandingJSON);
req.send(x); if (this.outstandingJSON.length > 0) {
let x = JSON.stringify(this.outstandingJSON);
req.send(x);
}
} else { } else {
// we add an acknowledgment to clear the server side queue // we add an acknowledgment to clear the server side queue
// //
@ -203,15 +203,15 @@ class Channel {
// //
let payload = [ let payload = [
...this.outstandingJSON, ...this.outstandingJSON,
{action: "ack", "event-id": parseInt(this.lastEventId)} {action: "ack", "event-id": this.lastEventId}
]; ];
if(j) { if (j) {
payload.push(j) payload.push(j)
} }
let x = JSON.stringify(payload); let x = JSON.stringify(payload);
req.send(x); req.send(x);
this.lastEventId = this.lastAcknowledgedEventId; this.lastAcknowledgedEventId = this.lastEventId;
} }
this.outstandingJSON = []; this.outstandingJSON = [];
@ -227,7 +227,7 @@ class Channel {
this.eventSource = new EventSource(this.channelURL(), {withCredentials:true}); this.eventSource = new EventSource(this.channelURL(), {withCredentials:true});
this.eventSource.onmessage = e => { this.eventSource.onmessage = e => {
this.lastEventId = e.lastEventId; this.lastEventId = parseInt(e.lastEventId, 10);
let obj = JSON.parse(e.data); let obj = JSON.parse(e.data);
let pokeFuncs = this.outstandingPokes.get(obj.id); let pokeFuncs = this.outstandingPokes.get(obj.id);

126
pkg/arvo/gen/tally.hoon Normal file
View File

@ -0,0 +1,126 @@
/- gr=group, md=metadata-store, ga=graph-store
/+ re=resource
!:
:- %say
|= $: [now=@da eny=@uvJ =beak]
args=?(~ [shy=? ~])
~
==
::
=/ shy=? ?~(args & shy.args)
=* our=@p p.beak
::
|^
=; out=(list @t)
:- %tang
%- flop ::NOTE tang is bottom-up
:* ''
'tallied your activity score! find the results below.'
::
?: shy
'to show non-anonymized resource identifiers, +tally |'
'showing plain resource identifiers, share with care.'
::
'counted from groups and channels that you are hosting.'
'groups are listed with their member count.'
'channels are listed with activity from the past week:'
' - amount of top-level content'
' - amount of unique authors'
''
(snoc out '')
==
:: gather local non-dm groups, sorted by size
::
=/ groups=(list [local=? resource:re members=@ud])
%+ murn
%~ tap in
%~ key by
dir:(scry arch %y %group-store /groups)
|= i=@ta
=/ r=resource:re (de-path:re (stab i))
=/ g=(unit group:gr)
%+ scry (unit group:gr)
[%x %group-store [%groups (snoc (en-path:re r) %noun)]]
?: |(?=(~ g) hidden.u.g)
~
`[=(our entity.r) r ~(wyt in members.u.g)]
=/ crowds=(list [resource:re @ud])
%+ sort (turn (skim groups head) tail)
|= [[* a=@ud] [* b=@ud]]
(gth a b)
:: gather local per-group channels
::
=/ channels=(map resource:re (list [module=term =resource:re]))
%- ~(gas by *(map resource:re (list [module=term =resource:re])))
%+ turn crowds
|= [r=resource:re *]
:- r
%+ murn
%~ tap by
%+ scry associations:md
[%x %metadata-store [%group (snoc (en-path:re r) %noun)]]
|= [[* m=md-resource:md] metadata:md]
::NOTE we only count graphs for now
?. &(=(%graph app-name.m) =(our creator)) ~
`[module (de-path:re app-path.m)]
:: count activity per channel
::
=/ activity=(list [resource:re members=@ud (list [resource:re mod=term week=@ud authors=@ud])])
%+ turn crowds
|= [g=resource:re m=@ud]
:+ g m
%+ turn (~(got by channels) g)
|= [m=term r=resource:re]
:+ r m
::NOTE graph-store doesn't use the full resource-style path here!
=/ upd=update:ga
%+ scry update:ga
[%x %graph-store /graph/(scot %p entity.r)/[name.r]/noun]
?> ?=(%add-graph -.q.upd)
=/ mo ((ordered-map atom node:ga) gth)
=/ week=(list [@da node:ga])
(tap:mo (subset:mo graph.q.upd ~ `(sub now ~d7)))
:- (lent week)
%~ wyt in
%+ roll week
|= [[* [author=ship *] *] a=(set ship)]
(~(put in a) author)
:: render results
::
:- (tac 'the date is ' (scot %da now))
:- :(tac 'you are in ' (render-number (lent groups)) ' group(s):')
:- =- (roll - tac)
%+ join ', '
%+ turn groups
|=([* r=resource:re *] (render-resource r))
:- :(tac 'you are hosting ' (render-number (lent crowds)) ' group(s):')
%- zing
%+ turn activity
|= [g=resource:re m=@ud chans=(list [resource:re term @ud @ud])]
^- (list @t)
:- :(tac 'group, ' (render-resource g) ', ' (render-number m))
%+ turn chans
|= [c=resource:re m=term w=@ud a=@ud]
;: tac ' chan, '
(render-resource c) ', '
m ', '
(render-number w) ', '
(render-number a)
==
::
++ scry
|* [=mold care=term app=term =path]
.^(mold (tac %g care) (scot %p our) app (scot %da now) path)
::
++ tac (cury cat 3)
::
++ render-resource
|= r=resource:re
?: shy
(crip ((x-co:co 8) (mug r)))
:(tac (scot %p entity.r) '/' name.r)
::
++ render-number
|= n=@ud
(crip ((d-co:co 1) n))
--

View File

@ -49,6 +49,14 @@
?> ?=(^ nodes.q.update) ?> ?=(^ nodes.q.update)
q.n.nodes.q.update q.n.nodes.q.update
:: ::
++ check-node-existence
|= [res=resource =index:store]
^- ?
%+ scry-for ,?
%+ weld
/node-exists/(scot %p entity.res)/[name.res]
(turn index (cury scot %ud))
::
++ get-update-log ++ get-update-log
|= rid=resource |= rid=resource
^- update-log:store ^- update-log:store

View File

@ -293,7 +293,7 @@
(on-agent:og wire sign) (on-agent:og wire sign)
[cards this] [cards this]
:_ this :_ this
~[(update-store:hc q.cage.sign)] ~[(update-store:hc rid q.cage.sign)]
== ==
++ on-leave ++ on-leave
|= =path |= =path
@ -469,15 +469,24 @@
/helper/pull-hook /helper/pull-hook
wire wire
:: ::
++ get-conversion
.^ tube:clay
%cc (scot %p our.bowl) %home (scot %da now.bowl)
/[update-mark.config]/resource
==
::
++ give-update ++ give-update
^- card ^- card
[%give %fact ~[/tracking] %pull-hook-update !>(tracking)] [%give %fact ~[/tracking] %pull-hook-update !>(tracking)]
:: ::
++ update-store ++ update-store
|= =vase |= [wire-rid=resource =vase]
^- card ^- card
=/ =wire =/ =wire
(make-wire /store) (make-wire /store)
=+ !<(rid=resource (get-conversion vase))
?> =(src.bowl (~(got by tracking) rid))
?> =(wire-rid rid)
[%pass wire %agent [our.bowl store-name.config] %poke update-mark.config vase] [%pass wire %agent [our.bowl store-name.config] %poke update-mark.config vase]
-- --
-- --

View File

@ -67,16 +67,6 @@
|* =config |* =config
$_ ^| $_ ^|
|_ bowl:gall |_ bowl:gall
::
:: +resource-for-update: get affected resource from an update
::
:: Given a vase of the update, the mark of which is
:: update-mark.config, produce the affected resource, if any.
::
++ resource-for-update
|~ vase
*(unit resource)
::
:: +take-update: handle update from store :: +take-update: handle update from store
:: ::
:: Given an update from the store, do other things after proxying :: Given an update from the store, do other things after proxying
@ -175,9 +165,11 @@
|^ |^
?- -.old ?- -.old
%1 %1
=. cards
:_(cards (build-mark:hc %sing))
=^ og-cards push-hook =^ og-cards push-hook
(on-load:og inner-state.old) (on-load:og inner-state.old)
[(weld cards og-cards) this(state old)] [(weld (flop cards) og-cards) this(state old)]
:: ::
%0 %0
%_ $ %_ $
@ -274,11 +266,18 @@
=^ cards push-hook =^ cards push-hook
(on-leave:og path) (on-leave:og path)
[cards this] [cards this]
::
++ on-arvo ++ on-arvo
|= [=wire =sign-arvo] |= [=wire =sign-arvo]
=^ cards push-hook ?. ?=([%helper %push-hook @ *] wire)
(on-arvo:og wire sign-arvo) =^ cards push-hook
[cards this] (on-arvo:og wire sign-arvo)
[cards this]
?. ?=(%resource-conversion i.t.t.wire)
(on-arvo:def wire sign-arvo)
:_ this
~[(build-mark:hc %next)]
::
++ on-fail ++ on-fail
|= [=term =tang] |= [=term =tang]
=^ cards push-hook =^ cards push-hook
@ -373,7 +372,7 @@
|= =vase |= =vase
^- (list card:agent:gall) ^- (list card:agent:gall)
=/ rid=(unit resource) =/ rid=(unit resource)
(resource-for-update:og vase) (resource-for-update vase)
?~ rid ~ ?~ rid ~
=/ prefix=path =/ prefix=path
resource+(en-path:resource u.rid) resource+(en-path:resource u.rid)
@ -390,7 +389,7 @@
|= =vase |= =vase
^- (list card:agent:gall) ^- (list card:agent:gall)
=/ rid=(unit resource) =/ rid=(unit resource)
(resource-for-update:og vase) (resource-for-update vase)
?~ rid ~ ?~ rid ~
=/ =path =/ =path
resource+(en-path:resource u.rid) resource+(en-path:resource u.rid)
@ -399,5 +398,30 @@
=/ dap=term =/ dap=term
?:(=(our.bowl entity.u.rid) store-name.config dap.bowl) ?:(=(our.bowl entity.u.rid) store-name.config dap.bowl)
[%pass wire %agent [entity.u.rid dap] %poke update-mark.config vase]~ [%pass wire %agent [entity.u.rid dap] %poke update-mark.config vase]~
::
++ get-conversion
.^ tube:clay
%cc (scot %p our.bowl) %home (scot %da now.bowl)
/[update-mark.config]/resource
==
::
++ resource-for-update
|= update=vase
=/ =tube:clay
get-conversion
%+ bind
(mole |.((tube update)))
|=(=vase !<(resource vase))
::
++ build-mark
|= rav=?(%sing %next)
^- card
=/ =wire
(make-wire /resource-conversion)
=/ =mood:clay
[%c da+now.bowl /[update-mark.config]/resource]
=/ =rave:clay
?:(?=(%next rav) [rav mood] [rav mood])
[%pass wire %arvo %c %warp our.bowl [%home `rave]]
-- --
-- --

View File

@ -7,6 +7,13 @@
|% |%
++ noun upd ++ noun upd
++ json (update:enjs upd) ++ json (update:enjs upd)
++ resource
?+ -.q.upd !!
?(%run-updates %add-nodes %remove-nodes %add-graph) resource.q.upd
?(%remove-graph %archive-graph %unarchive-graph) resource.q.upd
?(%add-tag %remove-tag) resource.q.upd
?(%add-signatures %remove-signatures) resource.uid.q.upd
==
++ mime [/application/x-urb-graph-update (as-octs (jam upd))] ++ mime [/application/x-urb-graph-update (as-octs (jam upd))]
-- --
:: ::

View File

@ -4,9 +4,13 @@
++ grow ++ grow
|% |%
++ noun upd ++ noun upd
++ resource
?< ?=(%initial -.upd)
resource.upd
::
++ json ++ json
%+ frond:enjs:format 'groupUpdate' %+ frond:enjs:format 'groupUpdate'
(update:enjs upd) (update:enjs upd)
-- --
++ grab ++ grab
|% |%

View File

@ -0,0 +1,14 @@
/+ resource
|_ rid=resource
++ grad %noun
++ grow
|%
++ noun rid
++ json (enjs:resource rid)
--
++ grab
|%
++ noun resource
++ json dejs:resource
--
--

View File

@ -34,7 +34,8 @@
== ==
:: ::
+$ logged-update-0 +$ logged-update-0
$% [%add-nodes =resource nodes=(map index node)] $% [%add-graph =resource =graph mark=(unit mark) overwrite=?]
[%add-nodes =resource nodes=(map index node)]
[%remove-nodes =resource indices=(set index)] [%remove-nodes =resource indices=(set index)]
[%add-signatures =uid =signatures] [%add-signatures =uid =signatures]
[%remove-signatures =uid =signatures] [%remove-signatures =uid =signatures]
@ -42,7 +43,6 @@
:: ::
+$ update-0 +$ update-0
$% logged-update-0 $% logged-update-0
[%add-graph =resource =graph mark=(unit mark) overwrite=?]
[%remove-graph =resource] [%remove-graph =resource]
:: ::
[%add-tag =term =resource] [%add-tag =term =resource]

View File

@ -33,7 +33,7 @@
(pure:m (need ugroup)) (pure:m (need ugroup))
:: ::
++ delete-graph ++ delete-graph
|= rid=resource |= [group-rid=resource rid=resource]
=/ m (strand ,~) =/ m (strand ,~)
^- form:m ^- form:m
;< =bowl:spider bind:m get-bowl:strandio ;< =bowl:spider bind:m get-bowl:strandio
@ -43,12 +43,9 @@
(poke-our %graph-push-hook %push-hook-action !>([%remove rid])) (poke-our %graph-push-hook %push-hook-action !>([%remove rid]))
;< ~ bind:m ;< ~ bind:m
%+ poke-our %metadata-hook %+ poke-our %metadata-hook
metadata-hook-action+!>([%remove (en-path:resource rid)])
;< ~ bind:m
%+ poke-our %metadata-store
:- %metadata-action :- %metadata-action
!> :+ %remove !> :+ %remove
(en-path:resource rid) (en-path:resource group-rid)
[%graph (en-path:resource rid)] [%graph (en-path:resource rid)]
(pure:m ~) (pure:m ~)
-- --
@ -69,11 +66,14 @@
(scry-group u.ugroup-rid) (scry-group u.ugroup-rid)
?. hidden.group ?. hidden.group
;< ~ bind:m ;< ~ bind:m
(delete-graph rid.action) (delete-graph u.ugroup-rid rid.action)
(pure:m !>(~)) (pure:m !>(~))
;< ~ bind:m ;< ~ bind:m
(poke-our %group-store %group-action !>([%remove-group rid.action ~])) (poke-our %group-store %group-action !>([%remove-group rid.action ~]))
;< ~ bind:m ;< ~ bind:m
(poke-our %group-push-hook %push-hook-action !>([%remove rid.action])) (poke-our %group-push-hook %push-hook-action !>([%remove rid.action]))
;< ~ bind:m (delete-graph rid.action) ;< ~ bind:m (delete-graph u.ugroup-rid rid.action)
;< ~ bind:m
%+ poke-our %metadata-hook
metadata-hook-action+!>([%remove (en-path:resource u.ugroup-rid)])
(pure:m !>(~)) (pure:m !>(~))

View File

@ -0,0 +1,37 @@
# Official Urbit Docker Image
This is the official Docker image for [Urbit](https://urbit.org).
Urbit is a clean-slate OS and network for the 21st century.
## Using
To use this image, you should mount a volume with a keyfile, comet file, or existing pier at `/urbit`, and map ports
as described below.
### Volume Mount
This image expects a volume mounted at `/urbit`. This volume should initially contain one of
- A keyfile `<shipname>.key` for a galaxy, star, planet, or moon. See the setup instructions for Urbit for information on [obtaining a keyfile](https://urbit.org/using/install/).
* e.g. `sampel-palnet.key` for the planet `sampel-palnet`.
- An empty file with the extension `.comet`. This will cause Urbit to boot a [comet](https://urbit.org/docs/glossary/comet/) in a pier named for the `.comet` file (less the extension).
* e.g. starting with an empty file `my-urbit-bot.comet` will result in Urbit booting a comet into the pier
`my-urbit-bot` under your volume.
- An existing pier as a directory `<shipname>`. You can migrate an existing ship to a new docker container in this way by placing its pier under the volume.
* e.g. if your ship is `sampel-palnet` then you likely have a directory `sampel-palnet` whose path you pass to `./urbit` when starting. [Move your pier](https://urbit.org/using/operations/using-your-ship/#moving-your-pier) directory to the volume and then start the container.
The first two options result in Urbit attempting to boot either the ship named by the name of the keyfile, or a comet. In both cases, after that boot is successful, the `.key` or `.comet` file will be removed from the volume and the pier will take its place.
In consequence, it is safe to remove the container and start a new container which mounts the same volume, e.g. to upgrade the version of the urbit binary by running a later container version. It is also possible to stop the container and then move the pier away e.g. to a location where you will run it directly with the Urbit binary.
### Ports
The image includes `EXPOSE` directives for TCP port 80 and UDP port 34343. Port `80` is used for Urbit's HTTP interface for both [Landscape](https://urbit.org/docs/glossary/landscape/) and for [API calls](https://urbit.org/using/integrating-api/) to the ship. Port `34343` is used by [Ames](https://urbit.org/docs/glossary/ames/) for ship-to-ship communication.
You can either pass the `-P` flag to docker to map ports directly to the corresponding ports on the host, or map them individually with `-p` flags. For local testing the latter is often convenient, for instance to remap port 80 to an unprivileged port.
## Extending
You likely do not want to extend this image. External applications which interact with Urbit do so primarily via an HTTP API, which should be exposed as described above. For containerized applications using Urbit, it is more appropriate to use a container orchestration service such as Docker Compose or Kubernetes to run Urbit alongside other containers which will interface with its API.
## Development
The docker image is built by a Nix derivation in the [`nix/pkgs/docker-image/default.nix`](https://github.com/urbit/urbit/tree/master/nix/pkgs/docker-image/default.nix) file under the Urbit git repository.

View File

@ -96,7 +96,7 @@ module.exports = {
] ]
} }
}, },
exclude: /node_modules/ exclude: /node_modules\/(?!(@tlon\/indigo-dark|@tlon\/indigo-light)\/).*/
}, },
{ {
test: /\.css$/i, test: /\.css$/i,

View File

@ -26,7 +26,7 @@ module.exports = {
] ]
} }
}, },
exclude: /node_modules/ exclude: /node_modules\/(?!(@tlon\/indigo-dark|@tlon\/indigo-light)\/).*/
}, },
{ {
test: /\.css$/i, test: /\.css$/i,

View File

@ -1687,15 +1687,20 @@
"@styled-system/css": "^5.1.5" "@styled-system/css": "^5.1.5"
} }
}, },
"@tlon/indigo-dark": {
"version": "1.0.6",
"resolved": "https://registry.npmjs.org/@tlon/indigo-dark/-/indigo-dark-1.0.6.tgz",
"integrity": "sha512-/c+3/aC+gSnLHiLwTdje7pYS84ZAR3zyMJhp2mT9BIPtk7ek/EGsrrugZjVJxeKXqy+mQpFD5TXktgAEh0Ko1A=="
},
"@tlon/indigo-light": { "@tlon/indigo-light": {
"version": "1.0.3", "version": "1.0.6",
"resolved": "https://registry.npmjs.org/@tlon/indigo-light/-/indigo-light-1.0.3.tgz", "resolved": "https://registry.npmjs.org/@tlon/indigo-light/-/indigo-light-1.0.6.tgz",
"integrity": "sha512-3OPSdf9cejP/TSzWXuBaYbzLtAfBzQnc75SlPLkoPfwpxnv1Bvy9hiWngLY0WnKRR6lMOldnkYQCCuNWeDibYQ==" "integrity": "sha512-kBzJueOoGDVF2knGt+Kf5ylvil6+V1qn8/RqAj1S6wUTnfUfAMRzDp4LQI2MxLI8Is0OG3XCErVSOUImU6R3lg=="
}, },
"@tlon/indigo-react": { "@tlon/indigo-react": {
"version": "1.2.15", "version": "1.2.16",
"resolved": "https://registry.npmjs.org/@tlon/indigo-react/-/indigo-react-1.2.15.tgz", "resolved": "https://registry.npmjs.org/@tlon/indigo-react/-/indigo-react-1.2.16.tgz",
"integrity": "sha512-h9umWEzYZwyb53ujWoCQCJQwY9RUuoDaf6189+0LH3C7y9fybJe6vzbW6g2cUVH8dXA2EZkedS5nriYR0IpQbw==", "integrity": "sha512-9bQ43cXiJGOsrihwy8+MBfG4WroKucZJOm4whfSjsNFCHorjS+5Y/6nWl2hEwHo068XONFmD7xlDE1QBMTk+pA==",
"requires": { "requires": {
"@reach/menu-button": "^0.10.5", "@reach/menu-button": "^0.10.5",
"react": "^16.13.1", "react": "^16.13.1",
@ -1703,16 +1708,16 @@
}, },
"dependencies": { "dependencies": {
"tslib": { "tslib": {
"version": "2.0.3", "version": "2.1.0",
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.0.3.tgz", "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.1.0.tgz",
"integrity": "sha512-uZtkfKblCEQtZKBF6EBXVZeQNl82yqtDQdv+eck8u7tdPxjLu2/lp5/uPW+um2tpuxINHWy3GhiccY7QgEaVHQ==" "integrity": "sha512-hcVC3wYEziELGGmEEXue7D75zbwIIVUMWAVbHItGPx0ziyXxrOMQx4rQEVEV45Ut/1IotuEvwqPopzIOkDMf0A=="
} }
} }
}, },
"@tlon/sigil-js": { "@tlon/sigil-js": {
"version": "1.4.2", "version": "1.4.3",
"resolved": "https://registry.npmjs.org/@tlon/sigil-js/-/sigil-js-1.4.2.tgz", "resolved": "https://registry.npmjs.org/@tlon/sigil-js/-/sigil-js-1.4.3.tgz",
"integrity": "sha512-meb0q0kf4S34oTKDulRMfVU6Wq/9lSOALeQil4EWttL72Lae9Fznsm+ix3tgT69g1xUpjeZIB+vqGOtAFhZX3g==", "integrity": "sha512-IaJUvAgXRmPFj5JA/MDfd+b+RFDhGdiMLfzJZKuFIQyl3Dl/3cC9HdDLCYSoK4GBTu3gZqoqi6wxZl5Xia/cSw==",
"requires": { "requires": {
"invariant": "^2.2.4", "invariant": "^2.2.4",
"svgson": "^4.0.0", "svgson": "^4.0.0",
@ -8132,6 +8137,11 @@
"integrity": "sha1-VNvzd+UUQKypCkzSdGANP/LYiKk=", "integrity": "sha1-VNvzd+UUQKypCkzSdGANP/LYiKk=",
"dev": true "dev": true
}, },
"remark-breaks": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/remark-breaks/-/remark-breaks-2.0.1.tgz",
"integrity": "sha512-CZKI8xdPUnvMqPxYEIBBUg8C0B0kyn14lkW0abzhfh/P71YRIxCC3wvBh6AejQL602OxF6kNRl1x4HAZA07JyQ=="
},
"remark-disable-tokenizers": { "remark-disable-tokenizers": {
"version": "1.0.24", "version": "1.0.24",
"resolved": "https://registry.npmjs.org/remark-disable-tokenizers/-/remark-disable-tokenizers-1.0.24.tgz", "resolved": "https://registry.npmjs.org/remark-disable-tokenizers/-/remark-disable-tokenizers-1.0.24.tgz",
@ -10128,8 +10138,7 @@
"ansi-regex": { "ansi-regex": {
"version": "2.1.1", "version": "2.1.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"aproba": { "aproba": {
"version": "1.2.0", "version": "1.2.0",
@ -10150,14 +10159,12 @@
"balanced-match": { "balanced-match": {
"version": "1.0.0", "version": "1.0.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"brace-expansion": { "brace-expansion": {
"version": "1.1.11", "version": "1.1.11",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"balanced-match": "^1.0.0", "balanced-match": "^1.0.0",
"concat-map": "0.0.1" "concat-map": "0.0.1"
@ -10172,20 +10179,17 @@
"code-point-at": { "code-point-at": {
"version": "1.1.0", "version": "1.1.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"concat-map": { "concat-map": {
"version": "0.0.1", "version": "0.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"console-control-strings": { "console-control-strings": {
"version": "1.1.0", "version": "1.1.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"core-util-is": { "core-util-is": {
"version": "1.0.2", "version": "1.0.2",
@ -10302,8 +10306,7 @@
"inherits": { "inherits": {
"version": "2.0.4", "version": "2.0.4",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"ini": { "ini": {
"version": "1.3.5", "version": "1.3.5",
@ -10315,7 +10318,6 @@
"version": "1.0.0", "version": "1.0.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"number-is-nan": "^1.0.0" "number-is-nan": "^1.0.0"
} }
@ -10330,7 +10332,6 @@
"version": "3.0.4", "version": "3.0.4",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"brace-expansion": "^1.1.7" "brace-expansion": "^1.1.7"
} }
@ -10338,14 +10339,12 @@
"minimist": { "minimist": {
"version": "1.2.5", "version": "1.2.5",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"minipass": { "minipass": {
"version": "2.9.0", "version": "2.9.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"safe-buffer": "^5.1.2", "safe-buffer": "^5.1.2",
"yallist": "^3.0.0" "yallist": "^3.0.0"
@ -10364,7 +10363,6 @@
"version": "0.5.3", "version": "0.5.3",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"minimist": "^1.2.5" "minimist": "^1.2.5"
} }
@ -10426,8 +10424,7 @@
"npm-normalize-package-bin": { "npm-normalize-package-bin": {
"version": "1.0.1", "version": "1.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"npm-packlist": { "npm-packlist": {
"version": "1.4.8", "version": "1.4.8",
@ -10455,8 +10452,7 @@
"number-is-nan": { "number-is-nan": {
"version": "1.0.1", "version": "1.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"object-assign": { "object-assign": {
"version": "4.1.1", "version": "4.1.1",
@ -10468,7 +10464,6 @@
"version": "1.4.0", "version": "1.4.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"wrappy": "1" "wrappy": "1"
} }
@ -10546,8 +10541,7 @@
"safe-buffer": { "safe-buffer": {
"version": "5.1.2", "version": "5.1.2",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"safer-buffer": { "safer-buffer": {
"version": "2.1.2", "version": "2.1.2",
@ -10583,7 +10577,6 @@
"version": "1.0.2", "version": "1.0.2",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"code-point-at": "^1.0.0", "code-point-at": "^1.0.0",
"is-fullwidth-code-point": "^1.0.0", "is-fullwidth-code-point": "^1.0.0",
@ -10603,7 +10596,6 @@
"version": "3.0.1", "version": "3.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"ansi-regex": "^2.0.0" "ansi-regex": "^2.0.0"
} }
@ -10647,14 +10639,12 @@
"wrappy": { "wrappy": {
"version": "1.0.2", "version": "1.0.2",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"yallist": { "yallist": {
"version": "3.1.1", "version": "3.1.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
} }
} }
}, },
@ -11135,8 +11125,7 @@
"ansi-regex": { "ansi-regex": {
"version": "2.1.1", "version": "2.1.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"aproba": { "aproba": {
"version": "1.2.0", "version": "1.2.0",
@ -11157,14 +11146,12 @@
"balanced-match": { "balanced-match": {
"version": "1.0.0", "version": "1.0.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"brace-expansion": { "brace-expansion": {
"version": "1.1.11", "version": "1.1.11",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"balanced-match": "^1.0.0", "balanced-match": "^1.0.0",
"concat-map": "0.0.1" "concat-map": "0.0.1"
@ -11179,20 +11166,17 @@
"code-point-at": { "code-point-at": {
"version": "1.1.0", "version": "1.1.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"concat-map": { "concat-map": {
"version": "0.0.1", "version": "0.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"console-control-strings": { "console-control-strings": {
"version": "1.1.0", "version": "1.1.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"core-util-is": { "core-util-is": {
"version": "1.0.2", "version": "1.0.2",
@ -11309,8 +11293,7 @@
"inherits": { "inherits": {
"version": "2.0.4", "version": "2.0.4",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"ini": { "ini": {
"version": "1.3.5", "version": "1.3.5",
@ -11322,7 +11305,6 @@
"version": "1.0.0", "version": "1.0.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"number-is-nan": "^1.0.0" "number-is-nan": "^1.0.0"
} }
@ -11337,7 +11319,6 @@
"version": "3.0.4", "version": "3.0.4",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"brace-expansion": "^1.1.7" "brace-expansion": "^1.1.7"
} }
@ -11345,14 +11326,12 @@
"minimist": { "minimist": {
"version": "1.2.5", "version": "1.2.5",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"minipass": { "minipass": {
"version": "2.9.0", "version": "2.9.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"safe-buffer": "^5.1.2", "safe-buffer": "^5.1.2",
"yallist": "^3.0.0" "yallist": "^3.0.0"
@ -11371,7 +11350,6 @@
"version": "0.5.3", "version": "0.5.3",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"minimist": "^1.2.5" "minimist": "^1.2.5"
} }
@ -11433,8 +11411,7 @@
"npm-normalize-package-bin": { "npm-normalize-package-bin": {
"version": "1.0.1", "version": "1.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"npm-packlist": { "npm-packlist": {
"version": "1.4.8", "version": "1.4.8",
@ -11462,8 +11439,7 @@
"number-is-nan": { "number-is-nan": {
"version": "1.0.1", "version": "1.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"object-assign": { "object-assign": {
"version": "4.1.1", "version": "4.1.1",
@ -11475,7 +11451,6 @@
"version": "1.4.0", "version": "1.4.0",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"wrappy": "1" "wrappy": "1"
} }
@ -11553,8 +11528,7 @@
"safe-buffer": { "safe-buffer": {
"version": "5.1.2", "version": "5.1.2",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"safer-buffer": { "safer-buffer": {
"version": "2.1.2", "version": "2.1.2",
@ -11590,7 +11564,6 @@
"version": "1.0.2", "version": "1.0.2",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"code-point-at": "^1.0.0", "code-point-at": "^1.0.0",
"is-fullwidth-code-point": "^1.0.0", "is-fullwidth-code-point": "^1.0.0",
@ -11610,7 +11583,6 @@
"version": "3.0.1", "version": "3.0.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true,
"optional": true,
"requires": { "requires": {
"ansi-regex": "^2.0.0" "ansi-regex": "^2.0.0"
} }
@ -11654,14 +11626,12 @@
"wrappy": { "wrappy": {
"version": "1.0.2", "version": "1.0.2",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
}, },
"yallist": { "yallist": {
"version": "3.1.1", "version": "3.1.1",
"bundled": true, "bundled": true,
"dev": true, "dev": true
"optional": true
} }
} }
}, },

View File

@ -8,9 +8,10 @@
"@reach/disclosure": "^0.10.5", "@reach/disclosure": "^0.10.5",
"@reach/menu-button": "^0.10.5", "@reach/menu-button": "^0.10.5",
"@reach/tabs": "^0.10.5", "@reach/tabs": "^0.10.5",
"@tlon/indigo-light": "^1.0.3", "@tlon/indigo-dark": "^1.0.6",
"@tlon/indigo-react": "1.2.15", "@tlon/indigo-light": "^1.0.6",
"@tlon/sigil-js": "^1.4.2", "@tlon/indigo-react": "1.2.16",
"@tlon/sigil-js": "^1.4.3",
"aws-sdk": "^2.726.0", "aws-sdk": "^2.726.0",
"big-integer": "^1.6.48", "big-integer": "^1.6.48",
"classnames": "^2.2.6", "classnames": "^2.2.6",
@ -36,6 +37,7 @@
"react-router-dom": "^5.0.0", "react-router-dom": "^5.0.0",
"react-virtuoso": "^0.20.0", "react-virtuoso": "^0.20.0",
"react-visibility-sensor": "^5.1.1", "react-visibility-sensor": "^5.1.1",
"remark-breaks": "^2.0.1",
"remark-disable-tokenizers": "^1.0.24", "remark-disable-tokenizers": "^1.0.24",
"style-loader": "^1.2.1", "style-loader": "^1.2.1",
"styled-components": "^5.1.0", "styled-components": "^5.1.0",

View File

@ -10,7 +10,7 @@
<!-- <link rel="manifest" href="%PUBLIC_URL%/manifest.json" /> --> <!-- <link rel="manifest" href="%PUBLIC_URL%/manifest.json" /> -->
<title>OS1</title> <title>Landscape</title>
</head> </head>
<body> <body>

View File

@ -52,13 +52,16 @@ const tokenizeMessage = (text) => {
} }
messages.push({ url: str }); messages.push({ url: str });
message = []; message = [];
} else if(urbitOb.isValidPatp(str) && !isInCodeBlock) { } else if (urbitOb.isValidPatp(str.replace(/[^a-z\-\~]/g, '')) && !isInCodeBlock) {
if (message.length > 0) { if (message.length > 0) {
// If we're in the middle of a message, add it to the stack and reset // If we're in the middle of a message, add it to the stack and reset
messages.push({ text: message.join(' ') }); messages.push({ text: message.join(' ') });
message = []; message = [];
} }
messages.push({ mention: str }); messages.push({ mention: str.replace(/[^a-z\-\~]/g, '') });
if (str.replace(/[a-z\-\~]/g, '').length > 0) {
messages.push({ text: str.replace(/[a-z\-\~]/g, '') });
}
message = []; message = [];
} else { } else {

View File

@ -50,11 +50,11 @@ const useS3 = (s3: S3State, { accept = '*' } = { accept: '*' }): IuseS3 => {
ACL: "public-read", ACL: "public-read",
ContentType: file.type, ContentType: file.type,
}; };
setUploading(true); setUploading(true);
const { Location } = await client.current.upload(params).promise(); const { Location } = await client.current.upload(params).promise();
setUploading(false); setUploading(false);
return Location; return Location;
@ -75,6 +75,7 @@ const useS3 = (s3: S3State, { accept = '*' } = { accept: '*' }): IuseS3 => {
const fileSelector = document.createElement('input'); const fileSelector = document.createElement('input');
fileSelector.setAttribute('type', 'file'); fileSelector.setAttribute('type', 'file');
fileSelector.setAttribute('accept', accept); fileSelector.setAttribute('accept', accept);
fileSelector.style.visibility = 'hidden';
fileSelector.addEventListener('change', () => { fileSelector.addEventListener('change', () => {
const files = fileSelector.files; const files = fileSelector.files;
if (!files || files.length <= 0) { if (!files || files.length <= 0) {
@ -82,10 +83,12 @@ const useS3 = (s3: S3State, { accept = '*' } = { accept: '*' }): IuseS3 => {
return; return;
} }
uploadDefault(files[0]).then(resolve); uploadDefault(files[0]).then(resolve);
document.body.removeChild(fileSelector);
}) })
document.body.appendChild(fileSelector);
fileSelector.click(); fileSelector.click();
}) })
}, },
[uploadDefault] [uploadDefault]
); );

View File

@ -1,4 +1,4 @@
import { useEffect } from 'react'; import { useEffect, useState } from 'react';
import _ from "lodash"; import _ from "lodash";
import f, { memoize } from "lodash/fp"; import f, { memoize } from "lodash/fp";
import bigInt, { BigInteger } from "big-integer"; import bigInt, { BigInteger } from "big-integer";
@ -13,7 +13,7 @@ export const MOMENT_CALENDAR_DATE = {
nextWeek: "dddd", nextWeek: "dddd",
lastDay: "[Yesterday]", lastDay: "[Yesterday]",
lastWeek: "[Last] dddd", lastWeek: "[Last] dddd",
sameElse: "DD/MM/YYYY", sameElse: "~YYYY.M.D",
}; };
export function appIsGraph(app: string) { export function appIsGraph(app: string) {
@ -357,7 +357,17 @@ export function pluralize(text: string, isPlural = false, vowel = false) {
return isPlural ? `${text}s`: `${vowel ? 'an' : 'a'} ${text}`; return isPlural ? `${text}s`: `${vowel ? 'an' : 'a'} ${text}`;
} }
export function useShowNickname(contact: Contact | null): boolean { // Hide is an optional second parameter for when this function is used in class components
const hideNicknames = useLocalState(state => state.hideNicknames); export function useShowNickname(contact: Contact | null, hide?: boolean): boolean {
const hideNicknames = typeof hide !== 'undefined' ? hide : useLocalState(state => state.hideNicknames);
return !!(contact && contact.nickname && !hideNicknames); return !!(contact && contact.nickname && !hideNicknames);
}
export function useHovering() {
const [hovering, setHovering] = useState(false);
const bind = {
onMouseEnter: () => setHovering(true),
onMouseLeave: () => setHovering(false)
};
return { hovering, bind };
} }

View File

@ -386,5 +386,7 @@ function archive(json: any, state: HarkState) {
notifIdxEqual(index, idxNotif.index) notifIdxEqual(index, idxNotif.index)
); );
state.notifications.set(time, unarchived); state.notifications.set(time, unarchived);
const newlyRead = archived.filter(x => !x.notification.read).length;
updateNotificationStats(state, index, 'notifications', (x) => x - newlyRead);
} }
} }

View File

@ -31,7 +31,7 @@ const useLocalState = create<LocalState>(persist((set, get) => ({
suspendedFocus: undefined, suspendedFocus: undefined,
toggleOmnibox: () => set(produce(state => { toggleOmnibox: () => set(produce(state => {
state.omniboxShown = !state.omniboxShown; state.omniboxShown = !state.omniboxShown;
if (state.suspendedFocus) { if (typeof state.suspendedFocus?.focus === 'function') {
state.suspendedFocus.focus(); state.suspendedFocus.focus();
state.suspendedFocus = undefined; state.suspendedFocus = undefined;
} else { } else {

View File

@ -11,8 +11,8 @@ import 'mousetrap-global-bind';
import './css/indigo-static.css'; import './css/indigo-static.css';
import './css/fonts.css'; import './css/fonts.css';
import light from './themes/light'; import light from '@tlon/indigo-light';
import dark from './themes/old-dark'; import dark from '@tlon/indigo-dark';
import { Text, Anchor, Row } from '@tlon/indigo-react'; import { Text, Anchor, Row } from '@tlon/indigo-react';
@ -40,7 +40,7 @@ const Root = styled.div`
background-size: cover; background-size: cover;
` : p.background?.type === 'color' ? ` ` : p.background?.type === 'color' ? `
background-color: ${p.background.color}; background-color: ${p.background.color};
` : '' ` : `background-color: ${p.theme.colors.white};`
} }
display: flex; display: flex;
flex-flow: column nowrap; flex-flow: column nowrap;
@ -90,7 +90,7 @@ class App extends React.Component {
this.themeWatcher.onchange = this.updateTheme; this.themeWatcher.onchange = this.updateTheme;
setTimeout(() => { setTimeout(() => {
// Something about how the store works doesn't like changing it // Something about how the store works doesn't like changing it
// before the app has actually rendered, hence the timeout // before the app has actually rendered, hence the timeout.
this.updateTheme(this.themeWatcher); this.updateTheme(this.themeWatcher);
}, 500); }, 500);
this.api.local.getBaseHash(); this.api.local.getBaseHash();

View File

@ -231,8 +231,8 @@ export const MessageWithSigil = (props) => {
}} }}
title={`~${msg.author}`} title={`~${msg.author}`}
>{name}</Text> >{name}</Text>
<Text flexShrink='0' gray mono className="v-mid">{timestamp}</Text> <Text flexShrink='0' fontSize='0' gray mono className="v-mid">{timestamp}</Text>
<Text flexShrink={0} gray mono ml={2} className="v-mid child dn-s">{datestamp}</Text> <Text flexShrink={0} gray mono ml={2} className="v-mid child dn-s">{datestamp}</Text>
</Box> </Box>
<ContentBox flexShrink={0} fontSize={fontSize ? fontSize : '14px'}> <ContentBox flexShrink={0} fontSize={fontSize ? fontSize : '14px'}>
{msg.contents.map(c => {msg.contents.map(c =>
@ -259,7 +259,7 @@ const ContentBox = styled(Box)`
export const MessageWithoutSigil = ({ timestamp, contacts, msg, measure, group }) => ( export const MessageWithoutSigil = ({ timestamp, contacts, msg, measure, group }) => (
<> <>
<Text flexShrink={0} mono gray display='inline-block' pt='2px' lineHeight='tall' className="child">{timestamp}</Text> <Text flexShrink={0} mono gray display='inline-block' pt='2px' lineHeight='tall' className="child" fontSize='0'>{timestamp}</Text>
<ContentBox flexShrink={0} fontSize='14px' className="clamp-message" style={{ flexGrow: 1 }}> <ContentBox flexShrink={0} fontSize='14px' className="clamp-message" style={{ flexGrow: 1 }}>
{msg.contents.map((c, i) => ( {msg.contents.map((c, i) => (
<MessageContent <MessageContent

View File

@ -258,7 +258,7 @@ export default class ChatWindow extends Component<ChatWindowProps, ChatWindowSta
const messageProps = { association, group, contacts, unreadMarkerRef, history, api }; const messageProps = { association, group, contacts, unreadMarkerRef, history, api };
const keys = graph.keys().reverse(); const keys = graph.keys().reverse();
const unreadIndex = keys[this.props.unreadCount]; const unreadIndex = graph.keys()[this.props.unreadCount];
const unreadMsg = unreadIndex && graph.get(unreadIndex); const unreadMsg = unreadIndex && graph.get(unreadIndex);
return ( return (

View File

@ -2,8 +2,9 @@ import React, { Component } from 'react';
import { UnControlled as CodeEditor } from 'react-codemirror2'; import { UnControlled as CodeEditor } from 'react-codemirror2';
import { MOBILE_BROWSER_REGEX } from "~/logic/lib/util"; import { MOBILE_BROWSER_REGEX } from "~/logic/lib/util";
import CodeMirror from 'codemirror'; import CodeMirror from 'codemirror';
import styled from "styled-components";
import { Row, BaseTextArea } from '@tlon/indigo-react'; import { Row, BaseTextArea, Box } from '@tlon/indigo-react';
import 'codemirror/mode/markdown/markdown'; import 'codemirror/mode/markdown/markdown';
import 'codemirror/addon/display/placeholder'; import 'codemirror/addon/display/placeholder';
@ -52,9 +53,40 @@ const inputProxy = (input) => new Proxy(input, {
if (property === 'setValue') { if (property === 'setValue') {
return (val) => target.value = val; return (val) => target.value = val;
} }
if (property === 'element') {
return input;
}
} }
}); });
const MobileBox = styled(Box)`
display: inline-grid;
vertical-align: center;
align-items: stretch;
position: relative;
justify-content: flex-start;
width: 100%;
&:after,
textarea {
grid-area: 2 / 1;
width: auto;
min-width: 1em;
font: inherit;
padding: 0.25em;
margin: 0;
resize: none;
background: none;
appearance: none;
border: none;
}
&::after {
content: attr(data-value) ' ';
visibility: hidden;
white-space: pre-wrap;
}
`;
export default class ChatEditor extends Component { export default class ChatEditor extends Component {
constructor(props) { constructor(props) {
super(props); super(props);
@ -161,31 +193,49 @@ export default class ChatEditor extends Component {
alignItems='center' alignItems='center'
flexGrow='1' flexGrow='1'
height='100%' height='100%'
paddingTop={MOBILE_BROWSER_REGEX.test(navigator.userAgent) ? '16px' : '0'}
paddingBottom={MOBILE_BROWSER_REGEX.test(navigator.userAgent) ? '16px' : '0'}
maxHeight='224px' maxHeight='224px'
width='calc(100% - 88px)' width='calc(100% - 88px)'
className={inCodeMode ? 'chat code' : 'chat'} className={inCodeMode ? 'chat code' : 'chat'}
color="black" color="black"
> >
{MOBILE_BROWSER_REGEX.test(navigator.userAgent) {MOBILE_BROWSER_REGEX.test(navigator.userAgent)
? <BaseTextArea ? <MobileBox
fontFamily={inCodeMode ? 'Source Code Pro' : 'Inter'} data-value={this.state.message}
fontSize="14px" fontSize="1"
lineHeight="tall" lineHeight="tall"
style={{ width: '100%', background: 'transparent', color: 'currentColor' }} onClick={event => {
placeholder={inCodeMode ? "Code..." : "Message..."} if (this.editor) {
onKeyUp={event => { this.editor.element.focus();
if (event.key === 'Enter') { }
this.submit(); }}
} else { >
<BaseTextArea
fontFamily={inCodeMode ? 'Source Code Pro' : 'Inter'}
fontSize="1"
lineHeight="tall"
rows="1"
style={{ width: '100%', background: 'transparent', color: 'currentColor' }}
placeholder={inCodeMode ? "Code..." : "Message..."}
onChange={event => {
this.messageChange(null, null, event.target.value); this.messageChange(null, null, event.target.value);
} }}
}} onKeyDown={event => {
ref={input => { if (event.key === 'Enter') {
if (!input) return; event.preventDefault();
this.editor = inputProxy(input); this.submit();
}} } else {
{...props} this.messageChange(null, null, event.target.value);
/> }
}}
ref={input => {
if (!input) return;
this.editor = inputProxy(input);
}}
{...props}
/>
</MobileBox>
: <CodeEditor : <CodeEditor
className="lh-copy" className="lh-copy"
value={message} value={message}

View File

@ -12,6 +12,7 @@ export default class CodeContent extends Component {
( (
<Text <Text
display='block' display='block'
fontSize='0'
mono mono
p='1' p='1'
my='0' my='0'
@ -37,6 +38,7 @@ export default class CodeContent extends Component {
overflow='auto' overflow='auto'
maxHeight='10em' maxHeight='10em'
maxWidth='100%' maxWidth='100%'
fontSize='0'
style={{ whiteSpace: 'pre' }} style={{ whiteSpace: 'pre' }}
> >
{content.code.expression} {content.code.expression}

View File

@ -2,6 +2,7 @@ import React, { Component } from 'react';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import ReactMarkdown from 'react-markdown'; import ReactMarkdown from 'react-markdown';
import RemarkDisableTokenizers from 'remark-disable-tokenizers'; import RemarkDisableTokenizers from 'remark-disable-tokenizers';
import RemarkBreaks from 'remark-breaks';
import urbitOb from 'urbit-ob'; import urbitOb from 'urbit-ob';
import { Text } from '@tlon/indigo-react'; import { Text } from '@tlon/indigo-react';
@ -26,10 +27,10 @@ const DISABLED_INLINE_TOKENS = [
const renderers = { const renderers = {
inlineCode: ({language, value}) => { inlineCode: ({language, value}) => {
return <Text mono p='1' backgroundColor='washedGray' style={{ whiteSpace: 'preWrap'}}>{value}</Text> return <Text mono p='1' backgroundColor='washedGray' fontSize='0' style={{ whiteSpace: 'preWrap'}}>{value}</Text>
}, },
paragraph: ({ children }) => { paragraph: ({ children }) => {
return (<Text fontSize="14px">{children}</Text>); return (<Text fontSize="1">{children}</Text>);
}, },
code: ({language, value}) => { code: ({language, value}) => {
return <Text return <Text
@ -38,6 +39,7 @@ const renderers = {
display='block' display='block'
borderRadius='1' borderRadius='1'
mono mono
fontSize='0'
backgroundColor='washedGray' backgroundColor='washedGray'
overflowX='auto' overflowX='auto'
style={{ whiteSpace: 'pre'}}> style={{ whiteSpace: 'pre'}}>
@ -66,6 +68,7 @@ const MessageMarkdown = React.memo(props => (
return true; return true;
}} }}
plugins={[[ plugins={[[
RemarkBreaks,
RemarkDisableTokenizers, RemarkDisableTokenizers,
{ block: DISABLED_BLOCK_TOKENS, inline: DISABLED_INLINE_TOKENS } { block: DISABLED_BLOCK_TOKENS, inline: DISABLED_INLINE_TOKENS }
]]} /> ]]} />

View File

@ -277,9 +277,6 @@ pre.CodeMirror-placeholder.CodeMirror-line-like { color: var(--gray); }
/* dark */ /* dark */
@media (prefers-color-scheme: dark) { @media (prefers-color-scheme: dark) {
body {
background-color: #333;
}
.bg-black-d { .bg-black-d {
background-color: black; background-color: black;
} }

View File

@ -1,5 +1,4 @@
import React, { useState } from 'react'; import React, { useState } from 'react';
import Helmet from 'react-helmet';
import styled from 'styled-components'; import styled from 'styled-components';
import { Box, Row, Icon, Text } from '@tlon/indigo-react'; import { Box, Row, Icon, Text } from '@tlon/indigo-react';
@ -14,6 +13,7 @@ import ModalButton from './components/ModalButton';
import { writeText } from '~/logic/lib/util'; import { writeText } from '~/logic/lib/util';
import { NewGroup } from "~/views/landscape/components/NewGroup"; import { NewGroup } from "~/views/landscape/components/NewGroup";
import { JoinGroup } from "~/views/landscape/components/JoinGroup"; import { JoinGroup } from "~/views/landscape/components/JoinGroup";
import { Helmet } from 'react-helmet';
const ScrollbarLessBox = styled(Box)` const ScrollbarLessBox = styled(Box)`
scrollbar-width: none !important; scrollbar-width: none !important;
@ -25,13 +25,38 @@ const ScrollbarLessBox = styled(Box)`
export default function LaunchApp(props) { export default function LaunchApp(props) {
const [hashText, setHashText] = useState(props.baseHash); const [hashText, setHashText] = useState(props.baseHash);
const hashBox = (
<Box
position={["relative", "absolute"]}
fontFamily="mono"
left="0"
bottom="0"
color="scales.black20"
bg="white"
ml={3}
mb={3}
borderRadius={2}
fontSize={0}
p={2}
boxShadow="0 0 0px 1px inset"
cursor="pointer"
onClick={() => {
writeText(props.baseHash);
setHashText('copied');
setTimeout(() => {
setHashText(props.baseHash);
}, 2000);
}}
>
<Text color="gray">{hashText || props.baseHash}</Text>
</Box>
);
return ( return (
<> <>
<Helmet> <Helmet defer={false}>
<title>OS1 - Home</title> <title>{ props.notificationsCount ? `(${String(props.notificationsCount) }) `: '' }Landscape</title>
</Helmet> </Helmet>
<ScrollbarLessBox height='100%' overflowY='scroll'> <ScrollbarLessBox height='100%' overflowY='scroll' display="flex" flexDirection="column">
<Welcome firstTime={props.launch.firstTime} api={props.api} /> <Welcome firstTime={props.launch.firstTime} api={props.api} />
<Box <Box
mx='2' mx='2'
@ -53,7 +78,7 @@ export default function LaunchApp(props) {
color="black" color="black"
icon="Mail" icon="Mail"
/> />
<Text ml="1" mt='1px' color="black">DMs + Drafts</Text> <Text ml="2" mt='1px' color="black">DMs + Drafts</Text>
</Row> </Row>
</Box> </Box>
</Tile> </Tile>
@ -77,36 +102,15 @@ export default function LaunchApp(props) {
icon="CreateGroup" icon="CreateGroup"
bg="green" bg="green"
color="#fff" color="#fff"
text="Create a Group" text="Create Group"
> >
<NewGroup {...props} /> <NewGroup {...props} />
</ModalButton> </ModalButton>
<Groups unreads={props.unreads} groups={props.groups} associations={props.associations} /> <Groups unreads={props.unreads} groups={props.groups} associations={props.associations} />
</Box> </Box>
<Box alignSelf="flex-start" display={["block", "none"]}>{hashBox}</Box>
</ScrollbarLessBox> </ScrollbarLessBox>
<Box <Box display={["none", "block"]}>{hashBox}</Box>
position="absolute"
fontFamily="mono"
left="0"
bottom="0"
color="gray"
bg="white"
ml={3}
mb={3}
borderRadius={2}
fontSize={0}
p={2}
cursor="pointer"
onClick={() => {
writeText(props.baseHash);
setHashText('copied');
setTimeout(() => {
setHashText(props.baseHash);
}, 2000);
}}
>
{hashText || props.baseHash}
</Box>
</> </>
); );
} }

View File

@ -20,8 +20,8 @@ export default class BasicTile extends React.PureComponent {
size='12px' size='12px'
display='inline-block' display='inline-block'
verticalAlign='top' verticalAlign='top'
pt='5px' mt='5px'
pr='2px' mr='2'
/> />
: null : null
}{props.title} }{props.title}

View File

@ -171,7 +171,7 @@ export default class WeatherTile extends React.Component {
onClick={() => this.setState({ manualEntry: !this.state.manualEntry })} onClick={() => this.setState({ manualEntry: !this.state.manualEntry })}
> >
<Box> <Box>
<Icon icon='Weather' display='inline-block' verticalAlign='top' pt='3px' pr='2px' /> <Icon icon='Weather' display='inline-block' verticalAlign='top' mt='3px' mr='2' />
<Text>Weather</Text> <Text>Weather</Text>
</Box> </Box>
<Text style={{ cursor: 'pointer' }}> <Text style={{ cursor: 'pointer' }}>
@ -217,15 +217,14 @@ export default class WeatherTile extends React.Component {
title={`${locationName} Weather`} title={`${locationName} Weather`}
> >
<Text> <Text>
<Icon icon='Weather' display='inline' style={{ position: 'relative', top: '.3em' }} /> <Icon icon='Weather' display='inline' mr='2' style={{ position: 'relative', top: '.3em' }} />
Weather
<Text <Text
cursor='pointer' cursor='pointer'
onClick={() => onClick={() =>
this.setState({ manualEntry: !this.state.manualEntry }) this.setState({ manualEntry: !this.state.manualEntry })
} }
> >
-> Weather ->
</Text> </Text>
</Text> </Text>
@ -268,7 +267,7 @@ export default class WeatherTile extends React.Component {
flexDirection="column" flexDirection="column"
justifyContent="flex-start" justifyContent="flex-start"
> >
<Text><Icon icon='Weather' color='black' display='inline' style={{ position: 'relative', top: '.3em' }} /> Weather</Text> <Text><Icon icon='Weather' color='black' display='inline' mr='2' style={{ position: 'relative', top: '.3em' }} /> Weather</Text>
<Text width='100%' display='flex' flexDirection='column' mt={1}> <Text width='100%' display='flex' flexDirection='column' mt={1}>
Loading, please check again later... Loading, please check again later...
</Text> </Text>

View File

@ -40,12 +40,12 @@ button {
/* stolen from indigo-react reset.css /* stolen from indigo-react reset.css
* TODO: remove and add reset.css properly * TODO: remove and add reset.css properly
*/ */
@keyframes loadingSpinnerRotation { @keyframes loadingSpinnerRotation {
from { from {
transform: rotate(0deg); transform: rotate(0deg);
} }
to { to {
transform: rotate(360deg); transform: rotate(360deg);
} }
@ -53,9 +53,6 @@ button {
/* dark */ /* dark */
@media all and (prefers-color-scheme: dark) { @media all and (prefers-color-scheme: dark) {
body {
background-color: #333;
}
.bg-gray0-d { .bg-gray0-d {
background-color: #333; background-color: #333;
} }

View File

@ -72,7 +72,7 @@ export const LinkItem = (props: LinkItemProps) => {
} }
return ( return (
<Box width="100%" {...rest}> <Box width="100%" {...rest}>
<Box <Box
lineHeight="tall" lineHeight="tall"
display='flex' display='flex'
@ -116,9 +116,9 @@ export const LinkItem = (props: LinkItemProps) => {
</Anchor> </Anchor>
</Text> </Text>
</Box> </Box>
<Row minWidth='0' flexShrink={0} width="100%" justifyContent="space-between" py={3} bg="white"> <Row minWidth='0' flexShrink={0} width="100%" justifyContent="space-between" py={3} bg="white">
<Author <Author
showImage showImage
contacts={contacts} contacts={contacts}
@ -136,9 +136,9 @@ export const LinkItem = (props: LinkItemProps) => {
</Box> </Box>
</Link> </Link>
</Box> </Box>
<Dropdown <Dropdown
width="200px" dropWidth="200px"
alignX="right" alignX="right"
alignY="top" alignY="top"
options={ options={
@ -156,7 +156,7 @@ export const LinkItem = (props: LinkItemProps) => {
> >
<Icon ml="2" display="block" icon="Ellipsis" color="gray" /> <Icon ml="2" display="block" icon="Ellipsis" color="gray" />
</Dropdown> </Dropdown>
</Row> </Row>
</Box>); </Box>);
}; };

View File

@ -132,7 +132,6 @@ const LinkSubmit = (props: LinkSubmitProps) => {
position="absolute" position="absolute"
px={2} px={2}
pt={2} pt={2}
fontSize={0}
style={{ pointerEvents: 'none' }} style={{ pointerEvents: 'none' }}
>{canUpload >{canUpload
? <> ? <>
@ -180,7 +179,6 @@ const LinkSubmit = (props: LinkSubmitProps) => {
type="url" type="url"
pl={2} pl={2}
width="100%" width="100%"
fontSize={0}
py={2} py={2}
color="black" color="black"
backgroundColor="transparent" backgroundColor="transparent"
@ -198,8 +196,8 @@ const LinkSubmit = (props: LinkSubmitProps) => {
pl={2} pl={2}
backgroundColor="transparent" backgroundColor="transparent"
width="100%" width="100%"
fontSize={0}
color="black" color="black"
fontSize={1}
style={{ style={{
resize: 'none', resize: 'none',
height: 40 height: 40

View File

@ -1,105 +0,0 @@
import React, { useCallback } from "react";
import _ from "lodash";
import { Link } from "react-router-dom";
import GlobalApi from "~/logic/api/global";
import {
Rolodex,
Associations,
ChatNotifIndex,
ChatNotificationContents,
Groups,
} from "~/types";
import { BigInteger } from "big-integer";
import { Box, Col } from "@tlon/indigo-react";
import { Header } from "./header";
import { pluralize } from "~/logic/lib/util";
import ChatMessage from "../chat/components/ChatMessage";
function describeNotification(mention: boolean, lent: number) {
const msg = pluralize("message", lent !== 1);
if (mention) {
return `mentioned you in ${msg} in`;
}
return `sent ${msg} in`;
}
export function ChatNotification(props: {
index: ChatNotifIndex;
contents: ChatNotificationContents;
archived: boolean;
read: boolean;
time: number;
timebox: BigInteger;
associations: Associations;
contacts: Rolodex;
groups: Groups;
api: GlobalApi;
}) {
const { index, contents, read, time, api, timebox } = props;
const authors = _.map(contents, "author");
const { chat, mention } = index;
const association = props?.associations?.chat?.[chat];
const groupPath = association?.["group-path"];
const appPath = index?.chat;
const group = props?.groups?.[groupPath];
const desc = describeNotification(mention, contents.length);
const groupContacts = props.contacts[groupPath] || {};
const onClick = useCallback(() => {
if (props.archived) {
return;
}
const func = read ? "unread" : "read";
return api.hark[func](timebox, { chat: index });
}, [api, timebox, index, read]);
return (
<Col onClick={onClick} flexGrow="1" p="2">
<Header
chat
associations={props.associations}
read={read}
archived={props.archived}
time={time}
authors={authors}
moduleIcon="Chat"
channel={chat}
contacts={props.contacts}
group={groupPath}
description={desc}
/>
<Col pb="3" pl="5">
{_.map(_.take(contents, 5), (content, idx) => {
let workspace = groupPath;
if (workspace === undefined || group?.hidden) {
workspace = '/home';
}
const to = `/~landscape${workspace}/resource/chat${appPath}?msg=${content.number}`;
return (
<Link key={idx} to={to}>
<ChatMessage
measure={() => {}}
msg={content}
isLastRead={false}
group={group}
contacts={groupContacts}
fontSize='0'
pt='2'
/>
</Link>
);
})}
{contents.length > 5 && (
<Box ml="4" mt="3" mb="2" color="gray" fontSize="14px">
and {contents.length - 5} other message
{contents.length > 6 ? "s" : ""}
</Box>
)}
</Col>
</Col>
);
}

View File

@ -25,7 +25,7 @@ import ChatMessage, {MessageWithoutSigil} from "../chat/components/ChatMessage";
function getGraphModuleIcon(module: string) { function getGraphModuleIcon(module: string) {
if (module === "link") { if (module === "link") {
return "Links"; return "Collection";
} }
return _.capitalize(module); return _.capitalize(module);
} }
@ -90,6 +90,8 @@ const GraphNodeContent = ({ group, post, contacts, mod, description, index, remo
content={contents} content={contents}
group={group} group={group}
contacts={contacts} contacts={contacts}
fontSize='14px'
lineHeight="tall"
/> />
} else if (idx[1] === "1") { } else if (idx[1] === "1") {
const [{ text: header }, { text: body }] = contents; const [{ text: header }, { text: body }] = contents;
@ -164,13 +166,14 @@ const GraphNode = ({
group, group,
read, read,
onRead, onRead,
showContact = false,
remoteContentPolicy remoteContentPolicy
}) => { }) => {
const { contents } = post; const { contents } = post;
author = deSig(author); author = deSig(author);
const history = useHistory(); const history = useHistory();
const img = ( const img = showContact ? (
<Sigil <Sigil
ship={`~${author}`} ship={`~${author}`}
size={16} size={16}
@ -178,7 +181,7 @@ const GraphNode = ({
color={`#000000`} color={`#000000`}
classes="mix-blend-diff" classes="mix-blend-diff"
/> />
); ) : <Box style={{ width: '16px' }}></Box>;
const groupContacts = contacts[groupPath] ?? {}; const groupContacts = contacts[groupPath] ?? {};
@ -192,10 +195,10 @@ const GraphNode = ({
}, [read, onRead]); }, [read, onRead]);
return ( return (
<Row onClick={onClick} gapX="2" pt="2"> <Row onClick={onClick} gapX="2" pt={showContact ? 2 : 0}>
<Col>{img}</Col> <Col>{img}</Col>
<Col flexGrow={1} alignItems="flex-start"> <Col flexGrow={1} alignItems="flex-start">
<Row {showContact && <Row
mb="2" mb="2"
height="16px" height="16px"
alignItems="center" alignItems="center"
@ -208,8 +211,8 @@ const GraphNode = ({
<Text ml="2" gray> <Text ml="2" gray>
{moment(time).format("HH:mm")} {moment(time).format("HH:mm")}
</Text> </Text>
</Row> </Row>}
<Row width="100%" p="1"> <Row width="100%" p="1" flexDirection="column">
<GraphNodeContent <GraphNodeContent
contacts={groupContacts} contacts={groupContacts}
post={post} post={post}
@ -244,16 +247,15 @@ export function GraphNotification(props: {
const desc = describeNotification(index.description, contents.length !== 1); const desc = describeNotification(index.description, contents.length !== 1);
const onClick = useCallback(() => { const onClick = useCallback(() => {
if (props.archived) { if (props.archived || read) {
return; return;
} }
const func = read ? "unread" : "read"; return api.hark["read"](timebox, { graph: index });
return api.hark[func](timebox, { graph: index });
}, [api, timebox, index, read]); }, [api, timebox, index, read]);
return ( return (
<Col flexGrow={1} width="100%" p="2"> <>
<Header <Header
onClick={onClick} onClick={onClick}
archived={props.archived} archived={props.archived}
@ -267,7 +269,7 @@ return (
description={desc} description={desc}
associations={props.associations} associations={props.associations}
/> />
<Col flexGrow={1} width="100%" pl="5"> <Box flexGrow={1} width="100%" pl={5} gridArea="main">
{_.map(contents, (content, idx) => ( {_.map(contents, (content, idx) => (
<GraphNode <GraphNode
post={content} post={content}
@ -282,9 +284,10 @@ return (
groupPath={group} groupPath={group}
read={read} read={read}
onRead={onClick} onRead={onClick}
showContact={idx === 0}
/> />
))} ))}
</Col> </Box>
</Col> </>
); );
} }

View File

@ -1,5 +1,5 @@
import React from "react"; import React from "react";
import { Text as NormalText, Row, Icon, Rule } from "@tlon/indigo-react"; import { Text as NormalText, Row, Icon, Rule, Box } from "@tlon/indigo-react";
import f from "lodash/fp"; import f from "lodash/fp";
import _ from "lodash"; import _ from "lodash";
import moment from "moment"; import moment from "moment";
@ -71,12 +71,13 @@ export function Header(props: {
channel; channel;
return ( return (
<Row onClick={props.onClick} p="2" flexWrap="wrap" gapX="1" alignItems="center"> <Row onClick={props.onClick} p="2" flexWrap="wrap" alignItems="center" gridArea="header">
{!props.archived && ( {!props.archived && (
<Icon <Icon
display="block" display="block"
mr="1" opacity={read ? 0 : 1}
icon={read ? "Circle" : "Bullet"} mr={2}
icon="Bullet"
color="blue" color="blue"
/> />
)} )}
@ -84,13 +85,13 @@ export function Header(props: {
{authorDesc} {authorDesc}
</Text> </Text>
<Text mr="1">{description}</Text> <Text mr="1">{description}</Text>
{!!moduleIcon && <Icon icon={moduleIcon as any} />} {!!moduleIcon && <Icon icon={moduleIcon as any} mr={1} />}
{!!channel && <Text fontWeight="500">{channelTitle}</Text>} {!!channel && <Text fontWeight="500" mr={1}>{channelTitle}</Text>}
<Rule vertical height="12px" /> <Rule vertical height="12px" mr={1} />
{groupTitle && {groupTitle &&
<> <>
<Text fontWeight="500">{groupTitle}</Text> <Text fontWeight="500" mr={1}>{groupTitle}</Text>
<Rule vertical height="12px"/> <Rule vertical height="12px" mr={1} />
</> </>
} }
<Text fontWeight="regular" color="lightGray"> <Text fontWeight="regular" color="lightGray">

View File

@ -61,20 +61,36 @@ export default function Inbox(props: {
}; };
}, []); }, []);
const [newNotifications, ...notifications] = const notifications =
Array.from(props.showArchive ? props.archive : props.notifications) || []; Array.from(props.showArchive ? props.archive : props.notifications) || [];
const calendar = {
...MOMENT_CALENDAR_DATE, sameDay: function (now) {
if (this.subtract(6, 'hours').isBefore(now)) {
return "[Earlier Today]";
} else {
return MOMENT_CALENDAR_DATE.sameDay;
}
}
};
const notificationsByDay = f.flow( let notificationsByDay = f.flow(
f.map<DatedTimebox>(([date, nots]) => [ f.map<DatedTimebox>(([date, nots]) => [
date, date,
nots.filter(filterNotification(associations, props.filter)), nots.filter(filterNotification(associations, props.filter)),
]), ]),
f.groupBy<DatedTimebox>(([date]) => f.groupBy<DatedTimebox>(([date]) => {
moment(daToUnix(date)).format("DDMMYYYY") date = moment(daToUnix(date));
), if (moment().subtract(6, 'hours').isBefore(date)) {
f.values, return 'latest';
f.reverse } else {
return date.format("YYYYMMDD");
}
}),
)(notifications); )(notifications);
notificationsByDay = new Map(Object.keys(notificationsByDay).sort().reverse().map(timebox => {
return [timebox, notificationsByDay[timebox]];
}));
useEffect(() => { useEffect(() => {
api.hark.getMore(props.showArchive); api.hark.getMore(props.showArchive);
@ -133,38 +149,24 @@ export default function Inbox(props: {
<Col zIndex={4} gapY={2} bg="white" top="0px" position="sticky"> <Col zIndex={4} gapY={2} bg="white" top="0px" position="sticky">
{inviteItems(invites, api)} {inviteItems(invites, api)}
</Col> </Col>
{newNotifications && ( {[...notificationsByDay.keys()].map((day, index) => {
<DaySection const timeboxes = notificationsByDay.get(day);
latest return timeboxes.length > 0 && (
timeboxes={[newNotifications]} <DaySection
contacts={props.contacts} key={day}
archive={!!props.showArchive} label={day === 'latest' ? 'Today' : moment(day).calendar(null, calendar)}
associations={props.associations} timeboxes={timeboxes}
groups={props.groups} contacts={props.contacts}
graphConfig={props.notificationsGraphConfig} archive={!!props.showArchive}
groupConfig={props.notificationsGroupConfig} associations={props.associations}
chatConfig={props.notificationsChatConfig} api={api}
api={api} groups={props.groups}
/> graphConfig={props.notificationsGraphConfig}
)} groupConfig={props.notificationsGroupConfig}
{_.map( chatConfig={props.notificationsChatConfig}
notificationsByDay, />
(timeboxes, idx) => );
timeboxes.length > 0 && ( })}
<DaySection
key={idx}
timeboxes={timeboxes}
contacts={props.contacts}
archive={!!props.showArchive}
associations={props.associations}
api={api}
groups={props.groups}
graphConfig={props.notificationsGraphConfig}
groupConfig={props.notificationsGroupConfig}
chatConfig={props.notificationsChatConfig}
/>
)
)}
</Col> </Col>
); );
} }
@ -181,21 +183,18 @@ function sortIndexedNotification(
} }
function DaySection({ function DaySection({
label,
contacts, contacts,
groups, groups,
archive, archive,
timeboxes, timeboxes,
latest = false,
associations, associations,
api, api,
groupConfig, groupConfig,
graphConfig, graphConfig,
chatConfig, chatConfig,
remoteContentPolicy
}) { }) {
const calendar = latest
? MOMENT_CALENDAR_DATE
: { ...MOMENT_CALENDAR_DATE, sameDay: "[Earlier Today]" };
const lent = timeboxes.map(([,nots]) => nots.length).reduce(f.add, 0); const lent = timeboxes.map(([,nots]) => nots.length).reduce(f.add, 0);
if (lent === 0 || timeboxes.length === 0) { if (lent === 0 || timeboxes.length === 0) {
return null; return null;
@ -206,7 +205,7 @@ function DaySection({
<Box position="sticky" zIndex="3" top="-1px" bg="white"> <Box position="sticky" zIndex="3" top="-1px" bg="white">
<Box p="2" bg="scales.black05"> <Box p="2" bg="scales.black05">
<Text> <Text>
{moment(daToUnix(timeboxes[0][0])).calendar(null, calendar)} {label}
</Text> </Text>
</Box> </Box>
</Box> </Box>

View File

@ -1,5 +1,5 @@
import React, { ReactNode, useCallback, useMemo } from "react"; import React, { ReactNode, useCallback, useMemo, useState } from "react";
import { Row, Box, Col, Text, Anchor, Icon, Action } from "@tlon/indigo-react"; import { Row, Box } from "@tlon/indigo-react";
import _ from "lodash"; import _ from "lodash";
import { import {
GraphNotificationContents, GraphNotificationContents,
@ -7,7 +7,6 @@ import {
GroupNotificationContents, GroupNotificationContents,
NotificationGraphConfig, NotificationGraphConfig,
GroupNotificationsConfig, GroupNotificationsConfig,
NotifIndex,
Groups, Groups,
Associations, Associations,
Contacts, Contacts,
@ -17,8 +16,8 @@ import { getParentIndex } from "~/logic/lib/notification";
import { StatelessAsyncAction } from "~/views/components/StatelessAsyncAction"; import { StatelessAsyncAction } from "~/views/components/StatelessAsyncAction";
import { GroupNotification } from "./group"; import { GroupNotification } from "./group";
import { GraphNotification } from "./graph"; import { GraphNotification } from "./graph";
import { ChatNotification } from "./chat";
import { BigInteger } from "big-integer"; import { BigInteger } from "big-integer";
import { useHovering } from "~/logic/lib/util";
interface NotificationProps { interface NotificationProps {
notification: IndexedNotification; notification: IndexedNotification;
@ -55,9 +54,6 @@ function getMuted(
if ("group" in index) { if ("group" in index) {
return _.findIndex(groups || [], (g) => g === index.group.group) === -1; return _.findIndex(groups || [], (g) => g === index.group.group) === -1;
} }
if ("chat" in index) {
return _.findIndex(chat || [], (c) => c === index.chat.chat) === -1;
}
return false; return false;
} }
@ -89,11 +85,21 @@ function NotificationWrapper(props: {
return api.hark[func](notif); return api.hark[func](notif);
}, [notif, api, isMuted]); }, [notif, api, isMuted]);
const { hovering, bind } = useHovering();
const changeMuteDesc = isMuted ? "Unmute" : "Mute"; const changeMuteDesc = isMuted ? "Unmute" : "Mute";
return ( return (
<Row width="100%" flexShrink={0} alignItems="top" justifyContent="space-between"> <Box
width="100%"
display="grid"
gridTemplateColumns="1fr 200px"
gridTemplateRows="auto"
gridTemplateAreas="'header actions' 'main main'"
pb={2}
{...bind}
>
{children} {children}
<Row gapX="2" p="2" pt='3' alignItems="top"> <Row gapX="2" p="2" pt='3' gridArea="actions" justifyContent="flex-end" opacity={[1, hovering ? 1 : 0]}>
<StatelessAsyncAction name={changeMuteDesc} onClick={onChangeMute} backgroundColor="transparent"> <StatelessAsyncAction name={changeMuteDesc} onClick={onChangeMute} backgroundColor="transparent">
{changeMuteDesc} {changeMuteDesc}
</StatelessAsyncAction> </StatelessAsyncAction>
@ -103,7 +109,7 @@ function NotificationWrapper(props: {
</StatelessAsyncAction> </StatelessAsyncAction>
)} )}
</Row> </Row>
</Row> </Box>
); );
} }
@ -166,26 +172,6 @@ export function Notification(props: NotificationProps) {
</Wrapper> </Wrapper>
); );
} }
if ("chat" in notification.index) {
const index = notification.index.chat;
const c: ChatNotificationContents = (contents as any).chat;
return (
<Wrapper>
<ChatNotification
api={props.api}
index={index}
contents={c}
contacts={props.contacts}
read={read}
archived={archived}
groups={props.groups}
timebox={props.time}
time={time}
associations={associations}
/>
</Wrapper>
);
}
return null; return null;
} }

View File

@ -2,6 +2,7 @@ import React, { useCallback, useState } from "react";
import _ from 'lodash'; import _ from 'lodash';
import { Box, Col, Text, Row } from "@tlon/indigo-react"; import { Box, Col, Text, Row } from "@tlon/indigo-react";
import { Link, Switch, Route } from "react-router-dom"; import { Link, Switch, Route } from "react-router-dom";
import Helmet from "react-helmet";
import { Body } from "~/views/components/Body"; import { Body } from "~/views/components/Body";
import { PropFunc } from "~/types/util"; import { PropFunc } from "~/types/util";
@ -52,74 +53,79 @@ export default function NotificationsScreen(props: any) {
render={(routeProps) => { render={(routeProps) => {
const { view } = routeProps.match.params; const { view } = routeProps.match.params;
return ( return (
<Body> <>
<Col overflowY="hidden" height="100%"> <Helmet defer={false}>
<Row <title>{ props.notificationsCount ? `(${String(props.notificationsCount) }) `: '' }Landscape - Notifications</title>
p="3" </Helmet>
alignItems="center" <Body>
height="48px" <Col overflowY="hidden" height="100%">
justifyContent="space-between" <Row
width="100%" p="3"
borderBottom="1" alignItems="center"
borderBottomColor="washedGray" height="48px"
> justifyContent="space-between"
<Text>Updates</Text> width="100%"
<Row> borderBottom="1"
<Box> borderBottomColor="washedGray"
<HeaderLink current={view} view="">
Inbox
</HeaderLink>
</Box>
<Box>
<HeaderLink current={view} view="preferences">
Preferences
</HeaderLink>
</Box>
</Row>
<Dropdown
alignX="right"
alignY="top"
options={
<Col
p="2"
backgroundColor="white"
border={1}
borderRadius={1}
borderColor="lightGray"
gapY="2"
>
<FormikOnBlur
initialValues={filter}
onSubmit={onSubmit}
>
<GroupSearch
id="groups"
label="Filter Groups"
caption="Only show notifications from this group"
associations={props.associations}
/>
</FormikOnBlur>
</Col>
}
> >
<Box> <Text>Updates</Text>
<Text mr="1" gray> <Row>
Filter: <Box>
</Text> <HeaderLink current={view} view="">
<Text>{groupFilterDesc}</Text> Inbox
</Box> </HeaderLink>
</Dropdown> </Box>
</Row> <Box>
{view === "preferences" && ( <HeaderLink current={view} view="preferences">
<NotificationPreferences Preferences
graphConfig={props.notificationsGraphConfig} </HeaderLink>
api={props.api} </Box>
dnd={props.doNotDisturb} </Row>
/> <Dropdown
)} alignX="right"
{!view && <Inbox {...props} filter={filter.groups} />} alignY="top"
</Col> options={
</Body> <Col
p="2"
backgroundColor="white"
border={1}
borderRadius={1}
borderColor="lightGray"
gapY="2"
>
<FormikOnBlur
initialValues={filter}
onSubmit={onSubmit}
>
<GroupSearch
id="groups"
label="Filter Groups"
caption="Only show notifications from this group"
associations={props.associations}
/>
</FormikOnBlur>
</Col>
}
>
<Box>
<Text mr="1" gray>
Filter:
</Text>
<Text>{groupFilterDesc}</Text>
</Box>
</Dropdown>
</Row>
{view === "preferences" && (
<NotificationPreferences
graphConfig={props.notificationsGraphConfig}
api={props.api}
dnd={props.doNotDisturb}
/>
)}
{!view && <Inbox {...props} filter={filter.groups} />}
</Col>
</Body>
</>
); );
}} }}
/> />

View File

@ -33,7 +33,7 @@ const SidebarItem = ({ children, view, current }) => {
backgroundColor={selected ? "washedGray" : "white"} backgroundColor={selected ? "washedGray" : "white"}
> >
<Icon mr={2} display="inline-block" icon={icon(view)} color='black' /> <Icon mr={2} display="inline-block" icon={icon(view)} color='black' />
<Text color='black' fontSize={0}> <Text color='black'>
{children} {children}
</Text> </Text>
</Row> </Row>
@ -47,7 +47,7 @@ export default function ProfileScreen(props: any) {
return ( return (
<> <>
<Helmet defer={false}> <Helmet defer={false}>
<title>OS1 - Profile</title> <title>{ props.notificationsCount ? `(${String(props.notificationsCount) }) `: '' }Landscape - Profile</title>
</Helmet> </Helmet>
<Switch> <Switch>
<Route <Route
@ -76,7 +76,7 @@ export default function ProfileScreen(props: any) {
height="100%" height="100%"
width="100%" width="100%"
display="grid" display="grid"
gridTemplateColumns={["100%", "200px 1fr"]} gridTemplateColumns={["100%", "250px 1fr"]}
gridTemplateRows={["48px 1fr", "1fr"]} gridTemplateRows={["48px 1fr", "1fr"]}
borderRadius={1} borderRadius={1}
bg="white" bg="white"
@ -95,8 +95,8 @@ export default function ProfileScreen(props: any) {
bg={sigilColor} bg={sigilColor}
borderRadius={8} borderRadius={8}
my={4} my={4}
height={128} height={160}
width={128} width={160}
display="flex" display="flex"
justifyContent="center" justifyContent="center"
alignItems="center" alignItems="center"

View File

@ -10,6 +10,7 @@ import CodeMirror from "codemirror";
import "codemirror/mode/markdown/markdown"; import "codemirror/mode/markdown/markdown";
import "codemirror/addon/display/placeholder"; import "codemirror/addon/display/placeholder";
import "codemirror/addon/edit/continuelist";
import "codemirror/lib/codemirror.css"; import "codemirror/lib/codemirror.css";
import { Box } from "@tlon/indigo-react"; import { Box } from "@tlon/indigo-react";
@ -54,6 +55,7 @@ export function MarkdownEditor(
scrollbarStyle: "native", scrollbarStyle: "native",
// cursorHeight: 0.85, // cursorHeight: 0.85,
placeholder: placeholder || "", placeholder: placeholder || "",
extraKeys: { 'Enter': 'newlineAndIndentContinueMarkdownList' }
}; };
const editor: React.RefObject<any> = useRef(); const editor: React.RefObject<any> = useRef();

View File

@ -47,7 +47,7 @@ export function Note(props: NoteProps & RouteComponentProps) {
const noteId = bigInt(index[1]); const noteId = bigInt(index[1]);
useEffect(() => { useEffect(() => {
api.hark.markEachAsRead(props.association, '/',`/${index[1]}/1/1`, 'note', 'publish'); api.hark.markEachAsRead(props.association, '/',`/${index[1]}/1/1`, 'note', 'publish');
}, [props.association]); }, [props.association, props.note]);
@ -67,7 +67,7 @@ export function Note(props: NoteProps & RouteComponentProps) {
color="red" color="red"
ml={2} ml={2}
onClick={deletePost} onClick={deletePost}
css={{ cursor: "pointer" }} style={{ cursor: "pointer" }}
> >
Delete Delete
</Text> </Text>
@ -75,6 +75,13 @@ export function Note(props: NoteProps & RouteComponentProps) {
); );
} }
const windowRef = React.useRef(null);
useEffect(() => {
if (windowRef.current) {
windowRef.current.parentElement.scrollTop = 0;
}
}, [windowRef, note]);
return ( return (
<Box <Box
my={3} my={3}
@ -86,6 +93,7 @@ export function Note(props: NoteProps & RouteComponentProps) {
width="100%" width="100%"
gridRowGap={4} gridRowGap={4}
mx="auto" mx="auto"
ref={windowRef}
> >
<Link to={rootUrl}> <Link to={rootUrl}>
<Text>{"<- Notebook Index"}</Text> <Text>{"<- Notebook Index"}</Text>

View File

@ -61,9 +61,9 @@ export function NotePreview(props: NotePreviewProps) {
overflow='hidden' overflow='hidden'
p='2' p='2'
> >
<WrappedBox mb={2}><Text bold fontSize='0'>{title}</Text></WrappedBox> <WrappedBox mb={2}><Text bold>{title}</Text></WrappedBox>
<WrappedBox> <WrappedBox>
<Text fontSize='14px'> <Text fontSize='14px' lineHeight='tall'>
<ReactMarkdown <ReactMarkdown
unwrapDisallowed unwrapDisallowed
allowedTypes={['text', 'root', 'break', 'paragraph', 'image']} allowedTypes={['text', 'root', 'break', 'paragraph', 'image']}

View File

@ -1,10 +1,9 @@
import React from "react"; import React from "react";
import { Link, RouteComponentProps } from "react-router-dom"; import { RouteComponentProps } from "react-router-dom";
import { NotebookPosts } from "./NotebookPosts"; import { NotebookPosts } from "./NotebookPosts";
import { Box, Button, Text, Row, Col } from "@tlon/indigo-react"; import { Col } from "@tlon/indigo-react";
import GlobalApi from "~/logic/api/global"; import GlobalApi from "~/logic/api/global";
import { Contacts, Rolodex, Groups, Associations, Graph, Association, Unreads } from "~/types"; import { Contacts, Rolodex, Groups, Associations, Graph, Association, Unreads } from "~/types";
import { useShowNickname } from "~/logic/lib/util";
interface NotebookProps { interface NotebookProps {
api: GlobalApi; api: GlobalApi;
@ -30,44 +29,14 @@ export function Notebook(props: NotebookProps & RouteComponentProps) {
association, association,
graph graph
} = props; } = props;
const { metadata } = association;
const group = groups[association?.['group-path']]; const group = groups[association?.['group-path']];
if (!group) { if (!group) {
return null; // Waiting on groups to populate return null; // Waiting on groups to populate
} }
const relativePath = (p: string) => props.baseUrl + p;
const contact = notebookContacts?.[ship];
const isOwn = `~${window.ship}` === ship;
let isWriter = true;
if (group.tags?.publish?.[`writers-${book}`]) {
isWriter = isOwn || group.tags?.publish?.[`writers-${book}`]?.has(window.ship);
}
const showNickname = useShowNickname(contact);
return ( return (
<Col gapY="4" pt={4} mx="auto" px={3} maxWidth="768px"> <Col gapY="4" pt={4} mx="auto" px={3} maxWidth="768px">
<Row justifyContent="space-between">
<Box>
<Text display='block'>{metadata?.title}</Text>
<Text color="lightGray">by </Text>
<Text fontFamily={showNickname ? 'sans' : 'mono'}>
{showNickname ? contact?.nickname : ship}
</Text>
</Box>
{isWriter && (
<Link to={relativePath('/new')}>
<Button primary style={{ cursor: 'pointer' }}>
New Post
</Button>
</Link>
)}
</Row>
<Box borderBottom="1" borderBottomColor="washedGray" />
<NotebookPosts <NotebookPosts
graph={graph} graph={graph}
host={ship} host={ship}

View File

@ -206,9 +206,6 @@
} }
@media all and (prefers-color-scheme: dark) { @media all and (prefers-color-scheme: dark) {
body {
background-color: #333;
}
.bg-black-d { .bg-black-d {
background-color: black; background-color: black;
} }

View File

@ -46,8 +46,8 @@ export default class TermApp extends Component {
render() { render() {
return ( return (
<> <>
<Helmet> <Helmet defer={false}>
<title>OS1 - Terminal</title> <title>{ this.props.notificationsCount ? `(${String(this.props.notificationsCount) }) `: '' }Landscape</title>
</Helmet> </Helmet>
<Box <Box
height='100%' height='100%'

View File

@ -52,13 +52,14 @@ export default function Author(props: AuthorProps) {
<Box <Box
ml={showImage ? 2 : 0} ml={showImage ? 2 : 0}
color="black" color="black"
fontSize='1'
lineHeight='tall' lineHeight='tall'
fontFamily={showNickname ? "sans" : "mono"} fontFamily={showNickname ? "sans" : "mono"}
fontWeight={showNickname ? '500' : '400'} fontWeight={showNickname ? '500' : '400'}
> >
{name} {name}
</Box> </Box>
<Box ml={2} color={props.unread ? "blue" : "gray"}> <Box fontSize='1' ml={2} color={props.unread ? "blue" : "gray"}>
{dateFmt} {dateFmt}
</Box> </Box>
{props.children} {props.children}

View File

@ -77,7 +77,7 @@ export function Comments(props: CommentsProps) {
if ('text' in curr) { if ('text' in curr) {
val = val + curr.text; val = val + curr.text;
} else if ('mention' in curr) { } else if ('mention' in curr) {
val = val + curr.mention; val = val + `~${curr.mention}`;
} else if ('url' in curr) { } else if ('url' in curr) {
val = val + curr.url; val = val + curr.url;
} else if ('code' in curr) { } else if ('code' in curr) {

View File

@ -21,6 +21,7 @@ interface DropdownProps {
alignY: AlignY | AlignY[]; alignY: AlignY | AlignY[];
alignX: AlignX | AlignX[]; alignX: AlignX | AlignX[];
width?: string; width?: string;
dropWidth?: string;
} }
const ClickBox = styled(Box)` const ClickBox = styled(Box)`
@ -111,14 +112,14 @@ export function Dropdown(props: DropdownProps) {
}); });
return ( return (
<Box flexShrink={1} position={open ? "relative" : "static"} minWidth='0'> <Box flexShrink={1} position={open ? "relative" : "static"} minWidth='0' width={props?.width ? props.width : 'auto'}>
<ClickBox width='100%' ref={anchorRef} onClick={onOpen}> <ClickBox width='100%' ref={anchorRef} onClick={onOpen}>
{children} {children}
</ClickBox> </ClickBox>
{open && ( {open && (
<Portal> <Portal>
<DropdownOptions <DropdownOptions
width={props.width || "max-content"} width={props?.dropWidth || "max-content"}
{...coords} {...coords}
ref={dropdownRef} ref={dropdownRef}
> >

View File

@ -19,28 +19,19 @@ interface MentionTextProps {
group: Group; group: Group;
} }
export function MentionText(props: MentionTextProps) { export function MentionText(props: MentionTextProps) {
const { content, contacts, contact, group } = props; const { content, contacts, contact, group, ...rest } = props;
return ( return (
<> <RichText contacts={contacts} contact={contact} group={group} {...rest}>
{_.map(content, (c, idx) => { {content.reduce((accum, c) => {
if ("text" in c) { if ("text" in c) {
return ( return accum + c.text;
<RichText
inline
key={idx}
>
{c.text}
</RichText>
);
} else if ("mention" in c) { } else if ("mention" in c) {
return ( return accum + `[~${c.mention}]`;
<Mention key={idx} contacts={contacts || {}} contact={contact || {}} group={group} ship={c.mention} />
);
} }
return null; return accum;
})} }, '')}
</> </RichText>
); );
} }
@ -53,7 +44,7 @@ export function Mention(props: {
const { contacts, ship } = props; const { contacts, ship } = props;
let { contact } = props; let { contact } = props;
contact = (contact?.nickname) ? contact : contacts?.[ship]; contact = (contact?.color) ? contact : contacts?.[ship];
const showNickname = useShowNickname(contact); const showNickname = useShowNickname(contact);

View File

@ -18,6 +18,7 @@ type ProfileOverlayProps = ColProps & {
group?: Group; group?: Group;
onDismiss(): void; onDismiss(): void;
hideAvatars: boolean; hideAvatars: boolean;
hideNicknames: boolean;
history: any; history: any;
api: any; api: any;
} }
@ -61,6 +62,7 @@ class ProfileOverlay extends PureComponent<ProfileOverlayProps, {}> {
bottomSpace, bottomSpace,
group = false, group = false,
hideAvatars, hideAvatars,
hideNicknames,
history, history,
onDismiss, onDismiss,
...rest ...rest
@ -89,7 +91,7 @@ class ProfileOverlay extends PureComponent<ProfileOverlayProps, {}> {
classes="brt2" classes="brt2"
svgClass="brt2" svgClass="brt2"
/>; />;
const showNickname = useShowNickname(contact); const showNickname = useShowNickname(contact, hideNicknames);
// TODO: we need to rethink this "top-level profile view" of other ships // TODO: we need to rethink this "top-level profile view" of other ships
/* if (!group.hidden) { /* if (!group.hidden) {
@ -127,7 +129,7 @@ class ProfileOverlay extends PureComponent<ProfileOverlayProps, {}> {
)} )}
<Text mono gray>{cite(`~${ship}`)}</Text> <Text mono gray>{cite(`~${ship}`)}</Text>
{!isOwn && ( {!isOwn && (
<Button mt={2} width="100%" style={{ cursor: 'pointer' }} onClick={() => history.push(`/~landscape/dm/${ship}`)}> <Button mt={2} fontSize='0' width="100%" style={{ cursor: 'pointer' }} onClick={() => history.push(`/~landscape/dm/${ship}`)}>
Send Message Send Message
</Button> </Button>
)} )}
@ -147,4 +149,4 @@ class ProfileOverlay extends PureComponent<ProfileOverlayProps, {}> {
} }
} }
export default withLocalState(ProfileOverlay, ['hideAvatars']); export default withLocalState(ProfileOverlay, ['hideAvatars', 'hideNicknames']);

View File

@ -8,14 +8,14 @@ const ReconnectButton = ({ connection, subscription }) => {
if (connectedStatus === "disconnected") { if (connectedStatus === "disconnected") {
return ( return (
<Button onClick={reconnect} borderColor='red' px='2'> <Button onClick={reconnect} borderColor='red' px='2'>
<Text display={['none', 'inline']} textAlign='middle' color='red'>Reconnect </Text> <Text display={['none', 'inline']} textAlign='middle' color='red'>Reconnect</Text>
<Text color='red'></Text> <Text color='red'> </Text>
</Button> </Button>
); );
} else if (connectedStatus === "reconnecting") { } else if (connectedStatus === "reconnecting") {
return ( return (
<Button borderColor='yellow' px='2' onClick={() => {}} cursor='default'> <Button borderColor='yellow' px='2' onClick={() => {}} cursor='default'>
<LoadingSpinner pr='2' foreground='scales.yellow60' background='scales.yellow30'/> <LoadingSpinner pr={['0','2']} foreground='scales.yellow60' background='scales.yellow30'/>
<Text display={['none', 'inline']} textAlign='middle' color='yellow'>Reconnecting</Text> <Text display={['none', 'inline']} textAlign='middle' color='yellow'>Reconnecting</Text>
</Button> </Button>
) )

View File

@ -3,8 +3,11 @@ import RemoteContent from '~/views/components/RemoteContent';
import { hasProvider } from 'oembed-parser'; import { hasProvider } from 'oembed-parser';
import ReactMarkdown from 'react-markdown'; import ReactMarkdown from 'react-markdown';
import RemarkDisableTokenizers from 'remark-disable-tokenizers'; import RemarkDisableTokenizers from 'remark-disable-tokenizers';
import { BaseAnchor, Text } from '@tlon/indigo-react'; import { BaseAnchor, Text } from '@tlon/indigo-react';
import { isValidPatp } from 'urbit-ob';
import { deSig } from '~/logic/lib/util';
import { Mention } from '~/views/components/MentionText';
const DISABLED_BLOCK_TOKENS = [ const DISABLED_BLOCK_TOKENS = [
'indentedCode', 'indentedCode',
@ -23,19 +26,25 @@ const RichText = React.memo(({ disableRemoteContent, ...props }) => (
<ReactMarkdown <ReactMarkdown
{...props} {...props}
renderers={{ renderers={{
link: (props) => { link: (linkProps) => {
if (disableRemoteContent) { const remoteContentPolicy = disableRemoteContent ? {
props.remoteContentPolicy = { imageShown: false,
imageShown: false, audioShown: false,
audioShown: false, videoShown: false,
videoShown: false, oembedShown: false
oembedShown: false } : null;
}; if (hasProvider(linkProps.href)) {
return <RemoteContent className="mw-100" url={linkProps.href} />;
} }
if (hasProvider(props.href)) {
return <RemoteContent className="mw-100" url={props.href} />; return <BaseAnchor target='_blank' rel='noreferrer noopener' borderBottom='1px solid' remoteContentPolicy={remoteContentPolicy} {...linkProps}>{linkProps.children}</BaseAnchor>;
},
linkReference: (linkProps) => {
const linkText = String(linkProps.children[0].props.children);
if (isValidPatp(linkText)) {
return <Mention contacts={props.contacts || {}} contact={props.contact || {}} group={props.group} ship={deSig(linkText)} />;
} }
return <BaseAnchor target='_blank' rel='noreferrer noopener' borderBottom='1px solid' {...props}>{props.children}</BaseAnchor>; return linkText;
}, },
paragraph: (paraProps) => { paragraph: (paraProps) => {
return <Text display={props.inline ? 'inline' : 'block'} mb='2' {...props}>{paraProps.children}</Text>; return <Text display={props.inline ? 'inline' : 'block'} mb='2' {...props}>{paraProps.children}</Text>;

View File

@ -64,6 +64,8 @@ export function ShipSearch(props: InviteSearchProps) {
if(valid) { if(valid) {
setInputShip(ship); setInputShip(ship);
setError(error === INVALID_SHIP_ERR ? undefined : error); setError(error === INVALID_SHIP_ERR ? undefined : error);
} else if (ship === undefined) {
return;
} else { } else {
setError(INVALID_SHIP_ERR); setError(INVALID_SHIP_ERR);
setInputTouched(false); setInputTouched(false);
@ -190,9 +192,9 @@ export function ShipSearch(props: InviteSearchProps) {
alignItems="center" alignItems="center"
py={1} py={1}
px={2} px={2}
border={1}
borderColor="washedGrey"
color="black" color="black"
borderRadius='2'
bg='washedGray'
fontSize={0} fontSize={0}
mt={2} mt={2}
mr={2} mr={2}

View File

@ -5,6 +5,7 @@ import ReconnectButton from './ReconnectButton';
import { StatusBarItem } from './StatusBarItem'; import { StatusBarItem } from './StatusBarItem';
import { Sigil } from '~/logic/lib/sigil'; import { Sigil } from '~/logic/lib/sigil';
import useLocalState from '~/logic/state/local'; import useLocalState from '~/logic/state/local';
import { cite } from '~/logic/lib/util';
const StatusBar = (props) => { const StatusBar = (props) => {
const invites = [].concat(...Object.values(props.invites).map(obj => Object.values(obj))); const invites = [].concat(...Object.values(props.invites).map(obj => Object.values(obj)));
@ -21,7 +22,7 @@ const StatusBar = (props) => {
pb='3' pb='3'
> >
<Row collapse> <Row collapse>
<Button borderColor='washedGray' mr='2' px='2' onClick={() => props.history.push('/')} {...props}> <Button width="32px" borderColor='washedGray' mr='2' px='2' onClick={() => props.history.push('/')} {...props}>
<Icon icon='Spaces' color='black'/> <Icon icon='Spaces' color='black'/>
</Button> </Button>
@ -54,14 +55,13 @@ const StatusBar = (props) => {
onClick={() => window.open( onClick={() => window.open(
'https://github.com/urbit/landscape/issues/new' + 'https://github.com/urbit/landscape/issues/new' +
'?assignees=&labels=development-stream&title=&' + '?assignees=&labels=development-stream&title=&' +
`body=commit:%20${process.env.LANDSCAPE_SHORTHASH}` `body=commit:%20urbit/urbit@${process.env.LANDSCAPE_SHORTHASH}`
)} )}
> >
<Text color='#000000'>Submit <Text color='#000000' display={['none', 'inline']}>an</Text> issue</Text> <Text color='#000000'>Submit <Text color='#000000' display={['none', 'inline']}>an</Text> issue</Text>
</StatusBarItem> </StatusBarItem>
<StatusBarItem px={'2'} flexShrink='0' onClick={() => props.history.push('/~profile')}> <StatusBarItem width={['32px', 'auto']} px={'2'} flexShrink='0' onClick={() => props.history.push('/~profile')}>
<Sigil ship={props.ship} size={16} color='black' classes='mix-blend-diff' icon /> <Sigil ship={props.ship} size={16} color='black' classes='mix-blend-diff' icon />
<Text ml={2} display={["none", "inline"]} fontFamily="mono">~{props.ship}</Text>
</StatusBarItem> </StatusBarItem>
</Row> </Row>
</Box> </Box>

View File

@ -234,10 +234,9 @@ export class Omnibox extends Component {
.filter(category => category.categoryResults.length > 0) .filter(category => category.categoryResults.length > 0)
.map(({ category, categoryResults }, i) => { .map(({ category, categoryResults }, i) => {
const categoryTitle = (category === 'other') const categoryTitle = (category === 'other')
? null : <Text gray ml={2}>{category.charAt(0).toUpperCase() + category.slice(1)}</Text>; ? null : <Row pl='2' height='5' alignItems='center' bg='washedGray'><Text gray bold>{category.charAt(0).toUpperCase() + category.slice(1)}</Text></Row>;
const selected = this.state.selected?.length ? this.state.selected[1] : ''; const selected = this.state.selected?.length ? this.state.selected[1] : '';
return (<Box key={i} width='max(50vw, 300px)' maxWidth='600px'> return (<Box key={i} width='max(50vw, 300px)' maxWidth='600px'>
<Rule borderTopWidth="0.5px" color="washedGray" />
{categoryTitle} {categoryTitle}
{categoryResults.map((result, i2) => ( {categoryResults.map((result, i2) => (
<OmniboxResult <OmniboxResult
@ -264,6 +263,7 @@ export class Omnibox extends Component {
if (state?.selected?.length === 0 && Array.from(this.state.results.values()).flat().length) { if (state?.selected?.length === 0 && Array.from(this.state.results.values()).flat().length) {
this.setNextSelected(); this.setNextSelected();
} }
return ( return (
<Box <Box
backgroundColor='scales.black30' backgroundColor='scales.black30'

View File

@ -22,7 +22,7 @@ export class OmniboxInput extends Component {
border='1px solid transparent' border='1px solid transparent'
borderRadius='2' borderRadius='2'
maxWidth='calc(600px - 1.15rem)' maxWidth='calc(600px - 1.15rem)'
fontSize='0' fontSize='1'
style={{ boxSizing: 'border-box' }} style={{ boxSizing: 'border-box' }}
placeholder='Search...' placeholder='Search...'
onKeyDown={props.control} onKeyDown={props.control}

View File

@ -37,7 +37,7 @@ export class OmniboxResult extends Component {
|| icon.toLowerCase() === 'links' || icon.toLowerCase() === 'links'
|| icon.toLowerCase() === 'terminal') || icon.toLowerCase() === 'terminal')
{ {
icon = (icon === 'Link') ? 'Links' : icon = (icon === 'Link') ? 'Collection' :
(icon === 'Terminal') ? 'Dojo' : icon; (icon === 'Terminal') ? 'Dojo' : icon;
graphic = <Icon display="inline-block" verticalAlign="middle" icon={icon} mr='2' size='16px' color={iconFill} />; graphic = <Icon display="inline-block" verticalAlign="middle" icon={icon} mr='2' size='16px' color={iconFill} />;
} else if (icon === 'inbox') { } else if (icon === 'inbox') {

View File

@ -48,7 +48,7 @@ export function ChannelMenu(props: ChannelMenuProps) {
const isOurs = ship.slice(1) === window.ship; const isOurs = ship.slice(1) === window.ship;
const isMuted = const isMuted =
props.graphNotificationConfig.watching.findIndex( props.graphNotificationConfig.watching.findIndex(
(a) => a.graph === appPath && a.index === "/" (a) => a.graph === appPath && a.index === "/"
) === -1; ) === -1;
@ -63,8 +63,10 @@ export function ChannelMenu(props: ChannelMenuProps) {
}, [api, association]); }, [api, association]);
const onDelete = useCallback(async () => { const onDelete = useCallback(async () => {
await api.graph.deleteGraph(name); if (confirm('Are you sure you want to delete this channel?')) {
history.push(`/~landscape${workspace}`); await api.graph.deleteGraph(name);
history.push(`/~landscape${workspace}`);
}
}, [api, association]); }, [api, association]);
return ( return (
@ -117,9 +119,9 @@ export function ChannelMenu(props: ChannelMenuProps) {
} }
alignX="right" alignX="right"
alignY="top" alignY="top"
width="250px" dropWidth="250px"
> >
<Icon display="block" icon="Menu" color="gray" /> <Icon display="block" icon="Menu" color="gray" pr='2' />
</Dropdown> </Dropdown>
); );
} }

View File

@ -1,21 +1,18 @@
import React from "react"; import React from 'react';
import { import {
Center,
Box, Box,
Col, Col,
Row, Row,
Text, Text,
IconButton, Icon
Button, } from '@tlon/indigo-react';
Icon, import { uxToHex } from '~/logic/lib/util';
} from "@tlon/indigo-react"; import { Link } from 'react-router-dom';
import { uxToHex } from "~/logic/lib/util";
import { Link } from "react-router-dom";
import { Association, Associations } from "~/types/metadata-update"; import { Associations } from '~/types/metadata-update';
import { Dropdown } from "~/views/components/Dropdown"; import { Dropdown } from '~/views/components/Dropdown';
import { Workspace } from "~/types"; import { Workspace } from '~/types';
import { getTitleFromWorkspace } from "~/logic/lib/workspace"; import { getTitleFromWorkspace } from '~/logic/lib/workspace';
const GroupSwitcherItem = ({ to, children, bottom = false, ...rest }) => ( const GroupSwitcherItem = ({ to, children, bottom = false, ...rest }) => (
<Link to={to}> <Link to={to}>
@ -47,7 +44,7 @@ function RecentGroups(props: { recent: string[]; associations: Associations }) {
return (e in associations?.contacts); return (e in associations?.contacts);
}).slice(1, 5).map((g) => { }).slice(1, 5).map((g) => {
const assoc = associations.contacts[g]; const assoc = associations.contacts[g];
const color = uxToHex(assoc?.metadata?.color || "0x0"); const color = uxToHex(assoc?.metadata?.color || '0x0');
return ( return (
<Link key={g} style={{ minWidth: 0 }} to={`/~landscape${g}`}> <Link key={g} style={{ minWidth: 0 }} to={`/~landscape${g}`}>
<Row px={1} pb={2} alignItems="center"> <Row px={1} pb={2} alignItems="center">
@ -60,7 +57,7 @@ function RecentGroups(props: { recent: string[]; associations: Associations }) {
bg={`#${color}`} bg={`#${color}`}
mr={2} mr={2}
display="block" display="block"
flexShrink='0' flexShrink={0}
/> />
<Text verticalAlign='top' maxWidth='100%' overflow='hidden' display='inline-block' style={{ textOverflow: 'ellipsis', whiteSpace: 'pre' }}>{assoc?.metadata?.title}</Text> <Text verticalAlign='top' maxWidth='100%' overflow='hidden' display='inline-block' style={{ textOverflow: 'ellipsis', whiteSpace: 'pre' }}>{assoc?.metadata?.title}</Text>
</Row> </Row>
@ -76,22 +73,20 @@ export function GroupSwitcher(props: {
workspace: Workspace; workspace: Workspace;
baseUrl: string; baseUrl: string;
recentGroups: string[]; recentGroups: string[];
isAdmin: any;
}) { }) {
const { associations, workspace, isAdmin } = props; const { associations, workspace, isAdmin } = props;
const title = getTitleFromWorkspace(associations, workspace); const title = getTitleFromWorkspace(associations, workspace);
const navTo = (to: string) => `${props.baseUrl}${to}`; const navTo = (to: string) => `${props.baseUrl}${to}`;
return ( return (
<Box zIndex="2" position="sticky" top="0px" p={2}> <Box height='48px' backgroundColor="white" zIndex="2" position="sticky" top="0px" py={3} pl='3' borderBottom='1px solid' borderColor='washedGray'>
<Col <Col
justifyContent="center"
bg="white" bg="white"
borderRadius={1}
border={1}
borderColor="washedGray"
> >
<Row alignItems="center" justifyContent="space-between"> <Row justifyContent="space-between">
<Dropdown <Dropdown
width="231px" width="100%"
dropWidth="231px"
alignY="top" alignY="top"
options={ options={
<Col <Col
@ -134,9 +129,9 @@ export function GroupSwitcher(props: {
<Icon mr="2" color="gray" icon="Plus" /> <Icon mr="2" color="gray" icon="Plus" />
<Text> Join Group</Text> <Text> Join Group</Text>
</GroupSwitcherItem> </GroupSwitcherItem>
{workspace.type === "group" && ( {workspace.type === 'group' && (
<> <>
<GroupSwitcherItem to={navTo("/popover/participants")}> <GroupSwitcherItem to={navTo('/popover/participants')}>
<Icon <Icon
mr={2} mr={2}
color="gray" color="gray"
@ -144,7 +139,7 @@ export function GroupSwitcher(props: {
/> />
<Text> Participants</Text> <Text> Participants</Text>
</GroupSwitcherItem> </GroupSwitcherItem>
<GroupSwitcherItem to={navTo("/popover/settings")}> <GroupSwitcherItem to={navTo('/popover/settings')}>
<Icon <Icon
mr={2} mr={2}
color="gray" color="gray"
@ -152,7 +147,7 @@ export function GroupSwitcher(props: {
/> />
<Text> Group Settings</Text> <Text> Group Settings</Text>
</GroupSwitcherItem> </GroupSwitcherItem>
{isAdmin && (<GroupSwitcherItem bottom to={navTo("/invites")}> {isAdmin && (<GroupSwitcherItem bottom to={navTo('/invites')}>
<Icon <Icon
mr={2} mr={2}
color="blue" color="blue"
@ -165,25 +160,25 @@ export function GroupSwitcher(props: {
</Col> </Col>
} }
> >
<Row p={2} alignItems="center" width='100%' minWidth='0'> <Row width='100%' minWidth='0' flexShrink={0}>
<Row alignItems="center" mr={1} flex='1' width='100%' minWidth='0'> <Row justifyContent="space-between" mr={1} flexShrink={0} width='100%' minWidth='0'>
<Text overflow='hidden' display='inline-block' flexShrink='1' style={{ textOverflow: 'ellipsis', whiteSpace: 'pre'}}>{title}</Text> <Text lineHeight="1.1" fontSize='2' fontWeight="700" overflow='hidden' display='inline-block' flexShrink='1' style={{ textOverflow: 'ellipsis', whiteSpace: 'pre' }}>{title}</Text>
<Icon size='12px' ml='1' mt="0px" display="inline-block" icon="ChevronSouth" />
</Row> </Row>
</Row> </Row>
</Dropdown> </Dropdown>
<Row pr={1} justifyContent="flex-end" alignItems="center"> <Row pr='3' verticalAlign="middle">
{(workspace.type === "group") && ( {(workspace.type === 'group') && (
<> <>
{isAdmin && (<Link to={navTo("/invites")}> {isAdmin && (<Link to={navTo('/invites')}>
<Icon <Icon
display="block" display="inline-block"
color='blue' color='blue'
icon="Users" icon="Users"
ml='12px'
/> />
</Link>)} </Link>)}
<Link to={navTo("/popover/settings")}> <Link to={navTo('/popover/settings')}>
<Icon color='gray' display="block" m={2} icon="Gear" /> <Icon color='gray' display="inline-block" ml={'12px'} icon="Gear" />
</Link> </Link>
</> </>
)} )}

View File

@ -7,6 +7,7 @@ import {
} from "react-router-dom"; } from "react-router-dom";
import { Col, Box, Text } from "@tlon/indigo-react"; import { Col, Box, Text } from "@tlon/indigo-react";
import _ from "lodash"; import _ from "lodash";
import Helmet from 'react-helmet';
import { Resource } from "./Resource"; import { Resource } from "./Resource";
import { PopoverRoutes } from "./PopoverRoutes"; import { PopoverRoutes } from "./PopoverRoutes";
@ -131,28 +132,36 @@ export function GroupsPane(props: GroupsPaneProps) {
const appPath = `/ship/${host}/${name}`; const appPath = `/ship/${host}/${name}`;
const association = associations.graph[appPath]; const association = associations.graph[appPath];
const resourceUrl = `${baseUrl}/join/${app}${appPath}`; const resourceUrl = `${baseUrl}/join/${app}${appPath}`;
let title = groupAssociation?.metadata?.title ?? 'Landscape';
if (!association) { if (!association) {
return <Loading />; return <Loading />;
} }
title += ` - ${association.metadata.title}`;
return ( return (
<Skeleton <>
recentGroups={recentGroups} <Helmet defer={false}>
mobileHide <title>{props.notificationsCount ? `(${String(props.notificationsCount)}) ` : ''}{ title }</title>
selected={appPath} </Helmet>
{...props} <Skeleton
baseUrl={baseUrl} recentGroups={recentGroups}
> mobileHide
<UnjoinedResource selected={appPath}
graphKeys={props.graphKeys} {...props}
notebooks={props.notebooks}
inbox={props.inbox}
baseUrl={baseUrl} baseUrl={baseUrl}
api={api} >
association={association} <UnjoinedResource
/> graphKeys={props.graphKeys}
{popovers(routeProps, resourceUrl)} notebooks={props.notebooks}
</Skeleton> inbox={props.inbox}
baseUrl={baseUrl}
api={api}
association={association}
/>
{popovers(routeProps, resourceUrl)}
</Skeleton>
</>
); );
}} }}
/> />
@ -184,20 +193,26 @@ export function GroupsPane(props: GroupsPaneProps) {
const hasDescription = groupAssociation?.metadata?.description; const hasDescription = groupAssociation?.metadata?.description;
const description = (hasDescription && hasDescription !== "") const description = (hasDescription && hasDescription !== "")
? hasDescription : "Create or select a channel to get started" ? hasDescription : "Create or select a channel to get started"
const title = groupAssociation?.metadata?.title ?? 'Landscape';
return ( return (
<Skeleton recentGroups={recentGroups} {...props} baseUrl={baseUrl}> <>
<Col <Helmet defer={false}>
alignItems="center" <title>{props.notificationsCount ? `(${String(props.notificationsCount)}) ` : ''}{ title }</title>
justifyContent="center" </Helmet>
display={["none", "flex"]} <Skeleton recentGroups={recentGroups} {...props} baseUrl={baseUrl}>
p='4' <Col
> alignItems="center"
<Box p="4"><Text fontSize="0" color='gray'> justifyContent="center"
{description} display={["none", "flex"]}
</Text></Box> p='4'
</Col> >
{popovers(routeProps, baseUrl)} <Box p="4"><Text fontSize="0" color='gray'>
</Skeleton> {description}
</Text></Box>
</Col>
{popovers(routeProps, baseUrl)}
</Skeleton>
</>
); );
}} }}
/> />

View File

@ -85,7 +85,7 @@ export function NewChannel(props: NewChannelProps & RouteComponentProps) {
moduleType moduleType
); );
} }
if (!group) { if (!group) {
await waiter(p => Boolean(p?.groups?.[`/ship/~${window.ship}/${resId}`])); await waiter(p => Boolean(p?.groups?.[`/ship/~${window.ship}/${resId}`]));
} }
@ -99,11 +99,11 @@ export function NewChannel(props: NewChannelProps & RouteComponentProps) {
actions.setStatus({ error: 'Channel creation failed' }); actions.setStatus({ error: 'Channel creation failed' });
} }
}; };
return ( return (
<Col overflowY="auto" p={3}> <Col overflowY="auto" p={3}>
<Box pb='3' display={['block', 'none']} onClick={() => history.push(props.baseUrl)}> <Box pb='3' display={['block', 'none']} onClick={() => history.push(props.baseUrl)}>
{'<- Back'} <Text fontSize='0' bold>{'<- Back'}</Text>
</Box> </Box>
<Box fontWeight="bold" mb={4} color="black"> <Box fontWeight="bold" mb={4} color="black">
New Channel New Channel

View File

@ -105,7 +105,7 @@ export function NewGroup(props: NewGroupProps & RouteComponentProps) {
<Checkbox <Checkbox
id="isPrivate" id="isPrivate"
label="Private Group" label="Private Group"
caption="Is your group private?" caption="Anyone can join a public group. A private group is only joinable by invite."
/> />
<AsyncButton>Create Group</AsyncButton> <AsyncButton>Create Group</AsyncButton>
</Col> </Col>

View File

@ -80,7 +80,7 @@ export function PopoverRoutes(
<Box <Box
display="grid" display="grid"
gridTemplateRows={["32px 1fr", "100%"]} gridTemplateRows={["32px 1fr", "100%"]}
gridTemplateColumns={["100%", "150px 1fr"]} gridTemplateColumns={["100%", "250px 1fr"]}
height="100%" height="100%"
width="100%" width="100%"
> >

View File

@ -1,7 +1,7 @@
import React, { useCallback } from "react"; import React, { useCallback } from "react";
import { Row, Box, Col } from "@tlon/indigo-react"; import { Row, Box, Col } from "@tlon/indigo-react";
import styled from "styled-components"; import styled from "styled-components";
import { Link } from "react-router-dom"; import Helmet from 'react-helmet';
import { ChatResource } from "~/views/apps/chat/ChatResource"; import { ChatResource } from "~/views/apps/chat/ChatResource";
import { PublishResource } from "~/views/apps/publish/PublishResource"; import { PublishResource } from "~/views/apps/publish/PublishResource";
@ -34,47 +34,60 @@ export function Resource(props: ResourceProps) {
const relativePath = (p: string) => const relativePath = (p: string) =>
`${props.baseUrl}/resource/${app}${appPath}${p}`; `${props.baseUrl}/resource/${app}${appPath}${p}`;
const skelProps = { api, association }; const skelProps = { api, association };
let title = props.association.metadata.title;
if ('workspace' in props) {
if ('group' in props.workspace && props.workspace.group in props.associations.contacts) {
title = `${props.associations.contacts[props.workspace.group].metadata.title} - ${props.association.metadata.title}`;
}
}
return ( return (
<Switch> <>
<Route <Helmet defer={false}>
path={relativePath("/settings")} <title>{props.notificationsCount ? `(${String(props.notificationsCount)}) ` : ''}{ title }</title>
render={(routeProps) => { </Helmet>
return ( <Switch>
<Route
path={relativePath("/settings")}
render={(routeProps) => {
return (
<ResourceSkeleton
baseUrl={props.baseUrl}
groupTags={props.groups?.[selectedGroup]?.tags}
{...skelProps}
>
<ChannelSettings
groups={props.groups}
contacts={props.contacts}
associations={props.associations}
api={api}
association={association}
/>
</ResourceSkeleton>
);
}}
/>
<Route
path={relativePath("")}
render={(routeProps) => (
<ResourceSkeleton <ResourceSkeleton
notificationsGraphConfig={props.notificationsGraphConfig}
notificationsChatConfig={props.notificationsChatConfig}
baseUrl={props.baseUrl} baseUrl={props.baseUrl}
groupTags={props.groups?.[selectedGroup]?.tags}
{...skelProps} {...skelProps}
atRoot
> >
<ChannelSettings {app === "chat" ? (
groups={props.groups} <ChatResource {...props} />
contacts={props.contacts} ) : app === "publish" ? (
associations={props.associations} <PublishResource {...props} />
api={api} ) : (
association={association} <LinkResource {...props} />
/> )}
</ResourceSkeleton> </ResourceSkeleton>
); )}
}} />
/> </Switch>
<Route </>
path={relativePath("")}
render={(routeProps) => (
<ResourceSkeleton
notificationsGraphConfig={props.notificationsGraphConfig}
notificationsChatConfig={props.notificationsChatConfig}
baseUrl={props.baseUrl}
{...skelProps}
atRoot
>
{app === "chat" ? (
<ChatResource {...props} />
) : app === "publish" ? (
<PublishResource {...props} />
) : (
<LinkResource {...props} />
)}
</ResourceSkeleton>
)}
/>
</Switch>
); );
} }

View File

@ -16,7 +16,7 @@ import { ChannelMenu } from "./ChannelMenu";
import { NotificationGraphConfig } from "~/types"; import { NotificationGraphConfig } from "~/types";
const TruncatedBox = styled(Box)` const TruncatedBox = styled(Box)`
white-space: nowrap; white-space: pre;
text-overflow: ellipsis; text-overflow: ellipsis;
overflow: hidden; overflow: hidden;
`; `;
@ -29,19 +29,33 @@ type ResourceSkeletonProps = {
children: ReactNode; children: ReactNode;
atRoot?: boolean; atRoot?: boolean;
title?: string; title?: string;
groupTags?: any;
}; };
export function ResourceSkeleton(props: ResourceSkeletonProps) { export function ResourceSkeleton(props: ResourceSkeletonProps) {
const { association, api, baseUrl, children, atRoot } = props; const { association, api, baseUrl, children, atRoot, groupTags } = props;
const app = association?.metadata?.module || association["app-name"]; const app = association?.metadata?.module || association["app-name"];
const appPath = association["app-path"]; const appPath = association["app-path"];
const workspace = const workspace =
baseUrl === "/~landscape/home" ? "/home" : association["group-path"]; baseUrl === "/~landscape/home" ? "/home" : association["group-path"];
const title = props.title || association?.metadata?.title; const title = props.title || association?.metadata?.title;
const [, , ship, resource] = appPath.split("/");
const resourcePath = (p: string) => baseUrl + `/resource/${app}/ship/${ship}/${resource}` + p;
const isOwn = `~${window.ship}` === ship;
let isWriter = (app === 'publish') ? true : false;
if (groupTags?.publish?.[`writers-${resource}`]) {
isWriter = isOwn || groupTags?.publish?.[`writers-${resource}`]?.has(window.ship);
}
return ( return (
<Col width="100%" height="100%" overflowY="hidden"> <Col width="100%" height="100%" overflowY="hidden">
<Box <Box
flexShrink="0" flexShrink="0"
height='48px'
py="2" py="2"
px="2" px="2"
display="flex" display="flex"
@ -54,6 +68,7 @@ export function ResourceSkeleton(props: ResourceSkeletonProps) {
borderRight={1} borderRight={1}
borderRightColor="gray" borderRightColor="gray"
pr={3} pr={3}
fontSize='1'
mr={3} mr={3}
my="1" my="1"
display={["block", "none"]} display={["block", "none"]}
@ -71,14 +86,14 @@ export function ResourceSkeleton(props: ResourceSkeletonProps) {
{atRoot && ( {atRoot && (
<> <>
<Box pr={1} mr={2}> <Box pr={1} mr={2}>
<Text display="inline-block" verticalAlign="middle"> <Text fontSize='2' fontWeight='700' display="inline-block" verticalAlign="middle" textOverflow="ellipsis" overflow="hidden" whiteSpace="pre">
{title} {title}
</Text> </Text>
</Box> </Box>
<TruncatedBox <TruncatedBox
display={["none", "block"]} display={["none", "block"]}
maxWidth="60%"
verticalAlign="middle" verticalAlign="middle"
maxWidth='60%'
flexShrink={1} flexShrink={1}
title={association?.metadata?.description} title={association?.metadata?.description}
color="gray" color="gray"
@ -93,6 +108,11 @@ export function ResourceSkeleton(props: ResourceSkeletonProps) {
</RichText> </RichText>
</TruncatedBox> </TruncatedBox>
<Box flexGrow={1} /> <Box flexGrow={1} />
{isWriter && (
<Link to={resourcePath('/new')} style={{ flexShrink: '0' }}>
<Text bold pr='3' color='blue'>+ New Post</Text>
</Link>
)}
<ChannelMenu <ChannelMenu
graphNotificationConfig={props.notificationsGraphConfig} graphNotificationConfig={props.notificationsGraphConfig}
chatNotificationConfig={props.notificationsChatConfig} chatNotificationConfig={props.notificationsChatConfig}

View File

@ -46,16 +46,17 @@ export function useGraphModule(
): SidebarAppConfig { ): SidebarAppConfig {
const getStatus = useCallback( const getStatus = useCallback(
(s: string) => { (s: string) => {
const unreads = graphUnreads?.[s]?.['/']?.unreads;
if(typeof unreads === 'number' ? unreads > 0 : unreads?.size ?? 0 > 0) {
return 'unread';
}
const [, , host, name] = s.split("/"); const [, , host, name] = s.split("/");
const graphKey = `${host.slice(1)}/${name}`; const graphKey = `${host.slice(1)}/${name}`;
if (!graphKeys.has(graphKey)) { if (!graphKeys.has(graphKey)) {
return "unsubscribed"; return "unsubscribed";
} }
const unreads = graphUnreads?.[s]?.['/']?.unreads;
if (typeof unreads === 'number' ? unreads > 0 : unreads?.size ?? 0 > 0) {
return 'unread';
}
return undefined; return undefined;
}, },
[graphs, graphKeys, graphUnreads] [graphs, graphKeys, graphUnreads]

View File

@ -1,26 +1,24 @@
import React, { ReactNode } from "react"; import React, { ReactNode } from 'react';
import styled from 'styled-components'; import styled from 'styled-components';
import { import {
Box, Col
Col, } from '@tlon/indigo-react';
} from "@tlon/indigo-react";
import { Link } from "react-router-dom";
import GlobalApi from "~/logic/api/global"; import GlobalApi from '~/logic/api/global';
import { GroupSwitcher } from "../GroupSwitcher"; import { GroupSwitcher } from '../GroupSwitcher';
import { import {
Associations, Associations,
Workspace, Workspace,
Groups, Groups,
Invites, Invites,
Rolodex, Rolodex
} from "~/types"; } from '~/types';
import { SidebarListHeader } from "./SidebarListHeader"; import { SidebarListHeader } from './SidebarListHeader';
import { useLocalStorageState } from "~/logic/lib/useLocalStorageState"; import { useLocalStorageState } from '~/logic/lib/useLocalStorageState';
import { getGroupFromWorkspace } from "~/logic/lib/workspace"; import { getGroupFromWorkspace } from '~/logic/lib/workspace';
import { SidebarAppConfigs } from './types'; import { SidebarAppConfigs } from './types';
import { SidebarList } from "./SidebarList"; import { SidebarList } from './SidebarList';
import { roleForShip } from "~/logic/lib/group"; import { roleForShip } from '~/logic/lib/group';
const ScrollbarLessCol = styled(Col)` const ScrollbarLessCol = styled(Col)`
scrollbar-width: none !important; scrollbar-width: none !important;
@ -30,7 +28,6 @@ const ScrollbarLessCol = styled(Col)`
} }
`; `;
interface SidebarProps { interface SidebarProps {
contacts: Rolodex; contacts: Rolodex;
children: ReactNode; children: ReactNode;
@ -48,38 +45,24 @@ interface SidebarProps {
workspace: Workspace; workspace: Workspace;
} }
// Magic spacer that because firefox doesn't correctly calculate
// position: sticky on a flex child
// remove when https://bugzilla.mozilla.org/show_bug.cgi?id=1488080
// is fixed
const SidebarStickySpacer = styled(Box)`
height: 0px;
flex-grow: 1;
@-moz-document url-prefix() {
& {
height: ${p => p.theme.space[6] }px;
}
}
`;
export function Sidebar(props: SidebarProps) { export function Sidebar(props: SidebarProps) {
const { invites, api, associations, selected, apps, workspace } = props; const { associations, selected, workspace } = props;
const groupPath = getGroupFromWorkspace(workspace); const groupPath = getGroupFromWorkspace(workspace);
const display = props.mobileHide ? ["none", "flex"] : "flex"; const display = props.mobileHide ? ['none', 'flex'] : 'flex';
if (!associations) { if (!associations) {
return null; return null;
} }
const [config, setConfig] = useLocalStorageState<SidebarListConfig>( const [config, setConfig] = useLocalStorageState<SidebarListConfig>(
`group-config:${groupPath || "home"}`, `group-config:${groupPath || 'home'}`,
{ {
sortBy: "lastUpdated", sortBy: 'lastUpdated',
hideUnjoined: false, hideUnjoined: false
} }
); );
const role = props.groups?.[groupPath] ? roleForShip(props.groups[groupPath], window.ship) : undefined; const role = props.groups?.[groupPath] ? roleForShip(props.groups[groupPath], window.ship) : undefined;
const isAdmin = (role === "admin") || (workspace?.type === 'home'); const isAdmin = (role === 'admin') || (workspace?.type === 'home');
return ( return (
<ScrollbarLessCol <ScrollbarLessCol
@ -107,8 +90,9 @@ export function Sidebar(props: SidebarProps) {
groups={props.groups} groups={props.groups}
initialValues={config} initialValues={config}
handleSubmit={setConfig} handleSubmit={setConfig}
selected={selected || ""} selected={selected || ''}
workspace={workspace} /> workspace={workspace}
/>
<SidebarList <SidebarList
config={config} config={config}
associations={associations} associations={associations}
@ -118,31 +102,6 @@ export function Sidebar(props: SidebarProps) {
apps={props.apps} apps={props.apps}
baseUrl={props.baseUrl} baseUrl={props.baseUrl}
/> />
<SidebarStickySpacer flexShrink={0} />
<Box
flexShrink="0"
display={isAdmin ? "flex" : "none"}
justifyContent="center"
position="sticky"
bottom={"8px"}
width="100%"
height="fit-content"
py="2"
>
<Link
to={!!groupPath ? `/~landscape${groupPath}/new` : `/~landscape/home/new`}
>
<Box
bg="white"
p={2}
borderRadius={1}
border={1}
borderColor="lightGray"
>
+ New Channel
</Box>
</Link>
</Box>
</ScrollbarLessCol> </ScrollbarLessCol>
); );
} }

View File

@ -27,7 +27,7 @@ function SidebarItemIndicator(props: { status?: SidebarItemStatus }) {
const getAppIcon = (app: string, mod: string) => { const getAppIcon = (app: string, mod: string) => {
if (app === "graph") { if (app === "graph") {
if (mod === "link") { if (mod === "link") {
return "Links"; return "Collection";
} }
return _.capitalize(mod); return _.capitalize(mod);
} }
@ -93,8 +93,8 @@ export function SidebarItem(props: {
justifyContent="space-between" justifyContent="space-between"
alignItems="center" alignItems="center"
py={1} py={1}
pl={4} pl={3}
pr={2} pr={3}
selected={selected} selected={selected}
> >
<Row width='100%' alignItems="center" flex='1 auto' minWidth='0'> <Row width='100%' alignItems="center" flex='1 auto' minWidth='0'>
@ -105,7 +105,7 @@ export function SidebarItem(props: {
/> />
<Box width='100%' flexShrink={2} ml={2} display='flex' overflow='hidden'> <Box width='100%' flexShrink={2} ml={2} display='flex' overflow='hidden'>
<Text <Text
lineHeight="short" lineHeight="tall"
display='inline-block' display='inline-block'
flex='1' flex='1'
overflow='hidden' overflow='hidden'

View File

@ -14,7 +14,8 @@ import { Dropdown } from "~/views/components/Dropdown";
import { FormikHelpers } from "formik"; import { FormikHelpers } from "formik";
import { SidebarListConfig, Workspace } from "./types"; import { SidebarListConfig, Workspace } from "./types";
import { Link, useHistory } from 'react-router-dom'; import { Link, useHistory } from 'react-router-dom';
import {ShipSearch} from "~/views/components/ShipSearch"; import { getGroupFromWorkspace } from "~/logic/lib/workspace";
import { roleForShip } from "~/logic/lib/group";
import {Groups, Rolodex} from "~/types"; import {Groups, Rolodex} from "~/types";
export function SidebarListHeader(props: { export function SidebarListHeader(props: {
@ -36,14 +37,18 @@ export function SidebarListHeader(props: {
[props.handleSubmit] [props.handleSubmit]
); );
const groupPath = getGroupFromWorkspace(props.workspace);
const role = props.groups?.[groupPath] ? roleForShip(props.groups[groupPath], window.ship) : undefined;
const isAdmin = (role === "admin") || (props.workspace?.type === 'home');
return ( return (
<Row <Row
flexShrink="0" flexShrink="0"
alignItems="center" alignItems="center"
justifyContent="space-between" justifyContent="space-between"
py={2} py={2}
pr={2} px={3}
pl={3} height='48px'
> >
<Box flexShrink='0'> <Box flexShrink='0'>
<Text> <Text>
@ -51,27 +56,32 @@ export function SidebarListHeader(props: {
</Text> </Text>
</Box> </Box>
<Box <Box
width='100%'
textAlign='right' textAlign='right'
mr='2' display='flex'
display={(props.workspace?.type === 'home') ? 'inline-block' : 'none'}
> >
<Link to={`${props.baseUrl}/invites`}> <Link
style={{
display: isAdmin ? "inline-block" : "none" }}
to={
!!groupPath ? `/~landscape${groupPath}/new` : `/~landscape/home/new`}>
<Icon icon="Plus" color="gray" pr='2'/>
</Link>
<Link to={`${props.baseUrl}/invites`}
style={{ display: (props.workspace?.type === 'home') ? 'inline-block' : 'none'}}>
<Text <Text
display='inline-block' display='inline-block'
verticalAlign='middle'
py='1px' py='1px'
px='3px' px='3px'
mr='2'
backgroundColor='washedBlue' backgroundColor='washedBlue'
color='blue' color='blue'
borderRadius='1'> borderRadius='1'>
+ DM + DM
</Text> </Text>
</Link> </Link>
</Box>
<Dropdown <Dropdown
flexShrink='0' flexShrink='0'
width="200px" width="auto"
alignY="top" alignY="top"
alignX={["right", "left"]} alignX={["right", "left"]}
options={ options={
@ -102,6 +112,7 @@ export function SidebarListHeader(props: {
> >
<Icon color="gray" icon="Adjust" /> <Icon color="gray" icon="Adjust" />
</Dropdown> </Dropdown>
</Box>
</Row> </Row>
); );
} }

View File

@ -1,5 +1,6 @@
import React, { Component, useEffect, useCallback } from 'react'; import React, { Component, useEffect, useCallback } from 'react';
import { Route, Switch, RouteComponentProps } from 'react-router-dom'; import { Route, Switch, RouteComponentProps } from 'react-router-dom';
import Helmet from 'react-helmet';
import './css/custom.css'; import './css/custom.css';
@ -67,8 +68,6 @@ export function DMRedirect(props: LandscapeProps & RouteComponentProps & { ship:
export default class Landscape extends Component<LandscapeProps, {}> { export default class Landscape extends Component<LandscapeProps, {}> {
componentDidMount() { componentDidMount() {
document.title = 'OS1 - Landscape';
this.props.subscription.startApp('groups'); this.props.subscription.startApp('groups');
this.props.subscription.startApp('graph'); this.props.subscription.startApp('graph');
} }
@ -78,71 +77,76 @@ export default class Landscape extends Component<LandscapeProps, {}> {
const { api } = props; const { api } = props;
return ( return (
<Switch> <>
<Route path="/~landscape/ship/:host/:name" <Helmet defer={false}>
render={routeProps => { <title>{ props.notificationsCount ? `(${String(props.notificationsCount) }) `: '' }Landscape</title>
const { </Helmet>
host, <Switch>
name <Route path="/~landscape/ship/:host/:name"
} = routeProps.match.params as Record<string, string>; render={routeProps => {
const groupPath = `/ship/${host}/${name}`; const {
const baseUrl = `/~landscape${groupPath}`; host,
const ws: Workspace = { type: 'group', group: groupPath }; name
} = routeProps.match.params as Record<string, string>;
const groupPath = `/ship/${host}/${name}`;
const baseUrl = `/~landscape${groupPath}`;
const ws: Workspace = { type: 'group', group: groupPath };
return ( return (
<GroupsPane workspace={ws} baseUrl={baseUrl} {...props} /> <GroupsPane workspace={ws} baseUrl={baseUrl} {...props} />
) )
}}/> }}/>
<Route path="/~landscape/home" <Route path="/~landscape/home"
render={routeProps => {
const ws: Workspace = { type: 'home' };
return (
<GroupsPane workspace={ws} baseUrl="/~landscape/home" {...props} />
);
}}
/>
<Route path="/~landscape/new"
render={routeProps=> {
return (
<Body>
<Box maxWidth="300px">
<NewGroup
groups={props.groups}
contacts={props.contacts}
api={props.api}
{...routeProps}
/>
</Box>
</Body>
);
}}
/>
<Route path='/~landscape/dm/:ship?'
render={routeProps => { render={routeProps => {
const ws: Workspace = { type: 'home' }; const { ship } = routeProps.match.params;
return ( return <DMRedirect {...routeProps} {...props} ship={ship} />
<GroupsPane workspace={ws} baseUrl="/~landscape/home" {...props} />
);
}} }}
/> />
<Route path="/~landscape/new" <Route path="/~landscape/join/:ship?/:name?"
render={routeProps=> { render={routeProps=> {
return ( const { ship, name } = routeProps.match.params;
<Body> const autojoin = ship && name ? `${ship}/${name}` : null;
<Box maxWidth="300px"> return (
<NewGroup <Body>
groups={props.groups} <Box maxWidth="300px">
contacts={props.contacts} <JoinGroup
api={props.api} groups={props.groups}
{...routeProps} contacts={props.contacts}
/> api={props.api}
</Box> autojoin={autojoin}
</Body> {...routeProps}
); />
}} </Box>
/> </Body>
<Route path='/~landscape/dm/:ship?' );
render={routeProps => { }}
const { ship } = routeProps.match.params; />
return <DMRedirect {...routeProps} {...props} ship={ship} /> </Switch>
}} </>
/>
<Route path="/~landscape/join/:ship?/:name?"
render={routeProps=> {
const { ship, name } = routeProps.match.params;
const autojoin = ship && name ? `${ship}/${name}` : null;
return (
<Body>
<Box maxWidth="300px">
<JoinGroup
groups={props.groups}
contacts={props.contacts}
api={props.api}
autojoin={autojoin}
{...routeProps}
/>
</Box>
</Body>
);
}}
/>
</Switch>
); );
} }
} }

View File

@ -1,171 +0,0 @@
import baseStyled, { ThemedStyledInterface } from "styled-components";
const base = {
white: "rgba(255,255,255,1)",
black: "rgba(0,0,0,1)",
red: "rgba(255,65,54,1)",
yellow: "rgba(255,199,0,1)",
green: "rgba(0,159,101,1)",
blue: "rgba(0,142,255,1)",
};
const scales = {
white10: "rgba(255,255,255,0.1)",
white20: "rgba(255,255,255,0.2)",
white30: "rgba(255,255,255,0.3)",
white40: "rgba(255,255,255,0.4)",
white50: "rgba(255,255,255,0.5)",
white60: "rgba(255,255,255,0.6)",
white70: "rgba(255,255,255,0.7)",
white80: "rgba(255,255,255,0.8)",
white90: "rgba(255,255,255,0.9)",
white100: "rgba(255,255,255,1)",
black05: "rgba(0,0,0,0.05)",
black10: "rgba(0,0,0,0.1)",
black20: "rgba(0,0,0,0.2)",
black30: "rgba(0,0,0,0.3)",
black40: "rgba(0,0,0,0.4)",
black50: "rgba(0,0,0,0.5)",
black60: "rgba(0,0,0,0.6)",
black70: "rgba(0,0,0,0.7)",
black80: "rgba(0,0,0,0.8)",
black90: "rgba(0,0,0,0.9)",
black100: "rgba(0,0,0,1)",
red10: "rgba(255,65,54,0.1)",
red20: "rgba(255,65,54,0.2)",
red30: "rgba(255,65,54,0.3)",
red40: "rgba(255,65,54,0.4)",
red50: "rgba(255,65,54,0.5)",
red60: "rgba(255,65,54,0.6)",
red70: "rgba(255,65,54,0.7)",
red80: "rgba(255,65,54,0.8)",
red90: "rgba(255,65,54,0.9)",
red100: "rgba(255,65,54,1)",
yellow10: "rgba(255,199,0,0.1)",
yellow20: "rgba(255,199,0,0.2)",
yellow30: "rgba(255,199,0,0.3)",
yellow40: "rgba(255,199,0,0.4)",
yellow50: "rgba(255,199,0,0.5)",
yellow60: "rgba(255,199,0,0.6)",
yellow70: "rgba(255,199,0,0.7)",
yellow80: "rgba(255,199,0,0.8)",
yellow90: "rgba(255,199,0,0.9)",
yellow100: "rgba(255,199,0,1)",
green10: "rgba(0,159,101,0.1)",
green20: "rgba(0,159,101,0.2)",
green30: "rgba(0,159,101,0.3)",
green40: "rgba(0,159,101,0.4)",
green50: "rgba(0,159,101,0.5)",
green60: "rgba(0,159,101,0.6)",
green70: "rgba(0,159,101,0.7)",
green80: "rgba(0,159,101,0.8)",
green90: "rgba(0,159,101,0.9)",
green100: "rgba(0,159,101,1)",
blue10: "rgba(0,142,255,0.1)",
blue20: "rgba(0,142,255,0.2)",
blue30: "rgba(0,142,255,0.3)",
blue40: "rgba(0,142,255,0.4)",
blue50: "rgba(0,142,255,0.5)",
blue60: "rgba(0,142,255,0.6)",
blue70: "rgba(0,142,255,0.7)",
blue80: "rgba(0,142,255,0.8)",
blue90: "rgba(0,142,255,0.9)",
blue100: "rgba(0,142,255,1)",
};
const theme = {
colors: {
white: base.white,
black: base.black,
darkGray: scales.black80,
gray: scales.black60,
lightGray: scales.black30,
washedGray: scales.black10,
red: base.red,
lightRed: scales.red30,
washedRed: scales.red10,
yellow: base.yellow,
lightYellow: scales.yellow30,
washedYellow: scales.yellow10,
green: base.green,
lightGreen: scales.green30,
washedGreen: scales.green10,
blue: base.blue,
lightBlue: scales.blue30,
washedBlue: scales.blue10,
none: "rgba(0,0,0,0)",
scales: scales,
},
fonts: {
sans: `"Inter", "Inter UI", -apple-system, BlinkMacSystemFont, 'San Francisco', 'Helvetica Neue', Arial, sans-serif`,
mono: `"Source Code Pro", "Roboto mono", "Courier New", monospace`,
},
// font-size
fontSizes: [
12, // 0
16, // 1
24, // 2
32, // 3
48, // 4
64, // 5
],
// font-weight
fontWeights: {
thin: 300,
regular: 400,
bold: 600,
},
// line-height
lineHeights: {
min: 1.2,
short: 1.333333,
regular: 1.5,
tall: 1.666666,
},
// border, border-top, border-right, border-bottom, border-left
borders: ["none", "1px solid"],
// margin, margin-top, margin-right, margin-bottom, margin-left, padding, padding-top, padding-right, padding-bottom, padding-left, grid-gap, grid-column-gap, grid-row-gap
space: [
0, // 0
4, // 1
8, // 2
16, // 3
24, // 4
32, // 5
48, // 6
64, // 7
96, // 8
],
// border-radius
radii: [
0, // 0
2, // 1
4, // 2
8, // 3
16, // 4
],
// width, height, min-width, max-width, min-height, max-height
sizes: [
0, // 0
4, // 1
8, // 2
16, // 3
24, // 4
32, // 5
48, // 6
64, // 7
96, // 8
],
// z-index
zIndices: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
breakpoints: ["550px", "750px", "960px"],
};
export type Theme = typeof theme;
export const styled = baseStyled as ThemedStyledInterface<Theme>;
export default theme;

View File

@ -1,186 +0,0 @@
import baseStyled, { ThemedStyledInterface } from "styled-components";
const base = {
white: "rgba(255,255,255,1)",
black: "rgba(0,0,0,1)",
red: "rgba(255,65,54,1)",
yellow: "rgba(255,199,0,1)",
green: "rgba(0,159,101,1)",
blue: "rgba(0,142,255,1)",
};
const scales = {
white05: "rgba(255,255,255,0.05)",
white10: "rgba(255,255,255,0.1)",
white20: "rgba(255,255,255,0.2)",
white30: "rgba(255,255,255,0.3)",
white40: "rgba(255,255,255,0.4)",
white50: "rgba(255,255,255,0.5)",
white60: "rgba(255,255,255,0.6)",
white70: "rgba(255,255,255,0.7)",
white80: "rgba(255,255,255,0.8)",
white90: "rgba(255,255,255,0.9)",
white100: "rgba(255,255,255,1)",
black05: "rgba(0,0,0,0.05)",
black10: "rgba(0,0,0,0.1)",
black20: "rgba(0,0,0,0.2)",
black30: "rgba(0,0,0,0.3)",
black40: "rgba(0,0,0,0.4)",
black50: "rgba(0,0,0,0.5)",
black60: "rgba(0,0,0,0.6)",
black70: "rgba(0,0,0,0.7)",
black80: "rgba(0,0,0,0.8)",
black90: "rgba(0,0,0,0.9)",
black100: "rgba(0,0,0,1)",
red05: "rgba(255,65,54,0.05)",
red10: "rgba(255,65,54,0.1)",
red20: "rgba(255,65,54,0.2)",
red30: "rgba(255,65,54,0.3)",
red40: "rgba(255,65,54,0.4)",
red50: "rgba(255,65,54,0.5)",
red60: "rgba(255,65,54,0.6)",
red70: "rgba(255,65,54,0.7)",
red80: "rgba(255,65,54,0.8)",
red90: "rgba(255,65,54,0.9)",
red100: "rgba(255,65,54,1)",
yellow05: "rgba(255,199,0,0.05)",
yellow10: "rgba(255,199,0,0.1)",
yellow20: "rgba(255,199,0,0.2)",
yellow30: "rgba(255,199,0,0.3)",
yellow40: "rgba(255,199,0,0.4)",
yellow50: "rgba(255,199,0,0.5)",
yellow60: "rgba(255,199,0,0.6)",
yellow70: "rgba(255,199,0,0.7)",
yellow80: "rgba(255,199,0,0.8)",
yellow90: "rgba(255,199,0,0.9)",
yellow100: "rgba(255,199,0,1)",
green05: "rgba(0,159,101,0.05)",
green10: "rgba(0,159,101,0.1)",
green20: "rgba(0,159,101,0.2)",
green30: "rgba(0,159,101,0.3)",
green40: "rgba(0,159,101,0.4)",
green50: "rgba(0,159,101,0.5)",
green60: "rgba(0,159,101,0.6)",
green70: "rgba(0,159,101,0.7)",
green80: "rgba(0,159,101,0.8)",
green90: "rgba(0,159,101,0.9)",
green100: "rgba(0,159,101,1)",
blue05: "rgba(0,142,255,0.05)",
blue10: "rgba(0,142,255,0.1)",
blue20: "rgba(0,142,255,0.2)",
blue30: "rgba(0,142,255,0.3)",
blue40: "rgba(0,142,255,0.4)",
blue50: "rgba(0,142,255,0.5)",
blue60: "rgba(0,142,255,0.6)",
blue70: "rgba(0,142,255,0.7)",
blue80: "rgba(0,142,255,0.8)",
blue90: "rgba(0,142,255,0.9)",
blue100: "rgba(0,142,255,1)",
};
const util = {
cyan: "#00FFFF",
magenta: "#FF00FF",
yellow: "#FFFF00",
black: "#000000",
gray0: "#333333"
};
const theme = {
colors: {
white: util.gray0,
black: base.white,
darkGray: scales.white80,
gray: scales.white60,
lightGray: scales.white30,
washedGray: scales.white05,
red: base.red,
lightRed: scales.red30,
washedRed: scales.red05,
yellow: base.yellow,
lightYellow: scales.yellow30,
washedYellow: scales.yellow10,
green: base.green,
lightGreen: scales.green30,
washedGreen: scales.green10,
blue: base.blue,
lightBlue: scales.blue30,
washedBlue: scales.blue10,
none: "rgba(0,0,0,0)",
scales: scales,
util: util,
},
fonts: {
sans: `"Inter", "Inter UI", -apple-system, BlinkMacSystemFont, 'San Francisco', 'Helvetica Neue', Arial, sans-serif`,
mono: `"Source Code Pro", "Roboto mono", "Courier New", monospace`,
},
// font-size
fontSizes: [
12, // 0
16, // 1
24, // 2
32, // 3
48, // 4
64, // 5
],
// font-weight
fontWeights: {
thin: 300,
regular: 400,
semibold: 500,
bold: 600,
},
// line-height
lineHeights: {
min: 1.2,
short: 1.333333,
regular: 1.5,
tall: 1.666666,
},
// border, border-top, border-right, border-bottom, border-left
borders: ["none", "1px solid"],
// margin, margin-top, margin-right, margin-bottom, margin-left, padding, padding-top, padding-right, padding-bottom, padding-left, grid-gap, grid-column-gap, grid-row-gap
space: [
0, // 0
4, // 1
8, // 2
16, // 3
24, // 4
32, // 5
48, // 6
64, // 7
96, // 8
],
// border-radius
radii: [
0, // 0
2, // 1
4, // 2
8, // 3
],
// width, height, min-width, max-width, min-height, max-height
sizes: [
0, // 0
4, // 1
8, // 2
16, // 3
24, // 4
32, // 5
48, // 6
64, // 7
96, // 8
],
// z-index
zIndices: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
breakpoints: ["550px", "750px", "960px"],
};
export type Theme = typeof theme;
export const styled = baseStyled as ThemedStyledInterface<Theme>;
export default theme;